From 443c73d9af58afc7f104c1bd3a2c8ae9459c7dfe Mon Sep 17 00:00:00 2001 From: fuuzen <80118151@qq.com> Date: Mon, 1 Jul 2024 11:40:18 +0800 Subject: [PATCH] Site updated: 2024-07-01 11:40:16 --- 404.html | 2 +- THANK-YOU/index.html | 2 +- about/index.html | 2 +- acgn/anime-log/index.html | 2 +- acgn/epiao/index.html | 2 +- archives/2023/10/index.html | 2 +- archives/2023/11/index.html | 2 +- archives/2023/12/index.html | 2 +- archives/2023/12/page/2/index.html | 2 +- archives/2023/index.html | 2 +- archives/2023/page/2/index.html | 2 +- archives/2023/page/3/index.html | 2 +- archives/2024/01/index.html | 2 +- archives/2024/01/page/2/index.html | 2 +- archives/2024/01/page/3/index.html | 2 +- archives/2024/02/index.html | 2 +- archives/2024/02/page/2/index.html | 2 +- archives/2024/03/index.html | 2 +- archives/2024/03/page/2/index.html | 2 +- archives/2024/03/page/3/index.html | 2 +- archives/2024/04/index.html | 2 +- archives/2024/04/page/2/index.html | 2 +- archives/2024/05/index.html | 2 +- archives/2024/05/page/2/index.html | 2 +- archives/2024/06/index.html | 2 +- archives/2024/index.html | 2 +- archives/2024/page/10/index.html | 2 +- archives/2024/page/11/index.html | 2 +- archives/2024/page/2/index.html | 2 +- archives/2024/page/3/index.html | 2 +- archives/2024/page/4/index.html | 2 +- archives/2024/page/5/index.html | 2 +- archives/2024/page/6/index.html | 2 +- archives/2024/page/7/index.html | 2 +- archives/2024/page/8/index.html | 2 +- archives/2024/page/9/index.html | 2 +- archives/index.html | 2 +- ...5\344\274\252\344\273\243\347\240\201.png" | Bin 154107 -> 885743 bytes assets/yatcpu/lab1/SingleCycle.html | 2 +- atom.xml | 48 +++++++++--------- categories/ACGN/index.html | 2 +- categories/cs/ai/index.html | 2 +- categories/cs/ctf/index.html | 2 +- .../data-structure-and-algorithm/index.html | 2 +- .../page/2/index.html | 2 +- categories/cs/devops/index.html | 2 +- categories/cs/front-end/index.html | 2 +- categories/cs/index.html | 2 +- categories/cs/languages/index.html | 2 +- categories/cs/os/index.html | 2 +- categories/cs/os/page/2/index.html | 2 +- categories/cs/os/ysosv2/index.html | 2 +- categories/cs/page/2/index.html | 2 +- categories/cs/page/3/index.html | 2 +- categories/cs/page/4/index.html | 2 +- categories/cs/page/5/index.html | 2 +- categories/cs/page/6/index.html | 2 +- categories/cs/page/7/index.html | 2 +- categories/cs/page/8/index.html | 2 +- .../index.html | 2 +- .../page/2/index.html | 2 +- .../yatcpu/index.html | 2 +- categories/cs/raspberrypi/index.html | 2 +- categories/cs/tools/index.html | 2 +- .../index.html" | 2 +- categories/ie/de/index.html | 2 +- categories/ie/index.html | 2 +- categories/ie/page/2/index.html | 2 +- categories/ie/signals-and-systems/index.html | 2 +- categories/index.html | 2 +- categories/math/cryptography/index.html | 2 +- .../math/discrete-math/graph/index.html | 2 +- categories/math/discrete-math/index.html | 2 +- .../math/discrete-math/page/2/index.html | 2 +- .../math/discrete-math/page/3/index.html | 2 +- .../index.html" | 2 +- .../index.html" | 2 +- categories/math/index.html | 2 +- categories/math/page/2/index.html | 2 +- categories/math/page/3/index.html | 2 +- categories/si-zheng/index.html | 2 +- categories/si-zheng/ma-yuan/index.html | 2 +- cs/ai/agent/index.html | 2 +- cs/ai/concepts/index.html | 2 +- cs/ai/dl/index.html | 2 +- .../index.html | 2 +- cs/ai/ml/index.html | 4 +- cs/ai/rl/index.html | 2 +- cs/ai/search/index.html | 6 +-- .../index.html | 2 +- cs/ctf/w4terctf2024/index.html | 4 +- .../b-tree/index.html | 2 +- .../binary-tree/index.html | 2 +- .../critical-path/index.html | 2 +- .../graph/index.html | 2 +- .../hash/index.html | 2 +- .../heap-sort/index.html | 2 +- .../minimum-spanning-tree/index.html | 2 +- .../pre-in-post-index/index.html | 2 +- .../radix-sort/index.html | 2 +- .../shortest-path/index.html | 2 +- .../stack-queue/index.html | 2 +- cs/devops/acme/index.html | 2 +- cs/devops/bash/index.html | 2 +- cs/devops/clone-vm/index.html | 2 +- cs/devops/haproxy-ssh/index.html | 2 +- cs/devops/nat-traversal/index.html | 2 +- cs/devops/nginx/index.html | 2 +- cs/devops/overleaf/index.html | 2 +- cs/devops/pve-passthrough-gpu/index.html | 2 +- cs/devops/pve/index.html | 4 +- cs/devops/sql/index.html | 2 +- cs/fronet-end/hexo/index.html | 4 +- cs/fronet-end/nodejs/index.html | 4 +- cs/languages/js/index.html | 2 +- cs/languages/rust/index.html | 2 +- cs/os/concepts/index.html | 2 +- cs/os/deadlock/index.html | 2 +- cs/os/fs/index.html | 2 +- cs/os/io/index.html | 2 +- cs/os/linux-customized-syscall/index.html | 2 +- cs/os/memory-management/index.html | 2 +- cs/os/process/index.html | 2 +- cs/os/scheduling/index.html | 2 +- cs/os/sync/index.html | 2 +- cs/os/thread/index.html | 2 +- cs/os/ysosv2/lab0/index.html | 2 +- cs/os/ysosv2/lab1/index.html | 2 +- cs/os/ysosv2/lab2/index.html | 2 +- cs/os/ysosv2/lab3/index.html | 2 +- cs/os/ysosv2/lab4/index.html | 2 +- cs/os/ysosv2/lab5/index.html | 2 +- cs/os/ysosv2/lab6/index.html | 2 +- cs/parallel-programing/pthread/index.html | 2 +- .../index.html | 4 +- .../cache/index.html | 2 +- .../cpu/index.html | 2 +- .../data-path/index.html | 2 +- .../dram-refresh/index.html | 2 +- .../exception-and-interruption/index.html | 2 +- .../instruction-system/index.html | 2 +- .../memory/index.html | 2 +- .../operation/index.html | 2 +- .../virtual-memory/index.html | 2 +- .../von-neumann-architecture/index.html | 2 +- .../yatcpu/chisel/index.html | 4 +- .../yatcpu/lab-reports/lab1/index.html | 2 +- .../yatcpu/lab-reports/lab2/index.html | 2 +- .../yatcpu/lab-reports/lab3/index.html | 2 +- .../yatcpu/lab-reports/lab4/index.html | 2 +- .../yatcpu/yatcpu-report/index.html | 2 +- cs/raspberrypi/static-ip/index.html | 2 +- cs/tools/cmake/index.html | 2 +- .../vscode-remote-ssh-login-shell/index.html | 2 +- css/app.css | 2 +- css/comment.css | 2 +- css/mermaid.css | 2 +- feed.json | 28 +++++----- friends/index.html | 2 +- ie/de/bool-algebra/index.html | 2 +- ie/de/coding/index.html | 2 +- ie/de/combinatorial-logic/index.html | 2 +- ie/de/digital-process/index.html | 2 +- ie/de/gate-circuit/index.html | 2 +- ie/de/integrated-circuit/index.html | 2 +- ie/de/memory-storage/index.html | 2 +- ie/de/numerical-system/index.html | 2 +- ie/de/sequential-circuit/index.html | 2 +- ie/de/shift-register/index.html | 2 +- ie/signals-and-systems/CTFT/index.html | 4 +- ie/signals-and-systems/DTFT/index.html | 4 +- ie/signals-and-systems/FS/index.html | 4 +- ie/signals-and-systems/LT/index.html | 4 +- .../communication/index.html | 2 +- ie/signals-and-systems/duality/index.html | 2 +- ie/signals-and-systems/lti/index.html | 2 +- .../signals-concepts/index.html | 4 +- .../systems-concepts/index.html | 2 +- ie/signals-and-systems/zT/index.html | 4 +- index.html | 2 +- js/app.js | 2 +- math/cryptography/lattice-attack/index.html | 4 +- .../algebra/coset-of-subgroup/index.html | 2 +- .../algebra/group-basics/index.html | 2 +- .../algebra/homomorphism/index.html | 2 +- .../algebra/normal-subgroup/index.html | 2 +- .../algebra/permutation-group/index.html | 2 +- .../algebra/prime-max-ideal/index.html | 2 +- math/discrete-math/algebra/ring/index.html | 2 +- .../discrete-math-basics/dmb/index.html | 2 +- .../dmbs/algebra-system/index.html | 2 +- .../dmbs/counting/index.html | 2 +- .../dmbs/difference-equations/index.html | 2 +- .../dmbs/functions/index.html | 2 +- .../index.html | 2 +- .../permutaion-and-combination/index.html | 2 +- .../dmbs/principles-of-counting/index.html | 2 +- .../dmbs/relations/index.html | 2 +- .../discrete-math-basics/problem1/index.html | 2 +- .../discrete-math-basics/set/index.html | 2 +- .../graph/algebra-graph/index.html | 2 +- math/discrete-math/graph/coloring/index.html | 2 +- .../graph/connectivity/index.html | 2 +- .../graph/euler_hamilton/index.html | 2 +- .../graph/graph-basics/index.html | 2 +- math/discrete-math/graph/match/index.html | 2 +- .../graph/planar-graph/index.html | 2 +- .../graph/shortest-path-algorithms/index.html | 2 +- .../graph/spectral-clustering/index.html | 4 +- math/discrete-math/graph/tree/index.html | 2 +- page/10/index.html | 2 +- page/11/index.html | 2 +- page/12/index.html | 2 +- page/13/index.html | 2 +- page/2/index.html | 2 +- page/3/index.html | 2 +- page/4/index.html | 2 +- page/5/index.html | 2 +- page/6/index.html | 2 +- page/7/index.html | 2 +- page/8/index.html | 2 +- page/9/index.html | 2 +- rss.xml | 48 +++++++++--------- si-zheng/ma-yuan/ke-hou-xi-ti/index.html | 2 +- si-zheng/ma-yuan/ti-gang/index.html | 2 +- tags/AI/index.html | 2 +- tags/DevOps/index.html | 2 +- tags/FT/index.html | 2 +- tags/JS/index.html | 2 +- tags/Linux/index.html | 2 +- tags/Node-js/index.html | 2 +- tags/Rust/index.html | 2 +- tags/algebra/index.html | 2 +- tags/algebra/page/2/index.html | 2 +- tags/chisel/index.html | 2 +- tags/combinatorics/index.html | 2 +- tags/coursework/index.html | 2 +- tags/coursework/page/2/index.html | 2 +- tags/cs/index.html | 2 +- tags/cs/page/2/index.html | 2 +- tags/ctf/index.html | 2 +- tags/discrete-math/index.html | 2 +- tags/discrete-math/page/2/index.html | 2 +- tags/discrete-math/page/3/index.html | 2 +- tags/graph/index.html | 2 +- tags/graph/page/2/index.html | 2 +- tags/index.html | 2 +- tags/number/index.html | 2 +- tags/os/index.html | 2 +- tags/os/page/2/index.html | 2 +- .../index.html | 2 +- .../page/2/index.html | 2 +- tags/scala/index.html | 2 +- "tags/\344\277\241\345\217\267/index.html" | 2 +- .../index.html" | 2 +- "tags/\345\211\215\347\253\257/index.html" | 2 +- "tags/\345\215\267\347\247\257/index.html" | 2 +- .../index.html" | 2 +- .../index.html" | 2 +- "tags/\345\271\266\350\241\214/index.html" | 2 +- .../index.html" | 2 +- "tags/\346\200\235\346\224\277/index.html" | 2 +- "tags/\346\220\234\347\264\242/index.html" | 2 +- .../index.html" | 2 +- "tags/\346\225\260\347\224\265/index.html" | 2 +- .../index.html" | 2 +- "tags/\346\240\274/index.html" | 2 +- .../index.html" | 2 +- "tags/\347\216\257/index.html" | 2 +- "tags/\347\256\227\346\263\225/index.html" | 2 +- "tags/\347\263\273\347\273\237/index.html" | 2 +- "tags/\347\276\244\350\256\272/index.html" | 2 +- "tags/\351\200\273\350\276\221/index.html" | 2 +- .../index.html" | 2 +- 274 files changed, 349 insertions(+), 349 deletions(-) diff --git a/404.html b/404.html index 9ace7e04..05708681 100644 --- a/404.html +++ b/404.html @@ -1 +1 @@ -Σ( ° △ °|||)︴404! | Balloon Party = 風船のパーティー = fuusen no party
Σ( ° △ °|||)︴404!
\ No newline at end of file +Σ( ° △ °|||)︴404! | Balloon Party = 風船のパーティー = fuusen no party
Σ( ° △ °|||)︴404!
\ No newline at end of file diff --git a/THANK-YOU/index.html b/THANK-YOU/index.html index c2b38bfc..54db21a0 100644 --- a/THANK-YOU/index.html +++ b/THANK-YOU/index.html @@ -1 +1 @@ -THANK YOU | Balloon Party = 風船のパーティー = fuusen no party

想在这里特别地感谢你!想在这里特别地感谢你!


每一句“生日快乐”每一句“生日快乐”


就算只是一句“生日快乐”就算只是一句“生日快乐”


都让我非常幸福!都让我非常幸福!


特别感谢:特别感谢:

Thank you for your birthday cake!Thank you for your CPU!Thank you for your ナナチ!\text{Thank you for your birthday cake!}\\ \ \\ \text{Thank you for your CPU!}\\ \ \\ \text{Thank you for your ナナチ!}


THANKYOUTHANK\ YOU

ありがとう!ありがとう!


Edited on Views times
\ No newline at end of file +THANK YOU | Balloon Party = 風船のパーティー = fuusen no party

想在这里特别地感谢你!想在这里特别地感谢你!


每一句“生日快乐”每一句“生日快乐”


就算只是一句“生日快乐”就算只是一句“生日快乐”


都让我非常幸福!都让我非常幸福!


特别感谢:特别感谢:

Thank you for your birthday cake!Thank you for your CPU!Thank you for your ナナチ!\text{Thank you for your birthday cake!}\\ \ \\ \text{Thank you for your CPU!}\\ \ \\ \text{Thank you for your ナナチ!}


THANKYOUTHANK\ YOU

ありがとう!ありがとう!


Edited on Views times
\ No newline at end of file diff --git a/about/index.html b/about/index.html index 8f019eab..e94665af 100644 --- a/about/index.html +++ b/about/index.html @@ -1 +1 @@ -About | Balloon Party = 風船のパーティー = fuusen no party

这里是我的博客,整理收纳了一些笔记、思维导图、资料

您可以叫我 fuuzen 或者「風船」,这是我随便起的一个用的比较久的网名,没有什么特别含义。fuusen 即「ふうせん」,但这个 ID 已经在 github 被注册了,所以只好叫做 fuuzen 了

这个博客网站的名字是 Balloon Party 或者「風船のパーティー」,也是我随便起的一个名字,没有什么特别含义

如果这里所整理收纳的资料对您的学习有帮助的话,欢迎您向您的朋友们推荐本博客!也欢迎您向我指出资料中的任何错误或提出宝贵的意见!

picture

Edited on Views times
\ No newline at end of file +About | Balloon Party = 風船のパーティー = fuusen no party

这里是我的博客,整理收纳了一些笔记、思维导图、资料

您可以叫我 fuuzen 或者「風船」,这是我随便起的一个用的比较久的网名,没有什么特别含义。fuusen 即「ふうせん」,但这个 ID 已经在 github 被注册了,所以只好叫做 fuuzen 了

这个博客网站的名字是 Balloon Party 或者「風船のパーティー」,也是我随便起的一个名字,没有什么特别含义

如果这里所整理收纳的资料对您的学习有帮助的话,欢迎您向您的朋友们推荐本博客!也欢迎您向我指出资料中的任何错误或提出宝贵的意见!

picture

Edited on Views times
\ No newline at end of file diff --git a/acgn/anime-log/index.html b/acgn/anime-log/index.html index 7aeb7d3b..10742bc3 100644 --- a/acgn/anime-log/index.html +++ b/acgn/anime-log/index.html @@ -1 +1 @@ -| Balloon Party = 風船のパーティー = fuusen no party

# 2023 看番年度总结

* 星号是没完全看完

# 寒假:

  1. EVA 旧版
  2. 凉宫春日
  3. Angel Beats!
  4. Little Busters
  5. Air
  6. 86 - 不存在的战区
  7. ISLAND
  8. 斩赤红之瞳
  9. 可塑性记忆
  10. 虚构推理
  11. 冰海战记
  12. 魔圆
  13. 魔纪
  14. 想要成为影之实力者

# 一些短的:

  1. 快要坏掉的八音盒
  2. 星之梦 + 星之人

# 大一下:

  1. Canon *
  2. 时光沙漏
  3. 安达与岛村
  4. 终将成为你
  5. 天国大魔境

# 暑假:

  1. 处刑少女的生存手册
  2. Happy Sugar Life

# 大二上:

  1. mygo *
  2. 黑之契约者 *
  3. Fate Zero
Edited on Views times
\ No newline at end of file +| Balloon Party = 風船のパーティー = fuusen no party

# 2023 看番年度总结

* 星号是没完全看完

# 寒假:

  1. EVA 旧版
  2. 凉宫春日
  3. Angel Beats!
  4. Little Busters
  5. Air
  6. 86 - 不存在的战区
  7. ISLAND
  8. 斩赤红之瞳
  9. 可塑性记忆
  10. 虚构推理
  11. 冰海战记
  12. 魔圆
  13. 魔纪
  14. 想要成为影之实力者

# 一些短的:

  1. 快要坏掉的八音盒
  2. 星之梦 + 星之人

# 大一下:

  1. Canon *
  2. 时光沙漏
  3. 安达与岛村
  4. 终将成为你
  5. 天国大魔境

# 暑假:

  1. 处刑少女的生存手册
  2. Happy Sugar Life

# 大二上:

  1. mygo *
  2. 黑之契约者 *
  3. Fate Zero
Edited on Views times
\ No newline at end of file diff --git a/acgn/epiao/index.html b/acgn/epiao/index.html index 923d1579..3ee0b1ce 100644 --- a/acgn/epiao/index.html +++ b/acgn/epiao/index.html @@ -1 +1 @@ -《饿殍:明末千里行》通关记录 - ACGN | Balloon Party = 風船のパーティー = fuusen no party

今天因为校青马班的活动又跑到南校开会、参观团一大。中涂我也不能用电脑,于是无聊之际,手机打开了一直感兴趣的《饿殍:明末千里行》的游戏录播,看了前面就上头了。于是一直到凌晨 3 点,打完了 2 个 TE ,非常有感触,于是一边听着 ED 写下本篇文章作记录。(即使第二天有早八我也不管了!)

这是我玩的第二篇国产的国产文字冒险游戏。(上一部是《OPUS: 龙脉长歌》,这两部观感都非常好)《饿殍》难怪这么火,是有些原因的:

  • 首先作为一个全年龄向的游戏,受众确实更加广泛
  • 人物立绘、CG 也是蛮出色的
  • 而且描写的饥荒惨状,敢于如此露骨地描写、刻画,也是非常值得称赞的。有些东西,一定要最露骨的刻画,才能真正描绘出来,传达出来
  • 当然也有一些比较出圈的 “名场面”,懂得都懂了
  • 音乐方面并不十分特别,但也不错

我尤其比较喜欢的,还得是双线叙事处理穗的回忆了。穗的回忆可以说是最压抑的部分,在回忆的开头,展现了一点穗原本家庭的美好画面,但后面便开始逐渐步入绝望,令人揪心。尤其是《绝望》这一篇,最为压抑,也最为强而有力,直击玩家的心扉!

这部作品的描绘,很有 “中国” 的历史特色,最后也展现了 “反叛” 这样的末代王朝常见的主旋律。与日系作品相比,少了一分细腻,但却传达了与 “物哀” 一样的悲剧的美。

要说从这作品思考到了什么,倒也没有什么。对我来说,我想也是对于大部分和我一样的观众来说,这样的作品给我们的,更多的是摆脱平常朝九晚五、三点一线生活的一点调味。相比平静的青春校园主题作品,这样的乱世、在乱世中旅行、在乱世中寻仇的题材,更给人刺激感、新鲜感。

1

穗这个角色,塑造的相当好。她给人一种 “小大人” 的感觉,但是这些成熟、懂事、机灵等等不符合其年龄的特点,都反衬着她的艰辛遭遇,都令人心酸。在穗的回忆中,她反而是没有那么 “懂事” 的,还会因为父母煮了小猫而赌气。这样的前后对照,也是对她悲惨遭遇的深层次的刻画,饥荒的灾难不仅摧残她的身体,更塑造了她的性格,催生了她的成长。

而一路上,玩家要想达到好结局,必须要做的只有一件事:尽量在穗面前表现得善良一些,增加所谓的 “好感度”。这也是和传统的文字冒险游戏选项分支不太一样,从这里实际上也看出了游戏的一大主题:原谅。玩家的最终目标,或者说主人公良的终极目标,就是获取穗的原谅。不得不说这还是蛮反差的,因为穗的回忆中,淋漓尽致地描写了良杀害了穗爹爹后,穗家庭的更加惨不忍睹的悲剧,而玩家在见识到这一切之后,还要去替这个 “罪魁祸首” 寻求原谅。

确实,这也是很多人会排斥、难以接受的地方。把穗一家害得如此惨的弑父仇敌,凭什么要原谅?

当然,游戏制作者还是 “上价值”,提醒了玩家们,罪魁祸首不是良,不是男主,而是荒淫无道的王爷、朝廷官府,这也是为什么这个游戏这么有 “中国” 的历史特色,原来一切都是人压迫人导致的悲剧。

2

整个作品描绘的是一个悲剧、一幅惨状、一对命运不幸的主人公。

穗的人生,自从家里只剩她自己之后,就总是在围绕着 “为了复仇而活着” 了。

良的人生,自从父亲去世只剩他一个人之后,直到遇到穗之前,总是围绕着 “为自己而活着”。

两个角色也就是这样,有一种恰到好处的 “距离感”,看似合作伙伴,实则杀父仇人,在相互帮助、依赖的过程中,产生了一点点的、非常少的 “为对方而活着” 的感觉。而尤其是到了 TE “同生” ,“为对方而活着” 的感觉就更强烈了。

贯穿整个故事的,一直都是两人这样的 “距离感”。在 TE 之前的 3 个结局,突出表现了距离感与陌生感:

  • 无言,穗实施了计划让良在茫然、错愕中被捕、处以极刑,可以说是最不好的结局之一
  • 快逃,和无言的区别在于穗给了良准备的机会,给了良体面赴死的机会
  • 不见,最让人放不下的结局
    • 穗的想法来看,可以有两种解读:
      • 可以认为穗似乎原谅良了,不再杀良,宁可自己一人消失
      • 可以认为穗依然复仇了,只是她的复仇,不再是设计良被捕,而是用自己的消失、荷包的提醒,告诉良真相之后一言不发地独自消失
    • 穗的去向来看,也有两种解读:
      • 很大可能,穗投湖自尽了,在失去 “复仇” 这一人生意义后,只好自杀
      • 很小的概率,穗或许伪造了投湖的迹象,依然活着,并且选择不再出现在良眼前

那张惊艳无比的烟花的 CG ,竟是良和穗的最后一面。而后,穗谎称解手,便一去不返,突然没了音讯。两人再次拉开了距离,突然一股陌生感、寂寞感向玩家袭来。

至此,穗这个角色的塑造堪称饱满,她不再是弱小无力的” 小羊 “,而是和良站在平等的位置上的,甚至站在审判良的位置上的,一个饱满的角色。并且,一直保持了一份 “距离感”,即穗与良的 “距离感”,也是玩家与” 穗 “这个角色的” 距离感 “,是不再相见的 “距离感”、是弑父血仇的 “距离感”、是生与死的” 距离感 “。

直到最后的结局 “同生” 也是,一直保留了这份若即若离的距离感,两人还是以仇人相称。这份 “距离感”,比皆大欢喜、甜甜蜜蜜发糖可要好太多了,这也是我非常喜欢的结局。

片段一 (同生)
良 [喂,我算了一下,如今你大概二十有三了吧。]
满穗 [嗯。]
良 [现在嫁人了没?]
满穗 [没嫁呢。]
良 [哦。]
满穗 [良呢?打仗的候娶妻了吗?]
良 [没娶,兵荒乱的,娶什么妻。]
满穗 [也是。]
片段二 (同生)
满穗 [其实,这么多年来,我跟别人提起我的名字,一直都是 “穗”。]
满穗 [唯独良爷,我说了全名,是叫满穗。]
良 [......]
良 [哦?为何只有我有这个区别?]
满穗 [呵...... 当初我就是觉得,要仇人知道我的全名,我才没有白复仇。]
满穗 [如今怎么想的,我也不知道了......]
良 [......]
满穗 [......]
良 [......]
满穗 [话说回来,良爷觉得怎么样?]
良 [什么怎么样?]
满穗 [你喜欢 “满穗”,还是 “穗”?]
良 [......]
满穗 [......]
良 [......]
良 [“满穗” 吧]
满穗 [......]
良 [我喜欢 “满穗”......]
满穗 [哦......]
良 [满穗、满穗、满穗......]
我念了一遍又一遍她的名字。
满穗 [......]
她没有说话,只是装著听不见我在念她的名字
她特不时地整笑著,不知在高兴些什么
她望著窗外,饮著茶,似乎在想著一些什么......
—— 咕
我也饮下了一杯茶,望著她所眺望的方向,嘴角不自地勾起了一丝笑意。
片段三 (共死)
满穗 [以前... 我装成哑巴... 现在... 你真成哑巴了... 真好笑...]
她轻声笑着
满穗 [良,抱紧我吧......]
Edited on Views times
\ No newline at end of file +《饿殍:明末千里行》通关记录 - ACGN | Balloon Party = 風船のパーティー = fuusen no party

今天因为校青马班的活动又跑到南校开会、参观团一大。中涂我也不能用电脑,于是无聊之际,手机打开了一直感兴趣的《饿殍:明末千里行》的游戏录播,看了前面就上头了。于是一直到凌晨 3 点,打完了 2 个 TE ,非常有感触,于是一边听着 ED 写下本篇文章作记录。(即使第二天有早八我也不管了!)

这是我玩的第二篇国产的国产文字冒险游戏。(上一部是《OPUS: 龙脉长歌》,这两部观感都非常好)《饿殍》难怪这么火,是有些原因的:

  • 首先作为一个全年龄向的游戏,受众确实更加广泛
  • 人物立绘、CG 也是蛮出色的
  • 而且描写的饥荒惨状,敢于如此露骨地描写、刻画,也是非常值得称赞的。有些东西,一定要最露骨的刻画,才能真正描绘出来,传达出来
  • 当然也有一些比较出圈的 “名场面”,懂得都懂了
  • 音乐方面并不十分特别,但也不错

我尤其比较喜欢的,还得是双线叙事处理穗的回忆了。穗的回忆可以说是最压抑的部分,在回忆的开头,展现了一点穗原本家庭的美好画面,但后面便开始逐渐步入绝望,令人揪心。尤其是《绝望》这一篇,最为压抑,也最为强而有力,直击玩家的心扉!

这部作品的描绘,很有 “中国” 的历史特色,最后也展现了 “反叛” 这样的末代王朝常见的主旋律。与日系作品相比,少了一分细腻,但却传达了与 “物哀” 一样的悲剧的美。

要说从这作品思考到了什么,倒也没有什么。对我来说,我想也是对于大部分和我一样的观众来说,这样的作品给我们的,更多的是摆脱平常朝九晚五、三点一线生活的一点调味。相比平静的青春校园主题作品,这样的乱世、在乱世中旅行、在乱世中寻仇的题材,更给人刺激感、新鲜感。

1

穗这个角色,塑造的相当好。她给人一种 “小大人” 的感觉,但是这些成熟、懂事、机灵等等不符合其年龄的特点,都反衬着她的艰辛遭遇,都令人心酸。在穗的回忆中,她反而是没有那么 “懂事” 的,还会因为父母煮了小猫而赌气。这样的前后对照,也是对她悲惨遭遇的深层次的刻画,饥荒的灾难不仅摧残她的身体,更塑造了她的性格,催生了她的成长。

而一路上,玩家要想达到好结局,必须要做的只有一件事:尽量在穗面前表现得善良一些,增加所谓的 “好感度”。这也是和传统的文字冒险游戏选项分支不太一样,从这里实际上也看出了游戏的一大主题:原谅。玩家的最终目标,或者说主人公良的终极目标,就是获取穗的原谅。不得不说这还是蛮反差的,因为穗的回忆中,淋漓尽致地描写了良杀害了穗爹爹后,穗家庭的更加惨不忍睹的悲剧,而玩家在见识到这一切之后,还要去替这个 “罪魁祸首” 寻求原谅。

确实,这也是很多人会排斥、难以接受的地方。把穗一家害得如此惨的弑父仇敌,凭什么要原谅?

当然,游戏制作者还是 “上价值”,提醒了玩家们,罪魁祸首不是良,不是男主,而是荒淫无道的王爷、朝廷官府,这也是为什么这个游戏这么有 “中国” 的历史特色,原来一切都是人压迫人导致的悲剧。

2

整个作品描绘的是一个悲剧、一幅惨状、一对命运不幸的主人公。

穗的人生,自从家里只剩她自己之后,就总是在围绕着 “为了复仇而活着” 了。

良的人生,自从父亲去世只剩他一个人之后,直到遇到穗之前,总是围绕着 “为自己而活着”。

两个角色也就是这样,有一种恰到好处的 “距离感”,看似合作伙伴,实则杀父仇人,在相互帮助、依赖的过程中,产生了一点点的、非常少的 “为对方而活着” 的感觉。而尤其是到了 TE “同生” ,“为对方而活着” 的感觉就更强烈了。

贯穿整个故事的,一直都是两人这样的 “距离感”。在 TE 之前的 3 个结局,突出表现了距离感与陌生感:

  • 无言,穗实施了计划让良在茫然、错愕中被捕、处以极刑,可以说是最不好的结局之一
  • 快逃,和无言的区别在于穗给了良准备的机会,给了良体面赴死的机会
  • 不见,最让人放不下的结局
    • 穗的想法来看,可以有两种解读:
      • 可以认为穗似乎原谅良了,不再杀良,宁可自己一人消失
      • 可以认为穗依然复仇了,只是她的复仇,不再是设计良被捕,而是用自己的消失、荷包的提醒,告诉良真相之后一言不发地独自消失
    • 穗的去向来看,也有两种解读:
      • 很大可能,穗投湖自尽了,在失去 “复仇” 这一人生意义后,只好自杀
      • 很小的概率,穗或许伪造了投湖的迹象,依然活着,并且选择不再出现在良眼前

那张惊艳无比的烟花的 CG ,竟是良和穗的最后一面。而后,穗谎称解手,便一去不返,突然没了音讯。两人再次拉开了距离,突然一股陌生感、寂寞感向玩家袭来。

至此,穗这个角色的塑造堪称饱满,她不再是弱小无力的” 小羊 “,而是和良站在平等的位置上的,甚至站在审判良的位置上的,一个饱满的角色。并且,一直保持了一份 “距离感”,即穗与良的 “距离感”,也是玩家与” 穗 “这个角色的” 距离感 “,是不再相见的 “距离感”、是弑父血仇的 “距离感”、是生与死的” 距离感 “。

直到最后的结局 “同生” 也是,一直保留了这份若即若离的距离感,两人还是以仇人相称。这份 “距离感”,比皆大欢喜、甜甜蜜蜜发糖可要好太多了,这也是我非常喜欢的结局。

片段一 (同生)
良 [喂,我算了一下,如今你大概二十有三了吧。]
满穗 [嗯。]
良 [现在嫁人了没?]
满穗 [没嫁呢。]
良 [哦。]
满穗 [良呢?打仗的候娶妻了吗?]
良 [没娶,兵荒乱的,娶什么妻。]
满穗 [也是。]
片段二 (同生)
满穗 [其实,这么多年来,我跟别人提起我的名字,一直都是 “穗”。]
满穗 [唯独良爷,我说了全名,是叫满穗。]
良 [......]
良 [哦?为何只有我有这个区别?]
满穗 [呵...... 当初我就是觉得,要仇人知道我的全名,我才没有白复仇。]
满穗 [如今怎么想的,我也不知道了......]
良 [......]
满穗 [......]
良 [......]
满穗 [话说回来,良爷觉得怎么样?]
良 [什么怎么样?]
满穗 [你喜欢 “满穗”,还是 “穗”?]
良 [......]
满穗 [......]
良 [......]
良 [“满穗” 吧]
满穗 [......]
良 [我喜欢 “满穗”......]
满穗 [哦......]
良 [满穗、满穗、满穗......]
我念了一遍又一遍她的名字。
满穗 [......]
她没有说话,只是装著听不见我在念她的名字
她特不时地整笑著,不知在高兴些什么
她望著窗外,饮著茶,似乎在想著一些什么......
—— 咕
我也饮下了一杯茶,望著她所眺望的方向,嘴角不自地勾起了一丝笑意。
片段三 (共死)
满穗 [以前... 我装成哑巴... 现在... 你真成哑巴了... 真好笑...]
她轻声笑着
满穗 [良,抱紧我吧......]
Edited on Views times
\ No newline at end of file diff --git a/archives/2023/10/index.html b/archives/2023/10/index.html index 9b350fa5..ade17171 100644 --- a/archives/2023/10/index.html +++ b/archives/2023/10/index.html @@ -1 +1 @@ -2023 Y / 10 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +2023 Y / 10 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/archives/2023/11/index.html b/archives/2023/11/index.html index 56b6589f..19850121 100644 --- a/archives/2023/11/index.html +++ b/archives/2023/11/index.html @@ -1 +1 @@ -2023 Y / 11 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +2023 Y / 11 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/archives/2023/12/index.html b/archives/2023/12/index.html index e2bb3fbe..fa7a5546 100644 --- a/archives/2023/12/index.html +++ b/archives/2023/12/index.html @@ -1 +1 @@ -2023 Y / 12 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +2023 Y / 12 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/archives/2023/12/page/2/index.html b/archives/2023/12/page/2/index.html index f000f004..1e56a5c4 100644 --- a/archives/2023/12/page/2/index.html +++ b/archives/2023/12/page/2/index.html @@ -1 +1 @@ -2023 Y / 12 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +2023 Y / 12 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/archives/2023/index.html b/archives/2023/index.html index 2cddd80c..877a2d96 100644 --- a/archives/2023/index.html +++ b/archives/2023/index.html @@ -1 +1 @@ -2023 Y - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +2023 Y - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/archives/2023/page/2/index.html b/archives/2023/page/2/index.html index 4fd45315..f357ee99 100644 --- a/archives/2023/page/2/index.html +++ b/archives/2023/page/2/index.html @@ -1 +1 @@ -2023 Y - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +2023 Y - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/archives/2023/page/3/index.html b/archives/2023/page/3/index.html index d3f4e2b5..3d051c85 100644 --- a/archives/2023/page/3/index.html +++ b/archives/2023/page/3/index.html @@ -1 +1 @@ -2023 Y - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +2023 Y - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/archives/2024/01/index.html b/archives/2024/01/index.html index bfce1902..b6c53940 100644 --- a/archives/2024/01/index.html +++ b/archives/2024/01/index.html @@ -1 +1 @@ -2024 Y / 01 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +2024 Y / 01 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/archives/2024/01/page/2/index.html b/archives/2024/01/page/2/index.html index 105dbc51..d5338cb1 100644 --- a/archives/2024/01/page/2/index.html +++ b/archives/2024/01/page/2/index.html @@ -1 +1 @@ -2024 Y / 01 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +2024 Y / 01 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/archives/2024/01/page/3/index.html b/archives/2024/01/page/3/index.html index 20335ca5..c3785525 100644 --- a/archives/2024/01/page/3/index.html +++ b/archives/2024/01/page/3/index.html @@ -1 +1 @@ -2024 Y / 01 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +2024 Y / 01 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/archives/2024/02/index.html b/archives/2024/02/index.html index 871123b2..ba06e5e1 100644 --- a/archives/2024/02/index.html +++ b/archives/2024/02/index.html @@ -1 +1 @@ -2024 Y / 02 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +2024 Y / 02 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/archives/2024/02/page/2/index.html b/archives/2024/02/page/2/index.html index 86afc7b5..a874fc9b 100644 --- a/archives/2024/02/page/2/index.html +++ b/archives/2024/02/page/2/index.html @@ -1 +1 @@ -2024 Y / 02 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +2024 Y / 02 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/archives/2024/03/index.html b/archives/2024/03/index.html index 1bab27ae..d80b5df8 100644 --- a/archives/2024/03/index.html +++ b/archives/2024/03/index.html @@ -1 +1 @@ -2024 Y / 03 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +2024 Y / 03 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/archives/2024/03/page/2/index.html b/archives/2024/03/page/2/index.html index 0a4e7988..d6e58caa 100644 --- a/archives/2024/03/page/2/index.html +++ b/archives/2024/03/page/2/index.html @@ -1 +1 @@ -2024 Y / 03 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +2024 Y / 03 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/archives/2024/03/page/3/index.html b/archives/2024/03/page/3/index.html index 78ff807e..d8be829a 100644 --- a/archives/2024/03/page/3/index.html +++ b/archives/2024/03/page/3/index.html @@ -1 +1 @@ -2024 Y / 03 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +2024 Y / 03 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/archives/2024/04/index.html b/archives/2024/04/index.html index ccbe1c64..d8da5f3e 100644 --- a/archives/2024/04/index.html +++ b/archives/2024/04/index.html @@ -1 +1 @@ -2024 Y / 04 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +2024 Y / 04 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/archives/2024/04/page/2/index.html b/archives/2024/04/page/2/index.html index 28eee4aa..4ef9dc43 100644 --- a/archives/2024/04/page/2/index.html +++ b/archives/2024/04/page/2/index.html @@ -1 +1 @@ -2024 Y / 04 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +2024 Y / 04 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/archives/2024/05/index.html b/archives/2024/05/index.html index 97cd4177..e0d3a4f6 100644 --- a/archives/2024/05/index.html +++ b/archives/2024/05/index.html @@ -1 +1 @@ -2024 Y / 05 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +2024 Y / 05 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/archives/2024/05/page/2/index.html b/archives/2024/05/page/2/index.html index 5a00213c..291ee424 100644 --- a/archives/2024/05/page/2/index.html +++ b/archives/2024/05/page/2/index.html @@ -1 +1 @@ -2024 Y / 05 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +2024 Y / 05 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/archives/2024/06/index.html b/archives/2024/06/index.html index 500328d2..d55fddf1 100644 --- a/archives/2024/06/index.html +++ b/archives/2024/06/index.html @@ -1 +1 @@ -2024 Y / 06 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +2024 Y / 06 M - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/archives/2024/index.html b/archives/2024/index.html index ae8c202d..16a117fa 100644 --- a/archives/2024/index.html +++ b/archives/2024/index.html @@ -1 +1 @@ -2024 Y - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +2024 Y - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/archives/2024/page/10/index.html b/archives/2024/page/10/index.html index 6b084405..cdbff36f 100644 --- a/archives/2024/page/10/index.html +++ b/archives/2024/page/10/index.html @@ -1 +1 @@ -2024 Y - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +2024 Y - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/archives/2024/page/11/index.html b/archives/2024/page/11/index.html index aadb7450..fb7f3a6e 100644 --- a/archives/2024/page/11/index.html +++ b/archives/2024/page/11/index.html @@ -1 +1 @@ -2024 Y - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +2024 Y - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/archives/2024/page/2/index.html b/archives/2024/page/2/index.html index 92c4fe7a..b8000831 100644 --- a/archives/2024/page/2/index.html +++ b/archives/2024/page/2/index.html @@ -1 +1 @@ -2024 Y - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +2024 Y - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/archives/2024/page/3/index.html b/archives/2024/page/3/index.html index 275b5038..9848fbcf 100644 --- a/archives/2024/page/3/index.html +++ b/archives/2024/page/3/index.html @@ -1 +1 @@ -2024 Y - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +2024 Y - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/archives/2024/page/4/index.html b/archives/2024/page/4/index.html index 17467dc7..12052499 100644 --- a/archives/2024/page/4/index.html +++ b/archives/2024/page/4/index.html @@ -1 +1 @@ -2024 Y - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +2024 Y - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/archives/2024/page/5/index.html b/archives/2024/page/5/index.html index dcdc309d..11035c7f 100644 --- a/archives/2024/page/5/index.html +++ b/archives/2024/page/5/index.html @@ -1 +1 @@ -2024 Y - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +2024 Y - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/archives/2024/page/6/index.html b/archives/2024/page/6/index.html index db6097d5..63cbda53 100644 --- a/archives/2024/page/6/index.html +++ b/archives/2024/page/6/index.html @@ -1 +1 @@ -2024 Y - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +2024 Y - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/archives/2024/page/7/index.html b/archives/2024/page/7/index.html index c445e87b..b13a8ccd 100644 --- a/archives/2024/page/7/index.html +++ b/archives/2024/page/7/index.html @@ -1 +1 @@ -2024 Y - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +2024 Y - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/archives/2024/page/8/index.html b/archives/2024/page/8/index.html index c93de5e3..692d711c 100644 --- a/archives/2024/page/8/index.html +++ b/archives/2024/page/8/index.html @@ -1 +1 @@ -2024 Y - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +2024 Y - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/archives/2024/page/9/index.html b/archives/2024/page/9/index.html index e2182d5e..15b75c2b 100644 --- a/archives/2024/page/9/index.html +++ b/archives/2024/page/9/index.html @@ -1 +1 @@ -2024 Y - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +2024 Y - Archive | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/archives/index.html b/archives/index.html index 40bc94ed..e8e8df78 100644 --- a/archives/index.html +++ b/archives/index.html @@ -1 +1 @@ -Archive | Balloon Party = 風船のパーティー = fuusen no party

Home / 127 posts in total. Good Keep on posting.

2024 Y/06 M ( 6 )

2024 Y/05 M ( 17 )

2024 Y/04 M ( 18 )

2024 Y/03 M ( 28 )

2024 Y/02 M ( 14 )

2024 Y/01 M ( 21 )

2023 Y/12 M ( 17 )

2023 Y/11 M ( 3 )

2023 Y/10 M ( 3 )

\ No newline at end of file +Archive | Balloon Party = 風船のパーティー = fuusen no party

Home / 127 posts in total. Good Keep on posting.

2024 Y/06 M ( 6 )

2024 Y/05 M ( 17 )

2024 Y/04 M ( 18 )

2024 Y/03 M ( 28 )

2024 Y/02 M ( 14 )

2024 Y/01 M ( 21 )

2023 Y/12 M ( 17 )

2023 Y/11 M ( 3 )

2023 Y/10 M ( 3 )

\ No newline at end of file diff --git "a/assets/cs/ai/search/\346\234\200\345\245\275\344\274\230\345\205\210\346\220\234\347\264\242\347\256\227\346\263\225\344\274\252\344\273\243\347\240\201.png" "b/assets/cs/ai/search/\346\234\200\345\245\275\344\274\230\345\205\210\346\220\234\347\264\242\347\256\227\346\263\225\344\274\252\344\273\243\347\240\201.png" index f8c09b5dd419989f54ea176e82fcd00fb9fab0af..eaa752a38a0469f18483c35d8823dc1cc9f5a6ec 100644 GIT binary patch literal 885743 zcmaHS1y~)s)+iK*;>BHxYjM{i#bx8}ZX4I)&Mu|c#*4eVI}~?!cXxZ7d+)pd{oXtO z&6iBpBug?$CYh|YCR|Zo5*ZO65ds1NSz1b583FNQK9%!)d4tVyA2W{2^=s37r$P3JpmMnG-ZFdNv{<^&tquSX8kx zI}))>m%%_x{p+U^c0MC>^kbjO%*>KVnC43=gv_J1%hoF&;F;f8>cdzv(EM~6>ieQ} zjMT5lBB&qK!#Mr0Pg8L!$_F`i-ytZ3A#V*RL+R03v$MM)e`P(b?W`bMAnz2lMjd6m zyaCwVy%BIADDgL)(}PmrF32Fx5zC}fQ6ZG_xKhB(KMm)+|2=A z04ZmWLvZF4^uw`xm4b-_K#={eEP^W!cC1fmU;b75SB?; zpb^}O%q6ARRd1aM!hCGty|AyEcNn>l22vsgh-4JTdJicXL5$y9+Ub5=$Nl-caPM-Z zT^kFT-H*LMnl&LcR_y#Rr);-A>6+Q!U&!<%p&k;ORxcl7^U2|f@kaFvatp6 z+6jUJ0#T4;&{2?l*NJoh`Gta##CQ1t685CXYZ9Qpwy3=NGRa-Y@ejPY&KrM$(+@8! zj{5rxSXP>Wjzd+VZ{bh&Z1)?s9~WbZ25CH?7eF>IS5g3Nz?>nrW|`kGhg&aW zCnkju7h2lvS<{0b#JZ1>fx-HKo9;dW#BFo;$=bcH`N((4&@T{s1dYm${-U8nOwjl7 zih+D`sOkD<9r&aDcny%u>u?`LXu>{Rkiy6YO_+DOFu|&H%Cew-3RJEjEP*8GFsnd5 z{@~RKsfZdApt8=;0KM4h;($|@&UF6a+TU%REr66VC}JEaq%V1a$p*K$P{xKz8gvQ% z;gt&-2x=h!BMuEr?=kgC`4XI`Fib~$`Tmm)M3hcU2PQ$7jA-UguJz_~-WHhl2%(;* zb9-pI&hG}2=adx)gF&)gs-mn}Q5As>Ne-~5xlCPr zwhbN9E`ev#dVVlmfK^wLDkFlPaxpUdjB&zSX zqve(5lBKzYu9i3>J!73(om%0nUMXj(psk!O&WZSm<>Jm_e+{1t3y};_oqXBomoz9| zE@#1L7Q#-#PLfJ@@sTikiq)f->6@vYqlEq0L%q3|9LYJP>7fbjEcJ3XeUiEQ8PG&V zQM>e8l<%2TX{;VaBZZ6vF|J2H1V;r&7)O#}#YpB%=GyoG%ev!Y=%Iv!IlGzK>m@@!L^%|kaFej2)}&@rELv<; z>@}-ut$##28$UZ|6UO;5g)k+W^Q5kzwAePmw(-Q2Z+dh@-k^S>%=nYxuYQ`1f&1oH zJjruh5Yhx=KlK0yV#6mk}>PSGzb<* z(;sEwm(lJ?2iR%Y3U~{c=Q-RZ&Ii#%r8EX2ex%rlaVF~ao+7!RreQ5&JH2DKC>7;^@rpIYs6n*ySgR`PT*Kd*<{W$ve!Pa@yM=x8}uT_mmE z8af=U`&^>ki1hhi`=E;vrl@q13WQkQHY=sA;<)2b^3W%LPP*p%Y=}G3@w^-yUq@UE zGDS8gJO%9aYmXdlcwMPqW|+F0mkjTZUXSX;FU0Mn01{^C4G=it^-M!e>_>ewFypJ) z%t^Uu>p2x@nHkHK3TTJX>ts2>{I*V^1M;d2hpC%9rX#-u9h2Ncp<`u#uy0#;l02vt z)?VWrbbrVQxC}s5!OSO4&1040E_G=-x$d3Hz=Fkk<%G)UOq<}JGTCiOMm1+Ow@r&4 zmd{w>@3$EZ;L!yJ?wx`qlM~s_?#TOp$A7ioqaLx>GH5V34d`!&epA*h(PCg~YaF-U z*!_A>-ub((*BHB8|G6|s-KlcAvZmyE2;=l;WIOLm)9IXlE}qO%LL=i+O=^Qq!*d_v z8dkHKRSAR6gFB$7sA=YU=5z5{S#r&;XtK=0Ztbty$=bA|t~rhpF8y}zmqcVug0jS? z#5pxwRfaP6_SpK1KUD2<94W|LiFU6wUn_QIXV*(z;uGRcm%^41>jYNP8(x;}qt=RC z_s?inE?n{*%=gOXbr}Jt06;UcD|4|#aqf(OSKpcC8bL*UR$rT$b*a4VfnbF~)7iso z%?lSw5Rrx80JjEkU{kTzaB<@&Z;qX(uD*|&kGa@4F-c>QsR<#K}y@OflZyG47Y9%$d*q~fr5wbFP8fA{UQc#YZ1;5_pZHATpZZ-o~m@X>w# zk>+$_rGL4fIbAB<`#|gb*JI6b@j_+XB2z z;#VXA1jqTCKpxKP$}+UUnyZGdh;Qu$gt!W1*d4@v5^J|@fm+;L{1>t=UIUrox$73^ zb9-?m;49kQm>a3F3#8E68;sL|x39%izta1WBEnQd+DuLkg6G_rSb z79c18OVGc=Kig^QZuviw?12AztoH|G{;P(Wm5GJ<-@M;R`Tr{ARkU^xui|3UEoRsE01{~*=)Pf|Aa|0Mm7s{fZ105o+Hv$uU0>MZy_1NJZC|E&BM zAwTn9&;B2M@z0L_N9lV{3nKC}|9jK~5xsex@!kiL&{AAM^*w)|W`Bhbhwm4ff9CHn zOr|++f5iy|gfN7(xQMDdlY&3Q&8{t)J(He^R3B5bo@uh-=vu9n|BEM!s{gXu|?-?3gLivV4UgIGUt*nK1PEF*rsja@ou1`Xo-0L?7;VK((Kfc*3X6-wMB&`kNWn|ZC!D$ogP=rAMNf_(`xcO zI8oWl_LuMdL;y#9Y&-1m#qM{_;4eKDf)CHSEGj%JJ4b3?2VYNYJdeaU_pqz#JA*~1 zEL$#SZ*+K=;K!!MGT$!SYpz{6r&d+RVvBgeyiEtw*F?5){&;2yv-8co#>;9l8UCz9 zoPzg@DMr?I_xg5se!#n?Creb9!AR+an$&tO)3^-V@4Dx>d9Al!=3uqZ4ezvXAM9_V zkG219bk^6?BWuMYo+E${m>&c#PdKcUY@Esn~gXv`F3Oed?Y$%td0r9a)oBl8Q_4#V1Jt(h2_!vn@Q2I3^R(!?pADUO=aJ2 zcX?~~navg{~-(R;`SFNeTT{QIgigG zSB~bk_HY8VmXzgd10m6t+uR_->7WT&63iG-EqWfE_-wqZ2!MHXwKY!Qh%2sv-#=$` z;$A-KFUkS^>wJ7i-8h-SKk5$#>db>HhB~DZObU40Vo8a$h~TGv8tI+$wviIO>YpU_ z+lod!xO>+68zVrh02qNp&22Ypy z1kbth{jgyVd3yr#la5E44aI?i+O9-i6@3qFm+w{*b zdMO=CQfRoDn42n<2sq1jU7_c1>r8IDh!8&#tK60J>r3A5u%9WqCGx{W5~Y zMR4PAph0c<JnEqtu&dJi_5H%JUHf<2K!+)vL+pF2CyL7_ zxZ)=&Z?Mv}C;xyS9n$^YE!rQbA&kn9!yq<bb(p;bS%`7-fH=&pnP*~|M0q~cVHIP zJj4HEWH>4LG#m*kpX!aCJoDm&n4~UxZDaGyYt;ZvL0-;=29YL|wJf9VJ_!ee`V@9l zv-Ti6LGIz}t+`W9Bz|(CHoisSD22H;#?o!#55S^$B9p;fYi==?&AybPECae2IW8gQ z_iAwP=yMIMbJjsg{V6eLlLa{R9Li${eet!p3#Op-ma5F7-bycgX$???Er)k&mg36xp zzLeOhd{zc0h}ctJr5Q|HNsH29qorNj95eMF`LhCTxJep@+i2qR>$o8(!~;JdB!}fb z`c0zt&k2qx*b>>!98k7gf3UXxK#R~CoM;aspH`9M_5dGT*cJfGThq%T*l&MnPnvG@ zjc564a2%5X-4NFpS0!7WPP_0iWtc-&+!@877!Z#CkbV9f-x~UDfWcDvq^9Sy32tX) zb~5iUoVv9NfLOvUW?IpKCA2HbhE4`JNTY5_3UzQ(#4^GZ#``(}qcIvzdaX=TR14Wck}QJ^@f}Ag$tiqPZs<)wot^PIHKL!5?V~ zU_TQEs;c6|y!71!z0AX+f#sgu`1;02e0))hAAItME$VL>RRy6savalpM87j@z7H=O zDYMaC&zWavO({Nb-+(A7Jy}>;V?9Z%RnkDGhbOr^4j7HiT<0#|kZVivSsWGb3N+00 zu!h1M=s!Bu7kkN5!Br2<_~c`j2J@{#!Lb@Gs=f$(yDN>(jtsdy^mbvOeLSM6ZiuPs zGrq%hA;jjigFZZEg}u49WH$kWj3jIzi~~IZJW1KPY_~gj826KyL275&6x#6=JQ^O? z+_J|k%5gLjAOZy&XTOl^>N$ON^H#!`S_$93|7Pqcr**gp+|3E*Z!bs&)MoKNZL9>% z*krrTc?vG+rF@vQDT(26wAIpW?ZcR;Hz|D#x-pI>*Ne3EZ%}IX+EkObv*X^EJLT zRr2#^lSpU*@F}^1kEeBQWR%GqEvaiyieAg33lY(0cD<0NbM#>6J>WL631j@6KW2CN zAUIU^Nn|wEPUs{nXwHh5328P_fJP2XuV?mRCy0Yp2H?{@Xkg_A$>^>w2LL`w zbCX}S6Y0D&KivN)oMUsHHlPaSFE6MfM=cdFWU|CE1J?$_wNqpV*M+%9yF9-NMO(ej`~9OKfnbW07n z+rFrHf^vH@KZv9MS!GUJ#?q1xD|k1`b6ei%gDt$p=$1nxA}ak+VTmokFeBo#0PF^q zKuk&*@gE2rAOkOew8RR{UTSzxSc^x%D#xXMSHr{uw}r7~^b|gG4jKV`XC&9-TrWUq*cY z0EfXL9=t+a*RVgz>=N~u=`myHdl!x8rhE2PD4)0}l8&bQD5q6M;B@?UVr4Jt=)06? zX#W>ScU;J_I_{*GNkd>@lg@3G>BbdhO^6Z+OB!EZC|hO*Zz9ojqy2AU%7Z5Wz5;u`f+Wr0OVfjJLaBwJ7~MliMbnHXt>_9~Y&C0+!{skkJ+stk z84fq3RCJ=c+J|J{a}lur$k$sC0BQd43wME<3+>8tOWNX}osbiDLy~Z<0R)~bH4c4b z257HE*JH;MFBj@_d5vx3Zp?C_oR02K_<~E0K}DCM_~53SyMm=%WYa$^1mJ0bqD(1) zq#~ek)^$kDw=Mr_`yR5fZEc?Zv@vO)p-iK zAT8kW<5x@$XPT`(6smc-#v}p~wpo<{{l7nQo$F&Zgoyx21n6>XrQH1!f7(bL67|9% z<*|;>e2X7dB+J3)pRYbMp0%GJW{ix&J#&GU^g_D56?YoBQAZcz|AosXrd1s$MQ=xu zI~X~*C3fMaBBJGjZ=LR!aY6)H(=)ugVNIb5Z{boRQ8QYW3Qjd~ZF^x|Q$-fX(7KJ_A@dPfVaj|-Bl5+eUS3)g6Daa{&hM>px>yO}Qmilv; z=hfq#-e!#mDwH0QeXP=aJ4|8FZ;2LvLWoLZjZDTg2HYP8R3bJ%YW*4ib!KpIN3WT% zJlsbS%YQSMlEmRkn#I|Ko)VnH_!i?r;yolW#0njq(K&qJ`0-8|GTV|lE&8)6CG%!8 zbA7d|H4`#6zSYYfi_IoLk{U{Nhnz+s%WLmw*sC6-C4@Dd0S~?j&79;+bA+>X@i=~b zqA0OK$tJx{Hx{XWn25hVvN$Lm$}`_v^KiDLPXY2K@92x$jt7CKqm)nN|tDu9PH_EaXKoZ7`xp zL9s4CY0Y@ri{3m`@|`b|J-1+RFKOL-Qff~r3LJ9Z9`wNlsd4zdxie1@Mmsz&V9F2+ zoYiREREShZGQMf_!-2}XN&8fy=j$>XSHb9R%N;e+u{|(L2J_BIwtR>@isNGUQPM%4 zDF%88zwe$|reX=JJBPyvN+12;l?(;%i@JiQ`|9X;ajkhK!717iZV^Y%<#=^Zm zU$akHnb-JYz4GP_euBu`snZg@fBn*)c3wW?smt@S}D(+KOP;; z)H8}Yww~&}a&vrV;Rtw5^i9o_OlbdmP0PAIyiQQ2CQeSBaz-;K^(4GRuPNmcGR`rB zKNPOdb2G1Nb!$t}>()oS>kroQy&fJq<`y0%S3twQfwzyUt~&J!yeyQbc9Es(4#XkH zZbPHSq2)GDLUDP^l#^m>{D z+!bsynz4$1-L+)8kC%{UTG_``AFQ~pNA$xi387;2?8mQi{P_nKnJo+%HMNld%`l=8 zr>2bNGNK_dW7Ue9H>B0avZRxp7};hZCu`^HM+djAm04MmBug(|RgD4(<%AsvhbNRe zCYem5eV?cZf+3&m6H9?}EE&%2i!lN0;WpZIG9@x-&xUmq!=Orhgymw({OYn_KaT~D zRYGK5D-)dSd4lBIPA$eu2x>LjBu#V69k-Tzrc)U$W=W*rPFx<2Jx>hPID>z_O=Ujp zPfV&82bk5C3VHt=ITY2F7zWFy(z&oo5ha4k6WesnT2CyrtP2knvpGhhE9z~iUWVp0 zd@3S)(l?f~B%V$bUL=^y2{HTwrr6zKrbPB(_b^}aq^9fA`-CXX+TKy-gsaZzm zqL9jV74rjjud(H$Sla*yK(x$)_^h>dZKOwZO$&CiCGhNd9;&o~zTBy{0{qp@Jko*5^0Q#aRHCaSEj z?obZSZTiu`WR1jgBxoYiN5dZywO4-CzfrPiv6`YFwqT~$df?)!zA#C|7H_HE%c>gR zkGp^H`x}@BbERN;mFaQXHKK|$t^~NW*qfAMZ?Jr?0jo_gnT-TawB5K>SPE-@u*y>) z>8_dgZ_vg!G%T?qy{nnjX-e$L9eQ9j6mp4s?c%r)n=UZS&tSZ{63pzAhLur8?g7 z)NkK-@#3@$8*Nkr{_fP$B>C{hOx6ZDJcf`@HvbU-Ed)c$gqnB)Q zGKnFZIVOG_u*LoM&YNcS*7xdae9m8^dR~OorPX&E*LEB z=9G>9LQIl9is=Su5)&vmIs^AG46Ltew0Yk?6 z_H#sO@9TNJ6*5=LTu0B^Te|gL4;$AafCT#hj=RXkSqR{r|5|WjWj&~%*m<@|azFdP zw7Dhect$Cty6c{xX1mSkl`+zM`5+ABP#BG*TCGhw?P+&>1L&m=G+lG(F(ilh9doBQyFhTP3G`QW@0PH&{+v4{>N050=ExbF7giViT1fEl*ce>$ z1?+!SVC^@N`H-a`Ge3{kF#~$_Zxm|@SL67hAjFUs&IDijyY&|0&Hd3w86nTD?}cA* zG@6tet0uz{?GQ&7=F{2AA1(zyS@~hPd)a2NP+y~wN0xLolf|gUAO+I53Id_{1%=_M zM1+axh}oIx8G-qNVP08jJM_;dR;s0^pZHB#E1Y}AHKQYt9XppaXTY;*GW`L&cf@c0 zdC92gT;}-oWAMG2L=}a;%}vHLjQJ$7YQY_gxA-xHJfa?^$fogf&UZ!NaY=PGzhaiA zLWsbh(Jf$%kh6jCmisyS3$-(igTP&JEOtL(t2JQ;v(-^!E@ZNE0zcH%xb;fk^^v}9&3Qq6V zB!I}6)5^8dGEV;HX`5EWdt*l5>6Q$Av|>uNweKXOuh;5h zhHGSB1kC=N3v+*iCN#`aEadU}#1uxc5mhuJk6S`k|Uc<8cDT{oG!bh@e;Y?J5y! z90b;DmuUy(;|v>C*<;C(2y~>t-mSCQ(Alrz=aC@z@H6o4o!;0_t7KE5>O4K`&+oDm zOet}lCl1=p7oxZAY1LR^l;76)6=2{Lx!OCh`?-66`t0Wg*6BvW-^$p&h;@bLPQ`N| zTG7Lid!QwJyk}i4Ae50DAJ*rz(6bsOiz9pNu=<`$>$=BVaLWLW#JnGZlCHTX=^QXW zrC9AmQwIdz9n0X11V0TtAA`J?zkH4V(inky7BktL#;7k~Ot6Oxc8Np9uDpn@zkltv z*VI}S(*_})JDcJ?dX5*c%!~tR5{!4RE=VL;ypj^}mHhZE^oIKV79+;Gf}=nree59M zO?M@BVP;KzDR^|#)<#~|WF%yzWFb|oTH<7Ho~{YFwmA8bS`zptni|7zq$C4+M7qUw z=+QW2bLt;tCTEj>T$ z!061ZO5IodH{Dy+{1e`;EYYCP76YKHF~uIo`;YoriD7-$!L(`Gp6>LG&A!5R$x-sv za-7bVt!z86?#`QF@}wZK?%CuL#m;!@*7mufFq{))*dzE#=_c-K)Z&MXP~JWglQo9H zXUBjVH5&~jboZN4=j)$Yx#jB^NOd7^f%+A}82Fjs5P{%dJA{<)O87ea#je#7ZxA*9hoxP`tN%!2SRZ_#K{i2J&`X#O%H>s^7nCsyKDiG9vZu#QvFm69@0soWpLWJ^yf9i4ht zx{0U>*h)sfc;1-9jetJ^=G?K?TI)%0IR3a@COK>hj0u&d3chcR!HpzIz%lDNbewVF zu{VKE^-+a&At%^!Jsfu<@y+nvjH|El;2;M+84|A$tXzq0d@r&~U z@;{!_-KrMAK@=?qMIt84`TYz`Q&q~2T1v5C9Q@tm%G5htGv2uC)WyOxO=`(qk`oHLY8pak0HhgAE=ohmB2-P2xA~)k)$=^DB=u;4~9@ioG zlD;m)-@)&vvK%vmEr_{pGXAW&=Awbx2v#@)by|#RGd#Ej>~iD4WPj`0PEc0U_?kU> zpbtAz1c0^r$-||v9Qffv(H4Nk`#SQ~jhKCMRaMNk8~xVz2|=1i*T%WQALsNm)cQb; zQ=t7jKmgKNlV@`5;(3nq zKt+GGQf)p}bY`5hQhl)nSbG^`TXWXwZf7>ePui|kA8p~x2<$o<^~vov(m%aKc|}>h zzl=?KKfnI1E$R!-J)X#wm({NfE|kD6_typbYl_$z_rZ+X=jTihR0NW;?dCfb$LA|i z4m@?}xYkh;iT6#BBYgN}J^~JB)TG3ok@?3Tn&KS@j#7V<2;NeZ-4%Rd4%}7aFw}SzN;fgQfCs*I@5gDXmT-0(m`Eg?`heHtGTdnIS4)Mg-GcZ>{hq=kFXc1MXKQi_UgFPMe0)GXxk zInANQgj(ZPJ1?T#&)sC*IQtG#DGC5;>hWdehCX)~aDRx5|6%>Z$2TZmTN^_LgJSw@ z`R}1&Scl?ZQe7@+ATJ>+Gh$< z$oTWaQ1Bz`d~c$CxtXyQCS&YH4~V#LzCakZaPWhR*DTzK&*6sD3Vwo5o4?!I2!+*# zj4Yr2K~x6E@5bccr84ucuSkzx!{Zk!?!I+zV-5ZL?k3__9?d$dpZ?Bwp@9LKX(Lk; zi+Z1@XSO_{>UqnyX6*t)rSB_n24V;p#NDJQBeQpkDn0+B%eMG~$7!-s6p%6Ne8v2J z;9Bo_I&7EgG;DTmTYwnzwI;E~2cLr{BP6CwFiLjGLOC0NsqYyh{zT70>`Q@EcHXIk zd@&8DA@p*N_hzRQ_X)dVBiMXlhDbb~Ztr}XoBEM2IrZgg(G~ylC(rpSW{WOnD3!_V zU}mEz7vB1j)t3%ED{sk*0cyYTIYw8ZLS{xfC?}7t0suY=5*q9 zqKuE{SxAg&6R~nK2UM?*<5c)ifc3}z(4+&afM&!Lzu4UJR4SL2454R8kJIeR=2ys5 zB9p*GI!~`M0lnq-!+um<++FukiGF>&`l6lTtF7_Eg!I5xu?@PY$LY`0Z_kV;Do_KM(x zv~4}u#t{x7qiiQ~?snoVK3KNNLIjBPc`2dlsvGBi{FvBCRqmpF6lsniwH5>@u4z3a zo_g1%q#Y6)+l3TLQ}kK&yuhIr!x1R+ZkeiGtvX&%CDHBEkV{I-eAdCIWqwdZxeRr`Wx9w)Z4*v{(L%C)%8o&Gq) zFFFHlj$%#}Ut%?{mZR(~9Q{*gA*??r_9O90%n8VEYwHDNdnszW95j~G9@x7;4dlx{ z0%T4r$y{v0JYQv^4|(NX>{3wRZJu>#N(Q=_sZ>z=E4*F1%_fiZ7%#g}2bgsqLR}Mk zXEc@AeArUe4)CLepyQhZ@nFm{eXi4_CkH@=)BQqySpLhH(Sp8 z)3e-jvf(;kRPk{Ab#Fm#<9B-ndtuPtklFdGN;a->n%vFeo#qep=(lqOe%CqoS?P+Ll8I6Y#&ez5gaA%Rk zRa7K4yS2sWqHyP;+w~ppp5eu?-!r}oh_&aoM4@_m+y67|cFyUtfo5r0;QdkyzF;rk?X)1fVQ-2sBB7sIMU=t$X)v?>GPD+8 z%ToH^+$zypIw|sY#S~@0Q&agMiY&7oXKc#z+kCG93nj?4`2!u+FA)|(IQ%6sR3eL` zu)G9AgCiqb{%rOsuuC-T<9SYdGlv9h zk!)Z1sa;-U)33{620-=#8YZd`EDCB@twSLq1KYi;7Y;>PYR`HndeH4v=gK0k3I)Kp zp*VQbiWQ^6ttD{fzFmY{aa|1wO)@nC=mW4*e$AWpLQ-1i!q%jQf0=A+Siz)r#BywJL# z+||o|m5m@BpO8IiP8rS9I*dT1VUVK$j_KW&u5>@j&-BszX=U@os@uON`P5=NqY@8R zICkUWql^A(=kk?r`oa)m;l+FNlGw=XuiN}({+CT>n0C48bq-k%H||8-^KSe5O3qBg zZqBsPu{hSSQ48gKA7gl)L!ZpWkxnRk| zTg@+IQGFu0Ivv$-Ae)ej9~Rl%Vjn`(M|Ayh&@7H%xE=dw5)Dkx+iZzoL=2$!ZjV<0PD4sfKiXuJF^ z2U{RfGiN%L)S6x|v?Sx3!B=LSU*siAxv?74Kt%6QdQy-rS1aiUT4)Ddmmt0$Xcy&F zlJU>Pgx8n^6EZb#su}*)%#>K%;Spe_JD+zr##tb}a{04B{`KdRx`zRdV5%fZa?q*F zLle!w_yDvDfmg|}TqQs8^C5>b#m{C-YpdMS+CV$N!f)yq0*pB2nNB z$%~JG696Uy(Q$>MKDCh22OO;NX|sx*Xt#M)tFN(3G-Y7cR2F7F9S{z}X^C=N_X)M> z3`SiYa+l4zIB}c(hgYjEPVH`{+6M^~pC=T0PSbPg1d6$3S4Ky%z;MOS49+t8#$?f7 zi-@oDN^R6?^{#kZaJ~A^8hyQ?;{t41l{1*-b-o#iV%Sp>U~+o1J7&YN3JwEcdZ`1= zpt7xsQ1XA&8DZ&D6(h`q^s}8Q)MJ|KOT6S^*(jL`a}r2wa7ab(65@$9q%?xrD?z6T z--@Q@;wpI4biE=hIP5Zec+%GmRou5uHI{XV00paIrjm#xf|aS;->jR;+Jcwhc+fU~ z_ubzXl9makO`oJZIPOTGm~f7NVu3juecZ2FQ& zGBU^-zPYexwrOt|5UnS6m&i(mouIL}U{omKdj#iaJtMe?8r$TEEX1VOuNMgKoh|vr zY#NPxdcdq0lMSlKy@D20RbryXpeN!q2CrkBS9w0c4M>^a?y_P(TFT;k!v|;vU>$2d zQAFZ*BjEF^!pvsf7h>R3yXI@xi#_a|Qh4kbsi{dA;c)o33kpH))tN;t@>v|H*`7(8 z?oBd!Ume=KFSF7?Iuzzq!3X8*2=Bp*VL0^^Ngf)uhdny6Ao{_K}6j#}4yD~M>r zwHgwu)-0lZBJSSbV2vVN4~tK>5_4nosXZvss0&%?#gC#I-ix__{CLNjUz`=YE09rP z1LOpi-fcGSm6>SiaJS7Cl@=+;jgRMB=4Mrf40GsL8s8wQdGPmKH*7rHl`wP0a3-bT z>Kb&EhqkY`dsesYbu^77Ki&BdT2}W7kfs*Usuxal8Gb}^4 znm?zwNzxX~ee(fGq~) z=j@AZRBkIHb^><3?9YP7x?*tD*{b0|f_K8N2G+g0BV2-$G2zf`kecTauy9bp@I`_H zaLpGF=+HySw6jbIi*zk=6M6oXXk}FJi~Zy_i(gY=a7paa}wtvQ*3U zZ(GzDOeuVEzkQ$&S5aG?t(z}E5*4Wa@wZ1hVdz)u4|@n;ZB<2`&d0eo5b{x7MPugA zXHS(VMnI_!vlo^_Rsbm$Jl{QrpuJ$V1hR?KI;~2p<;Fn2iWE)n_JLAdjJ622QPsTbi{jvf@W=@es3&j%8!~^TX-|!DC zH9v9G{N?sb0f0A|L&dSjk`IQ#z#tm@{zDkny`V!YxI^MGe8AJPe2|LSR)~-5bDXZ0 zgdqSbKDr4_i%~%9BeGoO5sA342?Khd5gy(us=m4PF<`S4$+HP;RX{&Pp&maYwURN~ zd8~}0UL0FMCoCj*IXS=qtUL!O2KNiMN99mw}lvIN5To*+N{vbN2OnKOS+x*(ffWt|` zN!MH_hc&V0r zeU#XyG_MRPXbHRuw3)`aarqJx*1=|c6%k+%qyxFi|{R1wZs9Z#C1&i zB1YeMl6P#Kj}CcpuZ>I$LCIIbutAb&JM*H>OA_f?ITDU@_*3O?5U|D8n-LSPQI(6u zs9Q|=bj1g~Xf z?JXF$g+4f(;ElZ_nf?3ZkIk>dyI-|1Eq=*~ox zeFDO8vHO1{Sx87c7~}^iv2lR)ivhI}M|6b)GU{_~U%c*$fkg!|o3OgHso8Vyzt4V_ zu!BMX0x5guRQzne77C)=*7eB?MA0nAc8vB=0ur{+IEad#<;t7#rp=$C(j2c3B;Pu% zwqGECR&~_Xds|q)8oZ1!IFrwn*xWBXX;v$OcQS0rTk}D;b-Y?BDKr*}dzHlkO!EDb zW=HtOxv%c58~$r-w7%S!D>0z4GqGYSLYwlDT?Z?=!SEvXkW*rn6)C{f#MbzVag;2> z^-_H%MQIS?!0ZVX2TzJJ*6Y4#EszQJ)F9*fV^k0LRbilV-7KasZjJS^}MItc;~G0doJOgqXDgq3SqpS(_`#r~R1F zNc>)pz8GGMQrf+N4F;4w9A)>M^gp{>h6xCLd7KT`X9~g&@K7J1&TH9}MCLExHS1Zi z_+2?CT((&>L#w!dsmsf|gd&1(5Q=^6CF*>wt}Ag2+sYbX?)m{RfKlqh8P}<)nH&jz z^Ph3aVs(h9pn}1UCKr+FLM*jYsP}e!%Z$V;186URHNSd#fbe%p|M{xFAvrG}56Y<@a;EJ%n5luD^(RqMX=1?_?x>LxkbniesezwuT->UjJ)Y&_zUM~Ax3hv}h%=TA7# zWp!jJ6kkIh%;4@}H3~w>SFFB>-3aIZ4lYXAnoD3cZzDxi?U zp>Qj_aCa-*-Q6{DcM5lx#NFN9-5nCAaCi61zPp|CbJt$y{k@;_71>%wiEND#|Jj=AN7}y;);XAi1?Cl zD5ok{hJfukU&x6lwp?0LPbSULv&Z~MN2cZai0c0H34`S^2@#R9t+fkfX{8UR8Lxtb zaU*BYD_B_R{GE4i&v!*wE-kPZR7?~BA0OS%Ep#rEO)$1oM>LChDMIVF)@V8rUSgQ= zkloaoPD-=N1F?PnS(L_@jd{tA_mZgdoG6Y-LP3}ayt2u!PC9DoE15f*aan)6>ma&3 zflenkdw0UVym`olb(DesEo;4H_TA-n(^Pk7x3z1xOmQ`5b?V59!{x>4B=ci^2XN?} z*{Qs4qA;j@46S3WSvlj=^^p4Uur zeRO)S<99B3a7E95y*Pekjq>@k7B>W9f4^}JDSPQ3k$t1{AmFoqqWJqvURO}JF(&f| zHgzo`ax*m8hsd-i@;znRf8=)g>S}*-t#mWAoFCIZsaa2Ka|`y_{{aI{jL#dIydj6L zbdA0!_qE6hYC2)v-Kl>^Kiezgv&kpb!~DWFLAtRHEz0urSwq{Asx^-y3=Uix?lR;|&SD}46YtkgUY)DNb}rCA;UPj4Sasy7Vx z{(6cgG6()vtXbz845$CnOjNj38 z*xi03v-YI7{$y08w*>utXyyMth?eRLOe$U3dE8*wKalCV3&=moZv`OoS)#uS@BLTI zxVaqSliO0>Dv3??_vZUI^1y#J9;lNg7XJ68RR3m;{HaCxv`6hRt$F$1F*5$EwOa0< zJP}=YFCrS0|7ynHc!Tel#HSyOmA2*A{Jr)6&BL=jg88)8(Oqc$e}?|E&HoG9zv22{ z(EeYS`~S()E^**e8?vo(N?MbM2b>%^p0;+4qJbSp1t6~fq0hoWc>D7^(3#!G-L5ls zx|*XU&b{|Xr0lrrwI3KyYtIV(V~@Xu_zaPQByd;(XXvm2!t?az1UxS9jyM&G+ zAt4Kjjm>1&OtC+TLR|bh(YExs3~GOve@M{2`{q!Y(HpPWG35|(o%ByVJfyqjUoTOW zZ@1wJz3`GS(D~vap*d1n_v7a#n=DxgOq$x0I7&m_g~9C-=01YcQ&ppcGwnYPRAt>9e*3)7?jF;Q&NC^m ztea%J*GnRNI_j=0ia}6+ytmC3IL{iC3aO3HuHnnq6=hfFzpn9kU^YcL3Pw703ITQ8 zS+HB%+^4bA^ertTO0Ej9!gX_h&r<&rrtE~aQQ1$AFB;5TACn+JotvH&--aiEmPzF6 zw%x0h(vS5yaSv%^QV$;zw-*_mDa`TlmiQKr4f=NYp56Y|Yk~hbYy4}t{o^D5hYv0_ zI==W9rwbZZ=vea=xWnKYW10tkvOqHStYu;E;l?>D=gAlaP06;H71IQdhTKOUU8R0V zU>=T)^|`BS>yDXzK+24)2*$(REJ$5cTbczeaY1RZX|6(^lpbyYA;N0jl*iVgwtucT z^)!xC0c_n3)$Btm05}MLD$z%}~cMei~q|w>X?WL=eFXj>nk9$A5Hd(=#*(dDcb$Comtg zFq*|a-MgY>v*zQolF*?BYUjLpKOws%?S93RG7jVF4uh|jlfnO%48DUhjYV#N1&)5m z^ifGfI&EM9G-=cf{A#T-Skwbxu`(NVsrK5@=+tcqC1}J@0Iih>m;m5TAsO9QEGse3 zhH%RuGCw?_K9;nyKiUsY50V|$ado;TQ?#q5OlKYdk6j7-sH?2>bv=&#p64CLWON%S z1q5DR?)}>0#wsY`_-;X@CC2%V?)rbc;C(uq2;?h8AMYoo#3##&v;sk`JD1oFJgAlF z6L@SgB>fwhP>SNIvU+2GTLAL!amn{X-S>mu>|AIrM+B+pGyP!A!{vNn+kL@V*+CZN z7VlrpHa%^x2Vb(vdeu`X0cYiPc?yd5&B;{5DM9W!6u}3{CuobQ+mSE8%fMy2;N=-W zG4?7(;;TsNN)q0ErJ#QD1aHApxpjp?UYkPpa6lceU$+Q!mq0v_j|IvC0AJxLXB+=Z z3qbrxq<$5O@uWu`?wCNjK8tOJw{U(BH@m(*uH8Ly#R5wCXmw<*JqhB#qvg8vsNEBA ztL|s=nXqn2odt&ylhu;gZv5WKttVR@P28P?waC-2&DiX4966K!UG=9@eHhxtI zzuE9RaW9W;Lb>{8^fmB5k1O3DmED0~C#akqE>MWAJDni>{nl*Rw{raUWzcR5&-dg^ z@Qk%-1J<4CRHI>y1+qDU)(cLb_Nf-DHiszv`6KXI!o>54LL+ZENzE$B03ce0ru=G; z8$!ywO5?-+npbNLBxlHsdLy#9lxDwZwg-b5A57vRX}woH8!fAG`jeC2lSByC+M=^7 z@a3CvgTABQ>}ANa(Qi9;BCDxeE`t#Pt&A8n-z{H*DhN78xgcE_y`SqHKC3mr7b>c@ z#CkO?)ox|?eKsR1WPi#V?qK@7vGfPGAnz8#riL#!yk=4Bstz}Ka5sM}v?VhTrrVay z4T+sJ{tW-mi2R0rrGwFun|__TNU|Sld!c7#?Vq0hrK|qr2tlLLNeqqcgoe20f=2AO ze#KLYsn78v^nmg{>L8WM>z9&Q1pLBLcXf^PPXj=6`&ZhZX7BEM zI_KIJgX=EQ#%9HoPzxu|w#h?A0r%=Nr&Gi3A-ArW$UnBPn*5>81vPHqo$^UWgT!Cv2!Q%%kq* zggr@_~BK?(lw}GA;FVy5>$`6_Ci6o{c1v>M9o=*c7Gh~?6Pw09eUOS>((V29-;E0^X`cpa{!E(WvZ!H`eoewxj!t*hLE89wTOBVRa^ zs){k&zv^EeXTWq;_4r$wQl%~lqM23};kuai@#jmYn&hgla2maDDK9YBMjnsmT%Q)r zq~^Iy8)ce0gNR{lYWO8g)6MC+kpCN)|L4c!1gutmc1!fexr)XZ)5zPoFfaG1VC9+Z z`%?@H7w@LP?1r-?LF!b3oGfIUroO&@X_nB%?$YO4Ob4E49k0I0DVKqMpu6{*W*w*f zBCWv~o22;Z#}7NhV7nVJS?URCde-^GEjdD4UABx0j@=8LsW?+zL*Yz1xHJm>8=Loe z^uPXdzxLIY{6hpW;Z<5Op=4@FZrF$2sT}{L9H1rKm2%_i9LAGzw7OJh3?&&iY8Qqt z+W%M+k%bghl!DAk9%lk?C5oV!(iodK<|40$9=6xt*5c7jV@=~RvRM<>)JahYa>4Yk zZ1z((1iP$d8ftHX=d@pHaOc2Lwb661fKgv~cZJI$;=ieKf*kSwhb^r44cGy6YFaVX zC?P-*B?T+gQmV{ExDlMmRG49U3MOrC$&udyil%{Op1?e}bIkXPogyYLN0eRZc%Zl< z@?P-VQma0OFdif`j;4W$RcPgW?a>t{T2N9lwv)cTny9=u6j|oXZFJhet{#pSe3?0| zq-o!5q|u_YBH;LUMFj`CNYL?3tC!2n#-EH+H!Bq=doT zmPpu%5(mQiPqVv}=XUV#`6M;*IS@e*OitD0t!tVfl7D3s1-3EK4P<@B+-q5_O)eUp zETtz*oNCyJhoX&UZ|(YJFeQuq8JsH$my*dS_qMT^c)e~vmMlF zT34ns9i(tDU3eaNZJPUO#+yM#V^gNx2hQ;Jr_3G2mPt+nNvh57>{Tk)`bQ&&e$Fh2 z?e8cCr?BzrPngZs@k|is9Zhp5$PmgnWf@E-=8Y|lMA4k6w^}fbxsWKzDSRc;Fx(7? z>VTSVs(|vB+P@j4Gd^VicuAPC_MI8Q9ZB;3IU^mh6#vQVH3{2MZF|*cdU&(O#QeLH zmOP*jiaJu#)TW05fs7Zz`EX(yBP%I=M&;4is>&rYcdm#P_R!DcfTEqP0Z(39YwD8y z4gO95Sq|d+7vW$3@#347^@~`*0WFbK_GeAmaG21@j_Ord)9fg{>Sv0rA6wJ5+u63nq+`z)Q`T_24! zQ6Wl_U21}+#`bz}eFR1HhVt7i<4@|U!tyBp-I&H&r|yvLX6DNnV|dn*Rb~i5|7OxLwOKZCmEL?xek# z18hQ7wcVpyt4e*){t3B%|5r)j`|%}5J-V_zrgw*nqvjHocVjca^JAm^;aSeFoTj>~ zuz7yuA5HM@2ThXN#=}l%Ozu+JBUTt*92>vBuXEbBZ3vg?8@hVy{pGx0XaU~o0ML4? zfc^;<_M0gCKVG_7$ZX@GCu*;BrHNiZR^%po_$Z{R_65iRNqa3pwY>tRr9w*1<{~FL zUWZFHXn0-~3{1*yOqF&X@@Ht~|8Rq`1y5TRBk?@_b>gi%{z{;VS~REV>`u<# zLXM;Bnd)Nnry~9c?_6Awt?b5HsUzn(0qU zYMj2Vx_4y~9RSNjoj`tbXHIim{*Pb&uO~A9`a_1Ciu1F@UGhJA{eRo_PtcF-fPI`? zY$d|~(MHW~Fh6si_A9uQ?f+*Vg9ei@>m)4YNBjTK1A)I$Qf)R`6-ib9NAUl*x&Io< ztp69?PjCIV@1E$uL((hB%WKG}+mX5hxXLZdt7~ez$02rojvu+8F)fN`aEcex49uYH#YEXNtp*J zuV0{{Z<{=(@MFi_DnObGVU+QoIm1Ev*EdN%?FU-x0ASr6Il&7H3zIFZ9tCvSyvg?F zRTXt8e3Fy(jjgO-CkA|6W>9tiz(U?87ATGOspK^E53Yd0Ixrb_dHWKq|3zT#`E^bi zGzV`>rK%B-N@=dEoS-DGYijPBSir&fsjo_SwvbIF+)8H-i@vlf@!{m_446LYr~X-# zKZxsJ@->RJ;p}b{Hrv~62Wfgx&uuVbO4abqBr+m)Apu=En~h;;d<7B!t2kZ>`t7pQ z-C{z@Fnps){880Ex^$tDnx<3dP=@+OhZ4i_TgGx+f@hymB`tK_Wv2o2h--H`oFmVP z06u!gz(py%Lq|u5qA^)rZA;JaifxJOBiw5Hqn8J(xMWdYk=SZXU=X{er0UI4v<`GP z$k7<6wbMFYMORX4w3gVCA$4brPvOAj7D8*iZ{b-w14Hc3ExF=(xdKnx;E+;k!9&)& zuC)(k`iwgvQ;gj4YhM)-wIo$Cd#qSg1Fs=I>{W;qQJp~WaEMNwYap;{)1$#NG6^%;@U z^QffLPHX;iV(A2f0XU^%Q9|E~OS6p;0Mb*hd%`%;w}q(v4013ggz-TA*|ew4bjKlQ_swhSaW|0qum$-(Iu&)b-p+mZ=O3M=}W7=*4!}vLzHr9o8Zq zEF^7ok-FZMvVbN(@MB2EJ+6d9;5&O_r!LwA(Aa|-U5cqqsFB%Iy~EoM3r z{h*H7*UuchdXxZ6W>X0QJAIdgz+b|_u%=nQ^O+(e;(W=Yn#?^lq6XZw&u~04wP#?U zGZMGXp0L?t-Q9m6f)$}7Qx;5pM=OqHmHTC^&WK}rQ82v*3v9$QCpi6GN6lWmqX0dO zndy}9dW(w7XzrT@sDD+Da+39=!rvRaD*yIp_ ziwBh!byYzD;b1KbA0wrkjt3F`h?y1zWQG;u4jFeV&|Bi1s)ms(%g!x7Suv1d|4 z7HQ1TkOP_$ZdT@f;na+;a912c-~n-V$Fsqg=s7~qD>zVzCz4&2qeg+X+*!Hc;p&~l zf$*V8t}Nk&E=a5?>r8UIP++0iN~ge*fZ^@;6!*1IZPei&|4rG|`WBgIkF#1YDR z91^mEHqE4sFsA*mG;hy+;3|bZ;y{f!lLi}N|H2B~$(T0vbW!+k+Zba$YofVcj2D%& zqZJx3A1DAWPMywJJj1T_c3IXoN4jCSN7~QlTXR7NAD1<3fW4}6S1J?$OC8W@w7l>C zifkik2Hb)>oC(s$^NO@Oge%avu%yWqNMmO`EpU|44?~Q1C9-lo#QNeYeX_KVT1+8Qx@7l1#8PQVUl#~cGU%PGV~fsGlF=9TkHFSC`2x2m)1 z>e_y;m0T@Bd0Z;ji+);C8GRv#)3LmQt<(JjkWE1=4`0WA>YNRTpJrXbhdUe9wXpw| z=^JFocTj8%;AElSD$S<1BM3h5?z;%C7;(1h%f8w4z`vCJj84>cF~v7&8eQVBPJVo&~~}2!Hb+m&%600 zO=TUE+5JaGsyphU8M%uzth-@`RF<^01S1q1{s0nosE6ENTb&4zOERew3gU~C1-NQ$ z0V(7>wiYvyGV8fVn`>C2SPqqMIYq5MqqlpBP^il792|P*lygz{1VGuUo%aLErrcS- z-|2&!tW0}=I6uF^4QZphtT>+t~S;u@v6@)I{-&Lo%{#VUi>V`?+uT z)+@b*L>Q-ioU^43OSo3t)u=r!mFn70(qXXz3_~u1NL_rs@r`HZ3xzNhY3|qu(IEHx zMOO}CK&(BkI%hafTXapsa4^~W%PegzmbSko&Ql6><+Bt*rOwDADb5mL1MH(w-tqLi zxKa$YkhtT8a$*-zO7CsQ>%9lHy6ZMRvM`$!{+@L-tZ)oYU)%zWC6`jDsH;P&Xk)f{ zHNlrCbP!Ey6S`2~Yy3nM7VL7v3&~ZIZ$Oi|tWjBEJz!rS7gRa)vs8OkE;E$&;V%YJ z$Ask89lRB#{lrE_>u_>UtF3vTk0I#KHM|9t61P|xDQ;Q&?C-mh>7zO>>!a7wrrO=E zuIs%rhh;o_SiuYIAwfkCR9djqhMZZ2r_w!@+AadU z0%k{NKf$U;l!G5JOJPfBc}E%vKKin5SZ}l~;dN&6Fd6Mg6s=Dr0UnC!w9VVg)9b9a zIu+CH9$Q2vKHUDO+SbIiE(C*4HlbZ`*cdv>R)?Od*O>ud_sQcJU$Q)JRpBO!$4$3kYk1aI88AGwR zngRRTpuFD^Y*a zfO~C`6`f8)S$Ksc82#oq3!5{o!QMdUbk&>6o2}FE{gDOd!k3AD#$19Aav$5TDD+SFU_aF~37h_Ew#(t2! z#7%x=k|YmD%a&fI^kU}1xX%qsAwdHfTQi*(zMHQ{-kSLZfEbkzIYp9jc#?{Y0?HHY z&DN>5@4WWwB`i?O%bf>$c$Y#ScEqP+m@nxtv?o@%?C@EFDQ2b&hWw%eJHsyKuc0GBPR{R+g|0YJ%)OaPZ3%ljF}dkil>BKio3)rn zKCD$kr;m`to@OB%mj5xfDKYNH;rYG>I&DJetDn$YiAWt)^nW!!v2iJ-vA9vFLF|96 zj+>WT5!l;{usFbw>TqyCndVjvXTJoAQa`v$D0+Rhs35Rk6?K5OwRa|RmB zVct-P1tiv|g%hl>NQi5^K&1g`561G49nsUA`jOO!V+RDdLel|%j#?kh>0WRO6mFSS zUQd)w-dmCgqFd*7N>rPTUw`2r9?sgnfK>H*;3wUcJBV+{8 zKPSnf&!XP0xjiesT1y0d-6V zFHTZYwuBE>35>=c`K9pY;h;+HPZ+|mJ*s<&l}Ks!ioS~DQaKIPxM8$5*ofTx<>Hxc zK^#t3u&~f&c=tR=_8F8ZD<`12r6> zi-t!?YHygO+s7m0BhOghlZHWKV4qh^Gt)UE%owjp#9d|wf;we$xtqDpf)GrPt8Ky3 z%W+`#+FJp~2ot88smt8AUYtq%6J|55FcbZF_XhFAG#F*P4u~&12sQSmg#os3#?#HQ zQW+~*=tN*#T8Yrq`poKfyJu;As|4dFN9ZZz2@t6KDfMngSK`Hbm|M_SaQ#ukl9O2sUM^k~$wVu<%aMUL|xb1T&&Ea-L8# z)8!+aF1Rq=KbQ-JTmBesR&N=P+mP|Gv;{DesA|;W9J_3@D5iI0&=qwIVGN(zj_J*d@%sM6>WPG0@QdO$b1n z^I7V-u2_Irb3yda%hzHgY|kbHa>tW0?O$J?lub88^fGk^SJ$CyksPGmvKrzh;v=|9X1V1EX0-z2$x+o^%5e`Cn;0B>F?GCi`8sQ}ja(|8dtlN2V0)VsBR3|( zP=D+)p;yQn2+QM;@!4?+dNdNPy!19`QZavY?;<)YEFZJFM|Q>g$e>rbiAsKETar5S zUoY)t%AYeaI@d&V4Zjomli7Xro2`6p*)mq7AX?FHocz=Vc{5lWGv(1jROmEipyPSr zD$(BwNqD~+Fn@!ojn|sc$V2k2aUDXWR%-~JV(lF^N2LFuF>B?f^O#vlsNqyfMJ6X} zmZ>P-E{GA-@e1bB=$sQ-)Ig>=k`f=H$rvt7;5i~2N&|yQpRh`RC9cZH_IgSRdZ9=g zl1ma%H^(QZ^t^IRs@6nKdsjsSj{|Uc5qz!TyBT!}W|zZ~Y7Q}dK(LZ&LjpP|;F{q( zs-Hfc0MNa!@HR6oW#%(n9EJo@4^Pb95?Kv*0st}+j#EL)+;>~7aZ<$47aYkLI=G?^ zE?mf8a6U4gcOePo$eCr|B9^9C51H+_ZTR>w`P6^1)0xRg;*cG+m6F^=PHNrH$7()m zmM;Bdp8B5ShQJ{`G~;}F)^0Cr`YLo;%P@nkRywTlMsZ_8yG6i9hT`^d1Bb;Bvzqs6 zhq|ilpfrn4nG-XZ$lI7!6wF1Z19=^F$p~p?(-Ok*z|;SH{2)H>Vs?e zS|JR<3p$)ziCSXbW0V;I3vav_9ze8`ivJoj-+&U!7|o`u%O7Scn+k_D7LWLl6Uwe) zCL;rSqyHXu<{05t)jlAV@-aDX)8~J|LprmwemEnhmX}i+a!lNHv*S|tbo=1-Juo7B zvzZ}+U@kth=bl}wpD_Q*V>VeSw$2No&D6UW)(4+_p$2^Gjm-8Sf&bQQUl^6@iirN( zMBS6OH{{-#%|WL}FoT^OKwtogyUGH)Yl$Pv%t@zv>kf7qqb+SGv+oF#6n}-Ly#{z( zc{YTNlv>0y)qLfjvN$n^8SA$G3>!)hQZ=$S+Q1msAaWS5aQ6%1?GBLVX{qwbmiXc* zWA#q=S`x4yzGz^jCbrn)iV{s0Cr6Z)ZtgW)BB-0h@9vIY6D)Da*i1k}vew(i7`rQ= zxNIhtX)%#ku@B*)Vr4SnIWh$D_6v}y*VD@nPv;kEN$}q#A+N0?iAnOx2S3J>cES3q zyi!6B4YBjpbr!LF@8!YUz_5^KfRG2%YbpcId`DEXYLHQMaeZO$qbxjavXYdK>FJGK zs;_lgWR}hmwz2m)_kj>n`yQduW_2QcZ)heM&hYRI)$K7Bx!#Vk?VF9Ez8un2Z1Mv2 z`;M&1po+rop|q=`!)Y924ZL-vX&16o7(#An+8N8JG)b?s23*Rj&)yRos5-BZCX*O{ z%fF<8^doo9xX6fJY;rrbHy)ST9=!J95B5+MlUGSv+NzX6 z%H_GHR0KkvCULK%^p-p&ipkCSNw}KVT3ji9e6x*`>T^9j;nLV-GS0&+yrtD|Dq`~y z4C<`509D}X%uCcN7E2+WzlM95X&f~-bE|}=tl0)&EcUmZ=k174>)7YE-_D*EhkV3O zMmP|GO4h}Vx4-XXJW_X%>TIh5HI;8?BxRu)p)0K>T6!mnOl8|kGpn*brxtfq(WREQ z@V8V3)fRZ$+?w>uh7(Q@-HuM>jw&grHIdvOJdGTQ&tZOAghU;JO{K-PDx8 zJ*??t`1#4_`VhlTO;k4fWw?f$1pg{$G6|Sf*PJ=szHWtypGUA;$JKiUtOrx(AUBTH zL%-P`4VfzMUCj+?*IKqzX3eaxr8Qk`WiR~{lX#sBKD1yDsSsi|lGwQIHT)EvWs0HN zDD)~^UrDJwp_~-*&F~SahEs?cH;CrNHD@xo1^s}C49OD9Dn}K zMHB>IX!YrU^`1;c2>u~ktq~cPtEt29@qcg^GR}!b7_((np97b(&=Ze4eV%n0eqp?E z1Y_uEMxKSd9AT~}@k<{ks`lTcMods^wkHbX%+{EZADpfRb)I=e;y;lM!pryP36PCw z7(}#U&*5o^jVD=oR2Szd_6xo=qrIQp~wIF(Gbj|Z4&d?|zZVq_yNMe;5#3h~R& z1^x^e^Yzr8jMoUTrkc#%_OEt;C9li;d_Kk^;fqx)9&7rhNNc(??I1r%+CN_Dd+6z?Qq3%qc8$x~YhsL<_HQ=nK(YAr=?HIaOeSg&yw`s%Li!A| zTf{i(UU9J_`vfG<{vc|*&(;L8)}x39d@Mu)?)FL!9Z14Q4Y{lEm#O~C_km+S2i zJ+1x?<|<^uifv3uzL(@>NrX#PdwSHOsKf-DiO>Y$;Z`!UX=S<0`bcf4xurx4LRccf z!`U1xywqka4A2Rmc;uWy*4i{gz1Lqx#Pfb}_NB~CcOr;SQba1z<4>szKI8l0UWB)< zr!s?^JR#QG6XnA2AU=yQCSm%Gf$L_8(KB%K)jF<S&>?}Xisvd8zQ2enN6 zU(&0{+z(kVp-EywA7f_;010EmN6#ocnx65Iub>~xL%*=kzwSYFKsT{?JRXHUq_i*x zI$0aKo2Cc_oQlHU+Er+B&}1fv{VBy+>uUA{w+`*`Aw)6wfz$X4Wl zX6cR;m-sGR(7m5oHc5j}sGldBSA*vogpqpUoJZ8N0^fJDJRdhKy|v_jIdvVl?nmKE z9b*OTpuw(bsMmW}`&o)3y4jv^+Vu|S&(@*&;AG`V*dN+@GeSZc!iH731iW?W(vYPS#dR~E55tI#2bIFcYT)!HmCty?Y)Z%< z)g-47ug#8L&<$cOJJoO2==QNbK@Eo1f;^S#I>eN}7R0=NOZb0PVdSte^7g*BIes(# zo{9lLedtJTz~%>j>B9u6^m?&0ox$Rz9{k=_S#LG~-~4FB(M}VGYjdKGURwu#20mQg z!%hNaoR90Umq)bwbMrG`)jHH5JNX(FPv_!Qz&#|*B%p8?r}Qpn6j)VRjQcNXYZ(mb zbK=02H9OjOwKogaY86xAuH!`KmhdgAtQ6e*W^ds=Z^x@FM{RRzcg*l{SHl{&&(2Ng z@LcsnYrq#5f6#gk`-#Y>&gf|+p2O2hA8xY5>4~5HSu8uiz3w|!a8`5N zIkI{0=}z(v$db8K`6#U|8NRB>r+sFt3Vm3BDVn_X*tWBdu1}QXQ5+q$sE%a>v_XSl zl<$1JHUL-ELgzkbAl+(PbzE4Qp_|7(#^!sNK}n#jOpB{gN^K{a8I4ZBSjkA}U9`!-qnOsJ{qn)oauWp$*PY;jjGgOCoO3`5v$pXmD7 zGUus8U3tk7r|0+T5aZGi?m-DA(mwEVj71X>L=JXt9b1 zsjs2oyl7WOtaQ|@d1K5BYSwr?RucmNRVJf`XVkvDVyQxh!fiV;9S@B1I(LB^eDz2d z96N$4#4FP7)uhx@f1?R^UyzAOHEo|foW8inXQ9xKU_}46<1Bd zXgf2C@fEL?EVVe+!uL^AaNReICPcF-F=Tff&RgT7Lq%13&1(|tr3`-G@@@bQUZzV( zjjp{T26>zVhk{wS>WP@2r#5u4DXF3-9 zL8QQqr;eu=>btCW?;*omOP7CN36a=w%&b%`=mS{gqdw&V2)iPn4-DOLr!Q4S#Wfyh z_--sur0-BtE46b%<2Fq)JE)SGn7TBT!sWpb&nshmx@r^tgX-mLz}jwRZ3|$T*_4c+ z;}&)mE;b;lJ+MvJD}-NX|B*GWu|UP_U3{?2zc%C=y3g3}N(w_Xw7GfM)u|PE*m}KH z;2XlE{&{+X=Zh&Wq$v5a-YBH#NJqRIWAvxKZO6%JK$ST1*V-Kk9lON*{^I=H@?^>@ zbjIeVkpKbi%z+(DmNk=!C~+yQmEU?JOE2vVz#+xkB0Js}Ve<5f> z6Sn7ZwQIfzpg$w_H`~#DfRk9NmFf3?g}6e{q=UT-UQ(V*s6;*2TkA)TusWdtwT!mM z`Vi!gwJWrE@&TnUx`xM}+8zlBBe>q*5C`EcC2c?wwXjB(wF$*PkuxSno9b@IbRX|l zBXN&juocN~^)v{4b|98r<@3B$`0Zks%vj=bWw?7;G`FbvuNWmoCm3oLRImDWKDOlM zRxk12|Dt<%V6|p_UgwRgVo^8$A`om8K!$Xj);nF&}TFSHdT;-XSybtWDm@ zK|Ne*qmqq);pzGj^sV&UoR}K3zg5fU0Tqshvv_$&XoSx7w-=EaAFrq^@1J?MfYLA& zdnZQPRW&Rt_Z_!SFTzDm1?n?C%DankCY!cev=|uAG|RD$%BXb@5zv}irAO7q@J1XF zmW;oioPxFJIAJY(#UioP>26&&@855palY+XW8|NGyi07KH&~&tE4*@VQRn^CwjO?# zNE$bcAEx;k@1|_X#B4QF8<4-?u+(WuQx^p_d(}t?hq;`Bo?17_(ejs*8#yVF6APaW zRnZbloX(E5p;CRm)eX94;Usfh=fV$%KndjLyX%V-!PE5kw&m9=z9&Ev2RR-WbHbn4rdA(Xz5xU;r7 zZ#4YsK%6D@LRZI@fJ$(OnZ>i^CHx(bxy(Vg!4@P+$1PG}~G z1I0{7I(8CL^XjlM@#?-7(N3m>VrjTG>wUKkk#v+}S4L!$CzZTVM6%Gt?l|THM`CUo z?>W@Rr}=jl+etO<2SU}_Y?8P<{(*L-^}C(;OI!MLLk_ciIy@WwTLz+9!l9V+?~tXr zLi&8|-_%lx5-U9wo;SM(uPkhVk&S|GJMmn}^!1VQhdfn(0bheuniPgof`=)FcZz!m z5XHTm-dX9ddDN*l>Rfs2Hr{*}2CU2(teF*c(|-Z5@~0vhtE>t+!{!uzQlhu9SF;f6 z&_0fKVyMb-0X^A{!$Coxg-mp#@;An)UkceXh>+oALZ-FKq7#Sd;+HP_2+G&f`&UX; z)^6tgBaY|ND`i?pOq3P1DnnzHF)8my&$q{tUrwCAwepQ4A#;$4fSkz!x%TT^CM-5| zFM56-obMo9?VeHaJTG{NiIqssm8?CHemuF;@bxF7ykE9S;jC3pmlym+Q8^o{u|uh^ zG0UJ;IIVp!<9PyQxAfp6*B*DVe|0FC+zimsr>IMLc){Ip^+ApB_8&nFzI?ruLoQ+N zd9_Duxrh^Tq$qnj<@2d#Ckx(LJi~I@JcB!$OTs5aS=4b&_8LR4zI_NRXFiiL`tX&) z2E`-rvJQ@H4^;ikFp_Se9^H$f-AEW98eYUBgH#`h|6MXAL`O+J<($^DGwaH@;;hfKzM)_Z4|v+nX|X08R%Zk-PzCAU^(J~hP% zz(nf4uQ0u014am&_}lOAz;EGhNUD1_*n{(}#?;=b4xVP{zdTqIyB&n;Y@;O*w|zfA z#ZVQfLzuX9bg^{J=5^RwQAIP4;fzLn@9;$nwlsgfm0Y95$rI)7==YyzC4?CC+$*-*Xr0CJ-bUT)t=sZArxUF2U5<;#AwM0=cM2) z@>-=_-x5dQt@v+v-*HRPw{8@sXVM z3MkYigvBVB(0dO_%!VY77^F3p^e@tVOssV+UMZ{a;d>Y}x=4lOhH`g{q*Zfj;>zwU z;a&ukGypBFcRciEpFZbhoda(*rm-;@BjaB5KSs%E`qfA?((=m9aT!yHk4ioip%|w=$Hd|FAr_xb? zMD`dR9eAm(6^XaPDQ0w2b&i+GctQ6G{dqlrAS%*%g_7qUu3B@j962mUe1Mvxk%YU) zP2{{J_U?`|LQL(|&Efq1c^tW%8Eq~zSLT9ESGUax?6Sgg+=$C6lF*|d znALW7T72K6LH{%(50j-D4>?YAwOTu1UtKcgRUUF@6C=k>OLy>jb5~K;i%}FbX}sKK zb2Gt@amXDNJN4$@rnxjNlxu*^S*c?-SyXyU4|u=%!t|h7-D3xBvq6_R-m@JVUklXT z)RKyFrFuccpUhX0*z9Wdq_Lr{p(WVx<1r041=P~x$<)-a7>H1Ma&Ca}8;q46LPW01Z%gL*p= z5B=&>ObN)x)B0$b)snotNxfZ^63Q;)ktY6DkOh#JwT&O(r;J~&p^+rgC?7~G$9u0` zmg~@qr)tguyNf?H>;evSSzKk7B?y1(=wL&r z2%@jdvdmtboHA?E(k8Hs`PtH8L?Tw&y1iISXoJ*|+P09{B%jj~k=}fF`u48O%Y`XX z;{Z#Bh%ltW3R7PUoSgWP>mmQ@?ycS$4ZgfM-`tq5TIGFTjT3sDKid2 zzUJp)BfqahGTB)Wk+|M|Kt-v>q&k>%YCViOd@PK)Nhm~DpSdEFI$I6kxIoDA^63z1 z@xWM=e^D(E8w^2fe|dZu$H?KbIRtoms*?C?c%-ZUF89e{3?-9Q8|lhS9bo`jKSq8^ zsy{;_nX$|0JBL}OW0JY~^Ozq2e3n^(jYyxK>frrmg+(wiMYikfW49V&Ep?=(iC*zR z|D8+$?*^I9iz4ba4o>H+(71n&`DNxRiFx3s)&hM1_&tOFIid1Kxc3szq#B)dph$%2jw z&PcY06cmP8<_(XBdC5FpEHwri{eFNEYcLe+LTKD~IaD~K#$&g<5XCiK{Gbp%lGrfWMqh5|+YYfAl6F=L5Lr#|6dAjEn*lfKH#P{(PGoaNOXCOr}d z<#-6b##fKWLu#}*)H?duVL>w##C@t{WjKJy^eFC(IfZ*KOBQuJ3Et7MJL+!F_CCJ& z9b9q;?#WI8;Ju5?N{P$muVBN8w_R^U^OPM7U99v!11F%p403diHTY8+V|j@uJm zwx3QwCmx&;=R7vCz6<J9Jn$7voN*qEI)#@&9byhDj&h*ySLRBD5o<>xk7 ztA#Z-1otTA@izm}et#%c8r^1Pw5jmyT0-dnAkkHt111DuZf4p?rE#u)$9`e? zU`GqaU;@aZ!Eiw0ozBh95dkR@ywPBsN_wAmpGR6tAQiA zI=x41ArCL-)Qsh6b0xBpD?+0t`u1ssMD6Ifo1(qjDb%v{vGe4$9_QLqSEtoTxq?P{ z{gCpzFQ-(pkW_?;y_n637LwFGS#am zC;$V&3zEG5i-T%?bh@U{A48AO&1T=^QfSXK5&CM$WII#qF$aN)$?ti7v-NN3$shQe z7k8w~B{+yuI1&IGrc*|jxYkoe-epVv(w?+)?cRa0E{St_UApfQ*IQ^sH6D{h&PA#5 zd*4S6XU?GSoWIL!XXZ$#?{Z}FTQpKjxiX{LBy2b)I)Y_@kp}G7CYYR=SAIs1_exR< z4mC%k?gXEwC;H%m9njU7qvx?xY?X$O6&gZ+{z|H3zF3MPBYcODFkASTj2#KpUR`E% zz!I;lm=1l2II2+|rwu7oKM^ClO-O)4E5YT8lNLzA;?;(nmA+)Nd4jKA1ESvxEnX?o-mcQQeVIrj&eC>^f3>GXHkE{7HH3PZQD{>OwS_x*robsaRc zFDX>b@HFd;`e4*Cv;e^yTeM4UUPV*VJ4qr90e8<-IFp*j{PJU34xWJdg4o@WWL>_^ zW_N1)4+$$l%!QQ-U#*)vd*-RN!I!j*@5;R5*H zlM6~Vwr5tdiU<-DG+0v-1|(DC`SwxiV6zZnJCd1+zGbBco@&*kA8am5zYOUZU5Z`N zBX+%pPAt2)r1*%-m|2lu8eY9DwXuEjlt}qJj2mv_Oc4}CeZ~b|7&@c#(_tJc*xrZz z!l_&LFSPm_1MFG92Q#_323W=|E@3+Dr-R-TS+UA~SRG(T_2cOS$Mjkozhx_i2=<{) zlu`a((svUlu}bYLeT%VPPVj9)OsJPgS|YQlScki$&S^nu+hM@feFD$=sE+JtEnK=! z?e!;4MaIguN>O?f^_7E+vJ$1p>L1)@x@CZR49acPrF5t5mwF=bYzZ@l>;9q@hu-W4q(2$|{XE^=JEgDCGn- zVMF!*0cSv%zXmJWFt1V3L3fmcn>I2_JO~q-j-g$zt~@fbDcYlV<~K6F7D?ODHqvqx zPe|Eh*TGV9t2+nMB*4QMQFz78=QKEjL@OR9Ghv$uQAf}Pccgj$ehdhe)EotCWHZ2X zJ(|VF&DzYaHH-NC!@t;L(jMOqJ-Pqx0fcC&$+R?X+B~WGG%-HyZ(CZeekL+jk^IHkp=45H0el!J~FeJh65T zu`5Ik_5-l9w!_rSnZQ;dc!{%-^QkdP z@#~n>B1uA_@Ld1^KmbWZK~&~zC$q^pnJiJ< zT?uI0p7Gt>#hY1dUFqf>CCy#>mX*ZC%VJ=T9ZmWSls&E_eW*71)y^ZH#<>UA^5wFV zWIIL^-se_cesVC;VY2VVSy898-#7lZ>?CB!zEsc6X8AE^Ebn}qH->xD(IK16xNRID z0w>>KNo(dQj(nn_99YKw6@T*fI#2w&^kc$pgXk*h^c<=Z$v$yF9Gmu0;@cLlwo&wL z=_b)U%XfLJZehodi&cL3QkhJYbH!>4bsi0aM1pMVmYFM+vc{?;8vgG9Vm zi%xDAQSDRSU06BhJ$5EV0N>U2D z;?gjYkOULSf7AFgCc*|{Xdf>nC)=EmmXd2#(k@t<)xRT6u+CuDd=Y5&<+IDsp2^d0 zk)2Q5(IfEECur9~Xny6L4O9+!h1+Eye05wisAA4*S7=XhWnB(8*9bJ9%p@)4Xt_%AiA@7GzObUe$N@KS+kSXdvd6B4JD*i zTUxhiMWlr3csko*Z(}9-fGx1L6T#Eb0e8u*WNspcV$KoD_Rr?`T{@0RYABcbD|TY^ z=_U7Z{O$v6jgfFS@jeLc)Qc{CdeArA5kD{SNS0+gE8RBceDs&>kSi%pT!t|9Y1s&1 zbQ=QXKFug5d+$2-eDx;Z9B`+g#ZZR5^gMTocU&tOtZK)65j@1pz921)Oj!V_)a6o` z5y#4pzG7p(h-Be|`Nv%y@vGiLnurK9Hro(7Xav0mM>90kl=L;fvSHm`e%$57=tn2g zr&lDA+H#_Jsg@+|AUSp(pRW?O?k(ssuos;r1&6PTgcCI&>>A1?Oy}TM7Ju;rae6*>oIyT&0Nfk`H<@dZWEIdDvOiZ{`)MpC5edW z+MZi(5if3E@dj+5f!q4G#4orB9fQT7XrMfeHWj0{%PFCG#^;iw8N5c{&BQ0~V`88M zZjBi+ji2xuej;fWB%HD|Cx@)8T=I)0)T}Ncyt!3sU37_sZJv1#9hmLmT9I@cLOlBX;A2pJm|u_40_eJe2*WdoaM`UTM5-yTz0 zl+m4B#+n_+Ss)8+uD0S;ZX(7<<_R{<+Dy8&AC@7q*;A0TJD&(iKVw1X;zfMBM%Jh7 zBe?hN$@uC_@eywZKhy2JBb(>Ni$SS*G(Qk&PPm&qBeH-I`_(JVSe47V(u)br@*K%C zl)d@`K0VfwblWyOF~C>)fO8t5L}KWxi%DGgDGOE~=FdYO41Y#8ZR;FPXSX_w{OqQo zc|aSAkFjz7Mz$XmujI~S88x5-y;}$1D_(JDt>H8%G>fOXq>xFAS;~%ifAHUp8KmgE zaTAZqzP*}?&@5w4VG8Ln%UQNNmjn6sgfwqM^vE&v@86Blk)D5((5wjnG65;dBPS+2L?yrC;hC0ETCN4_}9=KUgCQa;#ny+Me4rIY>3^@mVGBl&X5gq zbRt~HI9GR=x;DWgVge73jik4?XB zM8>~mEBLB(3}eUiW^#Yo98`XPS;Iu#jbGqjLwN?(vI*+<-)v(|yx3RwJjjSnvS3{$ z>3wFrzzcKjPp8i4Bp$g%F>6@*>vZ;I>q#;AqH}M@n27`E)u{=sr7twA$R=a=QnoFh z!^giTkmuE%h{0odS|WSGJ?t7B2QN^(OZ=$DXgihU=8z*}PF|tpyRBazR?hfIUT7Ox z6fR83$0{rgCr?*AFR4BnDuNarm_g~$HEh_Q$H7z)k1e$KawwK-rxny-@8OM;lLOAO z=%^i4xycls+{eyD@uc&LBD7sk?s8qm^3BKiR??4j8Ptt%KM_Z)q>mASIU}!>+`K}G zf=VYTw?rb5yl`kbm??J+q<@!YjpXZWsI2F_XAnNeV_E&~tNfZ6PJXjdeD?Yd{ObKbVF6bBhIiFcB{ z6AXGOgQU#D+?9YZUwlMNu+z#bP8pTOVzVY5kv(m66xPqNY$A)$79w&*1mfW$n{qY! z_}BGO(uYVGzh3fpYAM<3WwSd|vZ`2O#Hr_#Y2UE7{iF05Ka!8{Z*R1vjibA?am4F6 z{onyk984fv(j10_1=G^c)tKx5+Vvk&W_>0Fse737%4ckq)Hvmx#xQMUJ6f8?kg{do znbaA#-^MeY^yF-s&ib{R_%=pt2%QB!VJ(RqJVqQqn$W_}0Y`C6$vw21m2(&Io21h^ z;m}n!5*)|)siWvE5mwj6MwD`ky;dgTP`=ox*;&$8$bw{*u?kGFlsS@}iyNL^&Js16 zMMZ(wah5J3+!soeXA#o$&OaoT9ws(;HVwne!=>?lUaK3qc8S;&`8l~_9~V(pEnerR zu5INYZQs>huI-7FM10h3s9JkLt-*^On4M`A3hElUqQpcn3Fn$o*S8iiqjv zfw!araF98@mg48)N8LH)yJCwPWKWI?d9csQWMMpsjN{@s6fbj~<-gA4lilGA`o~mm zePA%1#ip(8#x*|j{2htmWLmE57ZpI;*3D=dASu_*6Ex+zVrv)6#;pr~lC*I}k`G*X zY1Y26)KV7l_9AKPR^s;@W_N`%K5cu_xknF)2gGd_}AXMA!&oyv48RKm;1@|?oZU%`+4csNLp$RV0EUrh??bvd88i86aiGy z9G1)87wIeb;_Dc;=Xl^Y{t>3Mv%tUX5GOV+VT(yG0;76M^l~&ICW#zhy^s|%m$NZI z_G}s0jl27WpvA)@Q4FRVH0@&wK4Q=UE2Otve$!_qGI`1zp!Y*K4QJb^1xkv7~jbP`=nXC z`SJ`&8)8e*u-EzIt_X&;X~_Fq7S+YWJ2&YB`-&xM(#MD9p&rIfch7FBlG0D+@I^=F z^Yy2TSsSm%W76aF4s)PcozN^!4K*?zR7)B`OBtU`&zkf!L{)f}Ws;kGh}Fy2iPoa0 z)WQ?LuqH%H`oXTz9f=Y>GUS-lHFK5Hb4Se?9E=ZBXBi5N_`Ze6%C zxskO{=Zsv(^ zXdwO}GB`L(-qrASM0RM$fS%zvNe5{zp>>8DnWW3OstxY-L*NZ!5S8!xmOp-9%zJ+( z;4tuUZk;rU`$t6JC_S^*DQaOLNa)IMuYSwV-|k_X*EpVhbprQHj>2C)H50=zZP#+P ze()3@Zx6%JX(W$-_axmVL6x7eVQF=kPr};8Z2ILV<|WF4N0E31i$Rx{oFO@3tuffT zN?Q}7uOvq%_PL~&TjAF40S1pB&ifP1Sor>IVvdPmc8d(i?Y(I4V|&WmK{MJ5L?}AS zhL8Wn?3glE8~X9gE4R?Uoot?XUXoB5I8H9*;I_^DdzL+2Cl6r2@b-ksplV*Ufo+TD z@a99`5bx2K@WG>b`k7l~!!&cFhx8e4GSST5%gkq9W%fQ>j{1+{KkrYVtL)C}QEPjg zUHhimASLx6JRAD>>@U~MJUVoilJzGnvh6|fM_3ys%*QbQ$7v5nW(Xxu)fw2Q5j7jpiX`3#XiKuEg zx}5oI_p|Vz0k0nSare~0jO%EL#j=0$-f~m+n0DZqH*cq-r>wZ27QahK+rE?-@dEpF zrz3gIhSB%Q7kH$j^QjGNq&~z*t`cEOn=l*gj+(>9T$mr<`x*PZhT%EtKK?yc^VqeZ zB6S~`b6(}2bMx4eVM&j!o+PB1%EFBegS)!1{;w^RIyI-+J zvwmjp&;R44$83MU*V6 zKK@W|eC0b0_;(wB+xUNqDJm!>Bc&Wqe+l=LB+2F%F<>t&G;4iCjfmGdMx)qVL~(5* zrB$mzJUfc^{>h4+$^3E1OBP%D$!^{Oc$&&W+3s(dzV--zZOJ3F;s6JVJV>z!V?rOv z)t@M#t4VpJxQKWd>_(Vu-emMFtCWS~_&7Gr-AsyQFrJ+TGT@&3xx05Tk$y71H-6LC z=94N}q&&2WpWgU{O=S@n+Kl9(soiMdZhN-pr}VWZW+GmRSC8h=a=OngKSo97O5S>U z4tw=2al7p)p6lgB3lH%EY_NnPPGlc9_HoM=u40X}jX24ptBGHxc3znKc46v>Py%F8 zCieTy_)8xbJ+vpoeJ_=S*f7WD$>z&*nf5^{VYl7Oh`Yxd)5Dy%hiQZ5yNov_m#enn z&-yPuWbTa3{8>GU$DSD@3p(=t?B{r~p3vM%A_;bFKFZr$19)eYv>)Q4X*caI1oNFpU-zuScGtFID0q7Nf3xjxRnsG+J*^t^*?-Im8OiR`d;kzC6A zma!{VQqF8GA!6ucIz>g&E$CF8i^ycCkx8s(XPG(52AgI$9{!m{hxDv04ClU)&5Vl^ zCL%KHkFI3q+QV#3v!-dETj zEIOP%%mT4-?ZTs9KgO`0Esc3QPk+|n3t7OD#m#aNK=UQ#RjD!5{Om6p%6A&0N4|S? zI)_@^OOK)bdE<^2#xH5@!61unRP1K$UyCKeGL}Tyn0)XZX9Ns+MKPboPF}k z{!470{3H@*y~*cuk25DVi1BZ}%=q909ODrJd7y_uSxTid8t9>1jqJn zVv|JSES$faxRgAyWVkeuQQITDBQ3i3mG;*atL+O(+ak8VM|W~}$C7tQqOfG5q{*IV z9VU`a%`Ak`Z@5~&&4jW)dpGg>w` zjt+Jq$f?fJLA#n3wVTh;#lNy*?h1a9d7f65Qf+lp#xx6gZu)l*W=I=rR{eOCz)s$T zbaiI-JF_^PEDPnvRnW6a-S{7VqZu}CAg?{!mQ64HgJmnWup*<@hClyD(hr$PKHv`h z=^ze36UKC>jX2i$G-g1ae*$?&?m;K{n#*$}B}Fymm6~W=@6d!_5t@%L_*v3-F6P_# zCiHlCJfm;xO*?aQaq5tkuSMF)e02B!WA8h_s=BtV2OM9bdgtlo}#PM7>PLu`V z_@@0hdMpxw(k__u@ia^uZwK=p5~8?FjAXn{JYLY2 z>mOf6@{L`Xz9kx^eJ8}8@he!&!fdk*1!h`ZEJf-n+7m`Zia~Q)n zbc2*Cnb=*csprpxTy`Ex#0+5KIT_{oA0MkZXwqmZjueTzgnR&uyF*(VMGhyw5BxE@|m%@(Ya?_%Q*Te~=BrTC_s0rCP~b-;gaiehfeRCYaL?Zdx5E-}yVw$DEHg6F;$+FFvN88Q z9!EdHDc=fM&6*8YFBfK0$%-Nn**0^ZD*~Fm7k`0c5z>g)98WjeNz~+01gFgc7u5B!-arL*{;V zo+N5&bTR#>Wf*R63=6V=|N7R87x8uBd+^(_2RJ|g zb};YLwBZ`)Fa87p>$W1WnxhcaL^5kLG}Q^bv#(c|3X2(TwRDZH;Fz6FwtRXr^7!*!w{e&<$pxQHH*aE; zG=1`gf7vJN$l9r*OhXCfZdxt=YEthY@?I>?`4_;*!wWV8O<=Fn4F&w1U|7@M)|hY! z(YH?EgY9wX=JYOHCQZZpmO_~{?*S7N$cezUt5HZO>H;Y%XE-_bg|W60V5=R_kV zzY>osRx8r{3PL$3pST&2re-_l3NI3WMPu&{mNk)1wgO%kcA z=>%J2CTGc}Za&5K+OO$7g6@xRA@Rzu_~s~w+rG}2@Wot8$8?3NP|n?E69BJVB+!iO z!ELfr%P1Ei+XD)mpM^0_jPhN@*>r#pB$!uOhO9yfK?&K-> zg_7l4t^tQhlQ5ETqJ2LN$WzYg?P9z)xgOzue*ehRK@7nj+-p(RseChFDK6a;|U3)r&W=`a&&SKmHW% z1UcMUVf6CGN(p*7gzwyhD+ELmb;n`V1e!$=jIS?DL0RAqY&{;0OBpH{IRA4@v+adJ zT9TsONRjQ^tveT}qB-^DJ-_aq>a?pIGmc`#x91R^O;)a%3ubO4K-O0o7IJTz2zlL6 zI{G`AoMhQOJaY;c!zo`?Aq9B@CREkfhv+RKc!OjmWFRM#Y&?2~C=VKsLH1T~CEzT@ zYO%b>aYzciOH=a8@TLUGz2aV&I&C0ySfSTqiA&j^@?tXAsB6t6u6PeQ7}{%cevL;` zFy;UD-vnIVfe~YCXPu@tZF{z?(kVzp@Zl3kW+jj~<$*>UXCc-n071Saf-vc3HtT)3 zPo+V&jS{*^)T1HlXp5tCDdM-T7e@zrDS#So|)+hg`9TbK{-2W?e3QM}pqSTDXR zCCf4avFFd@VPQSJW|~-*Z9Y{;W8lbW_?^0oEHwx8HaEstd(~DSVSi>-U}j<(lHzGb z)`KQy_QNsA)fE%WX-ds%(7$gJn2%3Uaeo(n*~PKyQZZyleSmkI0n<)VSpC@rbR9Jv zX3m4)%H(oNJ&i6kj93+Eiq1DSOsNL6H`6y zD3hp4z_kU~`YY!y7$jF^KfpO}f5c`Kux)6r*{v5Ds;u%9Nyu@{TSulaO^D;-laUd{ zs<2YRmyZ8|i@r20Y35l@(6jKtMMT7B;CO*0rn?x@>|2dvTMy_Cw1Td-DzsVx$LFUj z$W#_3-^2OiByuHIBC|me`UbktV3miQ6hY65r$|oEKx$4E2{!7m^mK#$U@JIkNO9j4 z)K$NS1fM@}Cb|=H8swp_LpdIj7Non>Qv0@qb8j57Q&W*aP&qPQ0;Ut(Fl@Ff6jS23 zW;~0~dy%+~!5GO{d#IU;X#LSDQsa5@InIh@b@E9Z%^C`Qrx94cW;kj>f{=9k5<&u* z{NnZM8yjQ%Ag#9FXM(7qBpn6$1!UoBV6eNB2p`bnI_1rVDOL<-MvF|}F2z>S_okL+ zdBXaoc@uuGLC*c#hzq-qJ6Q^pp|C}NJ7Z{a$ZDFU1zdyRUtA0_vS_+q)eXAVBj7m9 z0b|`QAn+Q`)X!$8HUmqj+^3VAFXe@4|$wPa_0vc}{!^uRMQXG;Z5&>7L{7N2t z*9br}8O;g$nEUIunBZ;;r)SlG^^Y$h*>|VNthv&B5=MNmoKko)=+#=Tu#VNd*=dNp zlStT_i37@67KeCa-);IIP=nol(KwiNp=8Qc#>$oPxOwnT zTsVCb$M2*=(#%;zJRIR*3Cn@3*i>ibiV&+l+Zq{H*CgG*t$+wzy_<=y@-nOl5#}V} z{x_hqkg_P{l_+8SsX%gKk6Hrhl%u+FFB^$fG$l88!Vp6l&Id1!7sB_eQT!wdY2hKb zQ>cWV=A+=~H3>^6nZZ!Gd&{#?#!8T|YuLT|0K%ngA!qM_B@0|($n}RDD;I@iM%;zN zxD%R+YuVaxojd}zlrc3H*v{3dcpMRh(4a>&p(JsG6&r@V4WTbrhw}7CNaOl9nZ|-D z8rslRSB5exK>z-jh^P4P_K&o{zd-XFH>*&V#mph~FN6Nvg-bVL;9Y8uxr^K|g8H>; zWo78Xl0tEcmNychE3Fr2$qhRYvxSW{58!{~ERO9Az|-NMVDv~E%zP%DksstuH`q&9 zd%YJ%a*;OYLl4+-d=M6S&C^B8aMI4n(loA~43ShW%(`}!hAIQc7D%L>&vVcy%SlGU zjdKWP8AO>Zy~9*x}c`81eX$VDX(H8O? zyHoB#os)!0H@d#&UBLD=hw=M~G^AQB!Z#nf;yt%MqB2aeRzUOC2nLq~n!nt+fLUlt z6t})GvOyc_YfvLW23gJ>WZWfq^0^~oSJ7TH*w1QJ|3j71ku_VLmkcN)~tqJ%(4s8HaXEjh&Xc+ zKOeb;(1Kp*+;=k8|1bf=%&EQi>JI;g=k^R}Ua=8pwLZeov0m6R*HQ#@3A=AB&m{qW z6V~s&kL~w*ViNU;#~I{8CFM3{Gwz|g^8^ySM#0N~S?V?sa%-U31d>C>V#&@AMF|s? zj?VjyfaXd>N*H7Ms%hva*%gYFX-sPPvt%m>sgl$oU$_Lr$YRm#M%D$v<_E`qg%89npq|?Q1ZKQ4ZC&+Lvrj=44EDRJtE)aLk!92g3#$VF)#k8!B^D&L#5wf#Wn4aW6<2ROM&-ygm^*10mW|YU#oWa{S@o2<_h|@@NJd>f8;grH zVXNXIJ0$P{4&5k(@w5*Kz}sTBO*h1ZQ{PX}5M**sf-(@T+qW^Di@@+(*nB31axb2+ zb{-B-!_EXhxL-c&jhSJ%bS;kNIV_{K7=jVwM`DPzG1>akd z2~OAD*@TVz{IJ2N1U6$w!&FxZT0)KZ>oywkHWo$U?3ss1>uLt|S!=OobT8Obv-8zs z-0ODyy1!QI-%yXBeNzc40pGFft^nbkge;14PT8$U$V42P_=3P;#>X{<_XH5 z-69k0!~y*0KoVp}F2zu5eY{__9jngE5NI~X^6l=(y-iuQZpyG5X~nUCRTe=f5qe-N z*56h|H*GoC+e#qhN&-s7$-?6EHirGd+DHSs^3v3u{kxN!!u=>jY3fZ};r;LrD?yiM z_hVo`ZPA?5S&YokS>lS zFF7HONt;%QOiIVFbgPkNvw39fW|nq^hPoW*XUcA*-@&fs8*#JR0=0ubz^`*m(1*#5 z+NbfzIll_uQFi2Fz9B|_`yJk~=|yI)&`bZyCIB(j$PV}ew@>cFO7B$US9XR%Z%Y`y z_bvY8td9|f|CFw}&^}CVJ!L{D@H!6ucn)#h*hftr;5v#iNna)CDw5^bAQ&Z4`}_7y z1O@~kLYxvKb}sN5%ZdwiX##Q_Ny;eoLQT-`&Q{=LmZ>O_Z%w%(f!NTrm6P4hSd@UY z-zk4Y6iOq<mr6-$wJhoH{3&+xO@T$u1PzkD#@c_qn< zM|%0S05m^F^ttnhX7NmBBP(ZK_>fqA7Lp^Q;CDO_nd%NOv@jvitp18QhpcOrK2SC= z$4E;J0?k6~_&08KJkh7;uy@N5{Pg=nB&U7|& zgrZSj(R|(*y$|Opk@3NnNc3?Z3Ju9Rbc+l|WM?BNTMmY^C#zx@7wXW|S-MRu_FtZR zBZ)~g!A$f&j{UxsDD9?!zK&MtOYK@CO%>?$6u>hf0n19XNPnC?dKA6^kC-r1h18^V z_+pYJX4oq7Gj+&3zaAI;qVQSzSp57UC8zAQUXlFcv(IDa$J|1s?`iyent*c;Bk0)< zgBRJa79_@v{LSGWiSP}aA=Z)sG83c$Q`s2#uY7~^$lb9xqSIwJn zj5=i9XM!L!oRuC5FlO??(#`~$T)%Y`Uv!bCDW0(rENO17B2G=-4%fUb#}N_&S<(JD zy>$y&vKlDTbH&`xC&83)Lib9p1EdHN_EO;*x{>}zSvddYXE=5$634Upv08Tp#=BW# z;Ip-16VSYiK=Wgou1<#Mhl?@av^&&9>!IdjQuc`iq-tuiNOU3CpB#e3BVXeyu5%0Y zhGONOMX>A~hs=w|@W;m+;j1zY`t)p=(Jv2|jz2(JCpC1ju!SvI)N%y&g#fXWgcLec z+KaB|C?jBBlNHBm^hjLri$|^T6xe!A!J3ckp(({!vCTukHYcgDEQKfRI~ng85)I$rCduie*Vt>}LwW7R* zIIF+gTok_`JU`Wx@F3Vr93V;{2$*@m#%m^4%(8+p6P*J8TC|l!AvHP_e;x=&WuL*& zx3edqQk7#oB_T^PC?`Vys2zK8F)$N;RTh}|(HvIIT4IELH;7fSsw#}03fDrAN$jk# z2MHM}x-itCCxv8pNJ-1UxQ`CCy;(uqe7t{u{-0@qe}U#VzEv#mEPR5ZaBl`^*WsHc z2gL>Ju-p+aV5U6a(nBbYbP%xz>dEemE|W$Pe%T*}E~d^f zcX7oS7ZcPLJY`nBqz$O>{B+4`uBlGO9e?WE_Mpy?sU~{nQaAB(0xpDghQs1_F=`OC ze^kjfU~m_cA%=oFdGt5cLU)#Uyc%E>2INAd(Htmf=SHmeQ-sX$iJ0~69Jq-y)42Z< z_8*Lc?AQDjOMM$PLQGI@utla zMf%MzhWNsfh1*sicMkE__G9XnY{pQ zBGU8{z?xSW_7YeBYacgR$xLM;0}t~~-2U1np{qL1+(OX>y7 zh|}mT9hG6*@xyO-$?hzKij^Vynwe8Gu{+eo7({r-BC(ztl%K4@Lb6zFdP#Kv8f=xQ9y{b|9#0e^A5}OD3A~*aUz??bOjfY?ay|-1?LH!Haeygo`stO*Qm0=pjnt zh*jpJikjZ31@#c`Nli#)Rc2iUb3&mto3?yranqBZ;7MT}dN?hG+l1-(^}T*9V}6qW zuz4@9`&o@5GQwkae}*mDPN?qfhF_Krq6Q|l+jwJ|x+9sfTqIQ$g<{tSYiTm3hD?WL z*g4+}1NCX1@VZ01(cj-5XdVSWNX6a4*9p#;#TrdOdM>D|nSZ>z?W-1IsOVcPpvlIs}<#cQYSe=8$tn=8p)f@Yd zL?NQY5UyV?!=zbm>_ZH081W05X^q8kIR5ibIB_k8tT|=4PMCyI!|mZZP)}r_B1NE5 zj0?!R)GIj7a{6O8^U>35E{1fAf=Xc|eoS-1Shsto(x>c7(>1o72KyHS@`lDlQ91XSU%U}JY8yt6 za=<*>7RaN`NjJYJ3q#}se{9|9i^`g=tZ-<866!O_wxIb;cn00A9cIAG%>?f`)#K7O zUqoh)L&wa5j4K&xjyLbH?dMy+5*7)9_d;wmnbZbD zVfN8Rd~Dy7QbsRAC9S{pSAQ4Axg0#YauR2Dc_VPZx0vGEAJeVm-h9V}S8(mfF|7Oi z1R7`m2#;}YST>X;@-OXyz7^RthS~Eeb`bcDRUR!G;{@GdX+9rKpXd2BQ2L-I0tc4; zfU_ah2pPB%yEw*;q9o1Rjd4ug<;J1l@)~?~C>OV?`eFVKmT*$f-{cMM$4jUG$~Rcz z`rrWk&mH2T_&$`~K4evcK4zEhz_$dNb99|qL9raRyZ{YSaD<8+L7TQR!58VbcHx_| zC5U~Tj!xA@1adtw%GDaiP zDvu(h-U5?hTUj>?Z>b(9& zLDEiWj&?>N)CMEy)JbfeOBoyM#c-ZPu=AB*HuqOoO(F+n0tj|=p?uI^O<9Oet@$6a z(24xx`1-xry)6)-ecr`8V|rq+a;gYu?&7ux4m0Oqxh=sRN)`$8kmz0)5*WRD18cqy zL|JJPva7|B)1@~&7rz4sR@n5dFGFV&YZ_}QLZ|y*efD{>l!}bJd;(|J??kfa2AXlY z;9W}^3kzeJa8A#!hwW9BgN77u{J8lf_FgFlCalNjldS$rpqc%S$G-c=KE_2y$5+Z^BU-XUCnz zgDZ!zcpZsOgBHMH>MX2s>x8I1KjP~3NL-Wkz|<)txW?w%yJsrAmwbt1ryt{DNt18N zxRL!~^$bC60-ASY>ed{RQzl{3Cm%2lmuOeRyXl(CkQ?oXdq01MO)>o;X6{CuW(u_G zSuJsz$?GLMk!0)%>H2bLKe+~~lN>9kEP(F&1 zJlVcy5mT%WMf=hC_~0@)uo6qVP1mAMHrery4}FCbzVr@=QH67lhe*%uLc*B^mi_Sw zJO&!Tv=u_!9?-ndvJ6_;H?VgN*X&OuQLa4*gWvlQ^IZ(6%S;aw_PGu>QOw#Dw?3N` zI{0V}N)m!abLHv_Vn|{&|30Okw%t=Bb+|`58fTBM=j5uZ#J>Dv)}(I zpjjvpsn3tW!-N7n&KAlprJ-A`wrC7zh(k+l`FGh6hjl{%}W(0I;G(eJLbpy*jSQ`Cm9U$Ikf+rP^&f?mc z)A(iQ72GfE2g~W>@b1?mq3`!AE?jkj$!NisPOQFy78JPX) zL=5YZkHo`2;=<7@_&IYhCb_7qd%ug9QVx;rEk6rrR@9b{z={xWpECOcG0iiVL= zl8}&;hJ-i)_QD7h#a_mDJ079IY8-WCmt%o>H}ql<+gMwH`T{Om`TQs6)S{cB3}j>( zbg+~x;?Lc;>8vD4P^C5nH}?7BVN4n_o=;TO)Cs4F#C@{*Sf*Pl z-3$HuYeJD1E5@?(%-96Ve5gXwa3CCMa^gnDMt?3;I^NhO0?>SjY>~zL@z{0&1~}W{ zox!Ripj(VA@OmyHDj95ukqs`KgzYyGhkU`CNR+^A~Ji^%& zLm6m_chiKi7E3>=V=cs-9p1ZEKyxo@w+(kP!vyt0{dh=(_6XmaK< zPKGkjB}-9Ssu5XaP*~X5!GV$-0|pwyfV$C&&!i0gexO;9rYJ+w)&02P8;0!>I`Ep| zhEW{jtkh}t*zQ9o%!@SaN=ZV-<5U!@_J;=BP>(F5jyLva$`Tb9=Hk(lY%*DdC|_v3 zdgR^QiR-K|+8LsV3EwQjm{I0%QDZyAk+ngyED@ub?3C8$Qvc*S;_qCI7!OO3PAk&#G-rhGqWsq(p! zYvGHCNpq~!9ENFL{V~L{*Rx6+;X0n}D|9Pgz&ru6QwV&GPv~L9y-vsWq%!P3v2Zf$do+9@4G|TkFB8TFcJ&K_GMd1{@1Zt%XKyXR*0gs za9sZSGhBru%5CRy&M|m@_w}0=?gD} zvEUideC#aEc}rmKOcP*wm8f+U7-f{H$tuI6G6Ke9=VGLGAqG_XVB?KC#M1CsS;rcy zz8DS@>N(e@gd_Mzg5myM5vMQ|pKSjSPMSTT+d-dx;Y2!qNdN(PH*oluKd|4Mx}awB z;Hi@VZDe3$ur6jV7>AkTZDB}O9PzKecdO^IE1A^XdRQc_^T2#6=1&+%c@|A5Jdd4P z4k_qg*Ayfm^zc@Aht$ER+6mvU_QH^UYEUkJ9izEoLqArzL!&=zjzFa z?y{;ggXV;;A7c?cU|fvko{h_Y9cXSo`wo8!K=aAt*!=D<2p_Nn&XdRE*M%LhrW+_{ zR{01)muXrVkcfPvv2b$ehasjaA|upS-e$`AHI}my?!XlUKS)7JjU*H;SQTk+iBZ&e z6{KR3Z8s)pSPhyMdKw?EjD(c)7|Qp&3s0qTWS)4D(X7X~trAJ< z7KpL{06+jqL_t)(#}Sj1hlpNNG1{CgXqht9X2l>ZPad6QlwoP2LPmWpir)li=I3fl zX&@YV6x;tyKv1?SMtr;kBaOIUg5rN|YX+LP;^Mx;*c?hpiLc(r>_zU+E2rD;pNn2< z^2Rf98hHOHt{3;iglSF~KnbT_lC2}CmTxr@ycH{dz{)6ZociMuj)fINdF*lwpWu#< zb%Swm2Z7!DHIQ(cim794Fw}w)7?R0NK=b7cT*gSO{p}r$9%>9dQO6=n`{5bHoZo>3 zdrHy6bt)zbK(l81K(pvf8ZudlafR#Ul{nNe8p?ym>{ueFUsUYYz8U0L_P^ zrQx_>8B9xpQT-?gXTpjgXEO=|h7QA|VU$y&w1+$;FIpY4>r27lYuFmvR0;5Hjr-W2B1(Cfhcx zKr5fzMk+y!%jJVHVw5f191TQe#xG8V9iOIP0GEwBd-fo(Ko-dcqZq*Hk)1{0qrERx zKE$It7qEHZR`{2a;<;x!(#ExJ*Q z@-;&v4JO30d~p5z6>RUJ*64;WW78mQC5q>Eg zk7CmClx0V++-jcEI3y0YF9abfs}Y4VhOnW-`pIB_s$E$WY=BPS4>-woO7b{IQyFa{W@L#JC2#Nzg0-Ij|u5J(r^ z*&Ff67&^|H%RM{oRK)wA$MNM~!@Ft(G>4CU&S-uaUKJK=Wk|V7!?Q#C@LiZG9N(Qx zi3?}+>y(bL9oul)`!>8KXW*mh7VzvRP0hVH6!lc3+>0u7R8^aRW&-Lmt?w5zr1jNR zXe@n`XKpy& zwNq|lH!^$QKG6KiyCwRgm^c^P%s49#;9}oI17>dsH2=Kh52TOX0MChTSn00$`U8mq zdYWVGU4yfS&SP_kE{Pe-Fm?8D0{5?z?0AFSxA_Qlq{DqhWn|Z7y&z>h28%t-prbA; z(r9jheJI;QIfYWLV;pjdJ>*kn=#sg6- zK?Znz=&#s(=mB;G0VC&*fRcn9R2#+NN*Y2)EVUT(XjIA4O*=~0^wnbE#y%v*giVV^ z^XsHD_yAF2rXnv4k3%kE^QQ+9QKt=YLu4u(t$@C_`Dgv6%m65ru1CI|qX*3~7 zXJnIf7K%fEh9R+$i*WZ@@Gy{|Mj=h4%JcB}%zpSq)FZl!WBWie7wGi_jBbXfvafc5 zMt^5mF|d^r+EfOLluH}f_8I3LL{Yp?%_^a zBdVq3AgkI73hIU!;N%EfE;LOPsgK40?X}B?Hb67kqMpu}V!FUoVCAnrjR zZrvcD9YGzs-os!jUkF)gOP~ zCbVBGp!r4$J%SRElwOROr;_Mrtp}wZl5Hd}ItH4B#YZ(#yuZTXQ}?j?mKqlC`~+iN zEnw5GEK(lwlR`ve+%>XLOUx&W#yAsQF7n6>YDu^V+Ko>Mz@A22P9yRLun%*L(|<`f z@C{m}zL2?$Kpa^8JC1WPn?8IsKAUQZ@dK&1{@Smzq?+bRX=lI3o+DSVo%pk5nB5c)XHz|IgXb%lY~JNRLq1+^)e2yNiP zj+HJX3jM{6ZuY66IQPZpxK?F~BC{#@$kP=4^px7aAIEtvSxo5gez^6KVVwB-7JTy) zYy~tQKTn{!0s}`{6KGb0JY(nLNI%3T?8J_Pp*Z<~$;$;RFik%Z zN*MuIx-lLmd?wtdkHHvSj?aHL(p;H|hP1QTvLTdmGBW75@?%VfUX?n zOadJ^9)TaPOJOz*rAOKIL4V5AyjnV=d5>=bG?Qz|M840iOE|Ro9(r&4K_vI(DA#HQ z!B|iD%%N}2!uLi#^w)j@-6v-$*Asyq8IG7cca+F*SesP_LY(mC(iy_Kst8Y0sH>Zl ziToN$+|d-SqfKCY3QXtP@)*S2xs9{GT|tt*C(In}G0(#o(gHj7#pOaX(0qmFa{);x z?(KqJ}ACa&91<##H8YUY(KPL?-X#}pRDIEVK z1JS9J;xllBDU(#LtVYq1NS5@8>J#!gKTZTCxGEalvKKkNRDsu29v03tN|O5PWsyb z&5YYBa+}8B8wJ!pw!pWmr@=`8JUd*N2|BQac(8XJE|5s`OT}cYogo2D>fP>HeFoA~ zKY*#@V2t9}qCx}D9#UOe;;`xz`W8jPdoO`$$d_@L+e8|iysf;n1g zxE4!2`pz=YU{#8m05mI7s)Q_NpNu`E_!E9Et2T>j}MiDu({qfHG!=Ohp zR?p^+)L4oFR;I__^2VVs6?COX)QEA0DBZad7j72eVub^~-m?%RtSAN1YMgn_XxogS7Pm?GYMRvZn9fo>((4^8FPC7lFR~^6lZolDQ8%dN5yDFM1|4h(9gV_kO z(C5NNLl)ZIDF@f;97V~khFAm~*pENH{}b1BSyeE@11n}*La3~6HQKj1cw?2|@xb^p zDhRi4gyTU{Ia0bC!O_DFg9rA5gHDqyTyqn)`7&QLlu#-&@h&bNIfpBh+6m8KMWVR} ztBO6~HQWYPDm>4>d#-))B>sKIO{ ziA$KbuMlI`aN$0dW@kDq>uugIHJXKB>++66_ z2v5|*fsGlK1~F52vvL7+rQ z45A~VaXFAIq(RFt_Z2{MoA18#LTL4NF8ZaWcEh+swl+9`>zY7b7<&fO?{z;+K1pc+$-ZBPho(c6eX(*H;or z*@y{cNDK&tpZxI;ZB0P)(JwdPl+0{cIy>RZ$%ZY*qS;wv^bzbn6ODtx^{{n!fbD4N z$=lO(MK%}F`?uiSF+Xh2poz~CG8e|#!jfzZaj`c6nw#!%BU$ExwsCr>5AI$%iJ!bf zk=m&bR0n!s_To8kF;Rea=|hnj;5Wy~icgnMJM8?6M2}h`ELN4YqEd>A8h_A03jK&aQJx@Mv%$eTk0QsP+PUw z|4%@3K57FuVbi{A&!jWR(pIK)#+$SQb-x>^VHX*df(lhzXwBGw^<#8kuP=o1|Io$- zXnoW!>^~KawSMYYwS)35hAMBUy-U^;+53Sssy%w~G4^ISknJ`Pqn(V~ETme_ybL87 z5xBeMQ_6Lf;Z99o(HJ*|rLuPY)J0>QxG=^E^FSxz=bO&rEk}6yHzuBu1fLC?aO@~q zZXDCT`otaYc+jMge|h!OS;Z30blly%1P9IqVE_G6(KjSa#?oCuti~+=KMq~NG<*;ETS#3-p{Li0aNoXRn#SNe~U@WFG7PK=W6SSFf zn9B;c;1dLzeNbpInFhM^@x>4_x~pmEQ!OR}NF>Dn>O4?JJ=-YJ{n#B)gn;Un`!UE= z;~%;oFW;f2cbfWe`sp8_a2j>@Bf5U`ww9}@jaG_s}ay^9_Kr{-nV4>SvL9|3+rI^+DU zXe^?1#=ch48CbN7BocSLZ!G}LcX4&M4-Rb!ht`G-a2435Dy=5fRzUN$o28=9cq-io z2y!g@EDQ!8Er;t=PfXD!lb&(H8w1S*w@@DmzavL+_TWVXQ%Aq7uHzovKp&I3D(VbB zrA~Yq6WJVK#Q9WomhOoz@(NI9`4x$aFKj|g!ncGm>Dga;n0lc9(6Q7UHl<8f$E2Kw z^t-6KvlUY|1>$08CMBLEAlAhW!{^V$qD}9?md8?fA&6`gDqqP4e(JZMI^c)!^^pw7 zj`J(O5at42{`bS4okcBBum7#wf2_lC?Ah_t{v@KDYwj!3Ee@;Dv+B2f>AgcrP_q z%a=O%O*p@|0h&1m7pI8km=me0xTif2JLj3gOp~Bn2S*Ur3Vbk7?DrEcoe#jgp1U$sFBg6|RS*;DqK%d^e#FJj|MiV{K3EEr4bwZiO}6?O%U@&$WlR(sva; z8^eORh##MGq7t zr$N;!0%IrH^EbFm-2u1?{r1$$6Eb~Vj(r(yL7RjyfnC&}tE7H^J&?YEmS zZj3!9>o-you>_4>B+*k%UZk}w$gs581cD&jI=_%fx5`?m$k1=4oa<&WdGt|{7D+-h zf4;Hq5$@k&wfm1JAvW$)5=#1Ev|0{su3LdW18dOlSAx6!54_{KE zP2@Z9w>QqeJNzeF;9sEmja!wD_r=sVj)_8nJk6Yx)nOzHWSse)K=UmuJJlWU|MmeK z2!_bry^CbBnMglfi?u#PiYo^=C+Zs``RkIRED*9QY#=%5%Sg$52an^m(!5v=J2%hY?8U&kqO) zOE|rc5tFB5^;msTfc)~5^ZG+KeWC#s_YNWC+#bBUJqFMl3iCnK!ZH%dP+lI2+U%8q ztyuAWgazKii3>D_BIEnLS)SN1r#~n3_6D*x-~KON2mwzKlKoHN_JJMv#fMr{%#_*9 zeisW}bfG6tXWZAEa?3#(9_HRdWN;u(9uGi{niJ&78ZpQ@hwXP+n9OT4QtwQLr7c-H zqcCaeFbrfSPKFuzx`HPt^<9T=_Qm2%rZ&d@umMy1tHYK$p-tP{0L=_Q>f>;8?_M0; zegQ#!mtp2)7kuO6^)1>Wh&|Ie6;(JN}63jHG%c^rt~W58kIX zL;;pV)zQs4=JrG6cT#7`%P_c)vLK6J3F>5!w)d=vnhbPBud$Qi+(!XA zJ=hnD$hyp{L|rE)QdBe{$x^lKq8h~38o;5y7J89+)C{q=v@d@n(;%N&-iv`XQ12t? zKkJGT0k7{J?0f$wfM!A3Aqxdh?%>L`NQ6ZsqkvK&rDO`zijjSw3zW!sGqJXzi~K&Up7{1IlFccZ}tmp2tG=Y4n^$>lv!qGSV44>K4DjF#8+bAoBPPafj_ z@uTq15hvHOCzM$MAeaJ4N|e&vfy9v z&8g0a?`)2h>!!h&+Jf@`#3TAriwR?SE&lfz%}h3yaZEk?4SqZtiXCBHFl5>s*qW%q zsJ->)i`T)5B(a^;LrWK{`coZrm|+zx(%f-;VA zT=>?A#yANm=+nWx|3Jz|P&$l;i%Qae^#KwV<@pHSwH~LB_+m%43l@Dd4YMZ=ggz&) z_U5ma>!!&N8O`bE*Ych)iE=iH(iw^{^q2+LIrH)H&|U;EyR_}gqB7AOk!h9xg^RdA zW2uN@N^n`RY<7hea&qcWSwcxPN^!~4s7FDGa#boCG`M4>L3c9Ao6(Zhp{P2GGMIaz3q22poV8Qz_#16)y`_UYi z{MQHV;EO_%yd05xR^sHTyI2!yjxTq8fQg>w=*#gpZr`UkbU6t-9{0tX6}?%Np@y_9 zS-88%A>+C)?w$)q-MrNpqLvBO$7iwpvlGZPngNT6^RQteS^9LRfAziY;Iu?4oMH-nxl)wG2H?utZ*VNv67}X@SoM`BdiU&#)}-E7?fvzC ztEcXC6#e49#mXzPh?jA|#$RT@gdpduhZVMIO(X)2c;onQmry!m8xx+k)Hp494bWVU zirhq+qg^Gq7K(^+1=NiG5Q}Wn(GYwYTi*K_ahhi6MBP$tS+;6JA#zizke{yr*UjrN zcBt9wfo8)>C@0;*$+cS%pgs+%L&o6ysV0ySFQaMOo0Def(VXP!9 zwc6b(QI-~oOy5J;@0W$>Ds}WU8--EbBhb)Y0pUGeG0$EZMk=kMPbRIaDk;sE*$F*m zWuZ>3R$(-fkWoQTMKu_z%aB0RaYCW7G+rdb_2n0T;0)J%N!3zN8u$(-kQw^XJV%j) zPTTw=&9WE+=7yiaMM`y?xgJTPN;!<(Ct{!jc796QLwR8xhb zV5lhQH@60w*Y3o*UW+hx>_B|#DuF9|jv%(t5S`52Fw@lly=1w@X}7Ije3SF*{~FL- zgo?r#(H!GbV1Tp%i}CXeLzt*jYy8F2Z1b9Z_VH2N^6|$ zbo!vXoC2$>*%DnSt0?q&9Tl*kvq{#II_`o5R06@oyVMK6oU2C@_$r#-r{cFi!cf~o z7SfcZs4s4$z0GdOx-*l=;Azto_)$>>dm!yaaudlIfI@zua)|>9_+XKyw zDW_STd_8jz)I z3JXn2VLg+Ud)}I?3TY^aMOX|IVAV=AI(CPPr2+bDD^fPA35XT3eNllYH_zcdt8Rko z_2KGZ2~!p zyCj;-sh^+J-}6gj#%0{4=DILfmK%>jFCA?-c6x}|u6@wM*aprcNvP871~sl%n5keG zbp>aCKa8KZ_@ihXiIS6sVTntxcKV@^9Lhj$gdeW5@-s547SeJ$^up=~BO`t2^-?C` zhB(6;v85QJP`)fLBOAq!Z?F>OGJe<*fhwC>u$}r2zVo20Cg=8lxBnY0@GsE(#`lR@ zkA+!yO83@eIhK^G>A^;&7P0$R;pCZ6tO@IfFSk#HgT4eB82~@7?u6ocVc6t^Rhy1S zcOwvd>oF?&xT3$g4r~RWxv>i6`E>WrD@AUV4740vF>r_jJWK?j`E9~u4*CsA2t9fL zXMa5mKivgz@fwM_UZzkcfY^lfHZ97tulKo8m1=%+Gk*HBeL=higD-3!5k*bIbbio@7t7&I!PA%1NWz7JPLJsna9=rkaL z3&ftfmarN;1Rm5bHB{?{ZY2rGIQcc!o-BZWjX4(n`Xxr{D^LdDSxYr0BlgBw{IO^S z&WU?M$87>O{xBYzypWc_;Mr9XG-xdF#jzbnv1?yA>L#tl9Iru`W5MiVvNy}!u7H*0 zEW6xHx z`l>>;4F({zdkxF~o*+9e46z|sC>>Ib_y1jiDUU`qo*Z32HJd%v_bP=_`*<1sD;*Fd?i4Qwbu;A}%S81<@}o9d-6iQf3X z0yGOJ+r(5zd_aa4b%38F#vw7U0)@q8B#0NFTv7p&3ffTDHior>J^HG3fqH!!?mwiX zb}>N_E9^ITsEfZsd4xiQl^bBvmbI8< zqX!4YrkmD~ABcz>r}5p=-3V(O01cOM_;S;`u$E;KEc~JfXxkVb0SO zv3e+#(U9wppet8!%G(cqal9T@F|;ggV5u%> zthV#8z0`0V^Gi5BKjA$o?aYv}D-^h}E6HXxKsHO@`CncN7l>sND3;VfZyLZ^S{T7x zR~8L4WD4f?Q-iFuL=&^)rQ@}FqZw%aHUPce`xFBPFd@@} zGAoZCA)Q9EB`iZ5L8mO#BtOK>+hK?+Re*$Y zZwwhsgP=5)MEd$;!H)mO-g|&Wb+ysL3seLVP(To=B25%TX@ZI%*uaL^V~ZMNG`7ST zTQs&7Fk6dOfFnxcXfQIOsg>Al0h&LDya29tdEzxR1^@s{|!>Hh-IOb7b-Cx|-6jY3jyL!#HCD-Fjg5=PgO}VaH$23oAt9{< zJ?kFORFb0<%g48n4d2d5k5B#I05r2Kn?OO*nHAV~{t12$kwPCbYRvVOprz~gmcX+f-(lAzU3BSC zqm}oUr=nhbb=5QcK8I_chZT6xWfp#0WzW)3RgsQ!#ke8~Ikar2hK}lVy0^BX6auBZS_m;NiH15QEOV8Cnh@hy^`#|Sf1+yqY&Dy} zJcRtY8LlVYup`VA(|?+UDN|Sy&DittyI){x6{9%H9ryj8;#xo|r8C+?v$u_?(xacM z6qCjERy6c|D9T7cO3)qZ=!PRW_ytN{QQtMCGuo+Z;LE8Bco^Rjg%#~6VWS2eZO)NzVz4m_Y7&LHU7*}|p$6Fl@wLg*`26Ahh!QG=P_qvoyvXnyJ) zQ`cKHLxwWS-m4V z%Qa^_>dyTBBLOVQ!y~SLE?y17jyPLv+C3F6gIR?^`IF#ZzQIYaR2+L_O@p*2l2qUGY43Gcf zlpp-L0!2w7NFu9yosSCQ6z#BU_XOw>9Af_TBL~|~Hvi*uCva{5StK~`g@c_HKGV*x z0h-UmGDfw<#7P#=;&oWJo}^Dk5m~>Ka!7c?J%OePdQTq*3%MX9T)#k|c^_i*tRbbL z1_K3NeAV=qh{{DuqBLx`ZNjML0L?~nFenH?^wk~MO9S&vMJvpuv6P8oDawPc!Ee|1 z*ef;*hPK17@1sV2ANl&CKM8$MhTOzZYVp&&{q#w==cu7ntv9R(4ugH=H6+mx>~xwH zewf{xdiqLeLz#%G!Zf^!4nag{2>fDOLP5VL{SHiFq9#K!jsWO3M z#;+u5G%FYnf@SX>7-*(OnZMfjQ?qv}$%sMfi$}P1&5N=F>0GPiAmn~Lbb9we7kxdn zicdvrHAS@RtO|7vRm$72E!>jU;uJk1N%Omv~eBJM1 z@wy~9Zk~@J&b?WQ$V$r>&yh@-$~bE7JdVmhdah8ZLO`op4sF{hLQ$m?Ix*gL?WRk; zab?I$P#gZt4qQKR0=r_=pvk$%Kuz$SAP^cEj7%|QwJVp!qMqIbXoJivqv-n|u#*wmwB`O(sG#$N-LxhZMb4v@GoiU^>8DLxzc#I5PAu;ycPi`Q;E$owXi6 z4D1AR0=`1~h5I%mfEChbc4NsUF~~bj!Gxt>!9}$(jpmwrC_nHBJTALo{=Re!T)Q03 zlLlc#$4bh3QA+7rDAopa!Rp;#VU#^(E6XB8*VVP)B0QDtjS>Adp(a!X)W5k|!C9F= zqK5A)gcYm9WajsnPkAXvO2XC6b~XF9=bk=D%oRtDz76_mwxyh59*T3rMdvvGNE}a< zN7n*XxJ-{5)a z623VY2ZLz~U~Q%en~aM%=ADk13KghZ+Ed=uibM%@7_bysz!1(HxrA+p!yxy?c6>In zA7)zCXWG}avVsyS>B*4@dJ;kTK@#nxI>OMH-vVPJhHaNk5zy6=>f&YN!X7@IwGcw3X_TP(hgfUfv4?w+}ZGlv6^Jzxn& zI0~k3?-fz-byX0jrl%A$fgOZ$zfUTU6oOYi;yq8m=a0SE^q5(s(GxN2n@Q-y-AIiy zs4$V`Ae_#hZfB^`{LBF3mwv%A2z!{ZbnLw#+w49OUMQTq0+|6P$+o+MO^>8uFvtai z2id|#riA8KWL49#Sy-giY|Z$3F6b&M7{K@pjhrE#b`u^Ke6VkCG!(f(7(B!r<4lyB zx2Sv9Rw3w9Z_+=EjL<5`^|BC^fROC?-XZAM31fKd^mS~xl8n{^S76R)Bh0aGg@+*} zkk`wcVH&j&NP^;-M>~wBh2Fp z4$WQ-&vq`*9XS?1uO9(bew(_D!P~D^n!`%ML-^%4ckH;42h;C&U@G-8hbyI{#Ah2; z{gH^Pue;#OgDc>u(+*l`k3>N8%JXmVq(TSwv!=j|Olp<-G>a;l&*eSG_1&&`l&y(; z%jwv*au7@Pl%f1yp9pK{*=3|Dkk7 z!*kdDUX{<1$J7V7>EVSNEEo3lOGB5r8!@;?TiBO+V&^Yjco;)NL*=fpU$_Kg?MyI; z5c|{n44L=8q>SYWhad zx2vqEKv{)QPRvbcF?Fi5BjM(Hne6ij+<#mN!=b}rGqfKWW8I<7U?b=~3dzN)syw7p z0wnMnUGC33M~S)xj0TUy)NupQgBv}SwnDjZ&3PLAw-#t#a}TP+N1$CNYA&-a23FiC zni|5?Kt)sm(1NkIBt4pjpR2IravE;sbi;)0D>2rj6D$>k6Oc(sQ`)L3aT*0mP-lA| zOVn?Q+`Rv}Msqq+L+|1Gs%7w0nGX5>ldyeEXELmWffeS3O61(xL8hWBzB~E6=4-Ma zt69oiT*7;-6&KXnHki44DTZ6>isqr37d7|aj{(h023GQ1yc5J))BptaySql}=}%1B z5doROUt`$xSy*BvOJ+-L0B&v@gxA#p&A*)U!@+Y~M@wG6L2zE1SM+LGS7Px$+fvQWwS=hVX0 z@XcUCbQ~*d22wVo-A4mz-RW5RTFr9GZBtnK`m{A(S}kBnHFZJ>vI+6xQ;sS!hbvMZ z;O@OUxZxfsGR@VPz6$;O>S3xRD-`y5!7Dfu8B!XsUqt$$uj0ZGB$xVdu=+$*h6Evtwx+IwOT!O0YY z7~@ANA}p~IMa5OMbRe%|i0;Zv zObH7#F@hZ8tcDvu5WQ`>qpM&(SNq0)>8B9mDpDyq#ZurKw}tVDhT621=-NZx z_Itte3D-QW)kNdr?AQ-u44X3^O*OD)-S%+b3hH}CU=Pdlf7v(+qwN^)%gGX>Jj=;D z88}-wM6_8SY_0_sMDxRn@%sJ=Jh*lqKO__6ka__N>RYezkcZvZUts#Y(dfe(@4t0B z3r(dwLsc;i6aP4Y6TjWVV~v@ZJasVUxfnsn9f|tAI2lO3h|L?$;fz0H+MpF!J;fZ( z=AGWo1HuRWlY_B4ITXp~S7PlwWyI3paOa*0&~FaVTpf+TvnO$N{{{4dbB`iuSb%m4-H zj2eyR1@Ghxj{u4+XA3gdlo z=i)`2J$VzpZ`we?b|P#iPRBQcv}nlLv|Prgk23c^WQA6et(+Q2CikuDSbMWA`VQ=f zUfSh2`QuecX;J1!LxTzQBqpnANZyw3Da}M4V@h=kAz5Cd*IqMzyeF!1k@=Fks6o+4 zknIjl9aWg>%himpxU`6@pf!M!UPesRs7g1Z$@PHd2bm)2j6VhG4959p&J^{|A5Ife+Vwi#MiX*p{y`bQJwD*kB=Q z2nU4=MgP*ZkaKSup1a>bu-`pu5=Y{8nhE-jalwq)gJCXTL=(7%W`K1X%^nY8aZck0 ztaH)BFr&t53_e9Vqh`K(dUO+k<{+FCfM(}U1)6z%utKoBfFOFX8!i)bID0!1*@jbS zaz7NG4cCK)WGvivU*g&|3zBn|Vo3;Ckp5BBk{p(RVOCcvPXBco|P`oEEJujVHU%7`zjtPOZne zz#N3NF@w(M8?b7aHU=rgqAK(tzT6Owkd~|fT)GBxds2VCgFsJs8{2))An?QL zjie3Xw4y_pyYlfWnLou^qoXQ4yF|}tHV!m1IoyFnxAHRX4T3%K_$K%FNADtx)lOB~ zBvF046JHO|;vTM%jj-r***9Dd3Tp@9Dt>P{Df;_g{lj)Lc^dr37Py|ijctMIWc+`F z2@_1wrnRR4za{ogNE^= z8ilcbh>3m)*H_vw?WG4ZT?I0`^N^HEj}DTigf_o-5c8^{?qC(=2L&KVoU&=P-$^Vh zLL@1KYEV&@7geLyAFlU4_rG8Muj_&TfaYco34=I|IVw_b z;OHH3q*ivsw5bO0Ub7OnnDz41S%hB}SYlW&$`zDy0%PgBC?F7MM%e@0I(!DdZ}Y^f z;p=$=+hDGpO5K@-@`Q3~>;}dnIjsr`)_tL`LeTM(*ohT19=L$8OM9^JOdf%ru^2LQ z4i;D`Kuwn0zv?0`M51tO`yQO9_Ni<8$yhRHwq`wIxl+z6i3ypOMTxQXjF9(T< z#gI_#L}7k8bR=`|(+;{FBN^^H58|FT!KgOfp{Gh9s)1RW0CJSS3dWPCk+>h00~Ip| z7}MFJbBi=QiYbG%p##hYkHV~>dgw&Jziv{gZH+JyvcxF!4Rv3i<4p@iw2)VZ8ks%q zB@2*x|6Co=>^K<9<`^RT>?WLNR`p7(A}klJ#gsv+bo_k=w-w89_H{QD7>&W+wT{rE ze8ju(UGWq_H!ow?w5*eWx@oJ zrDWtFCrz!|wo+&#-5Ti^R^ascyIAKTh5-xb!NH0pO&vtrx7rqnepW<#;w4KeSD$@` z0&N#qkC=fiUs+P(gt9Cjy=d94j9YL!aTaTrokLn@CLT0ZVbrNPCC9=T^L~2q6iKn! z2+cFYtUU|^^nLmgA4trl>F;j*V+Z@tP_^ zL7*GFw|s|vuNE7_K7;>2~cQC)*aXyoZD#7B)>nLZa}&9;`o}fQxZz z_p4zx{Spl%XT%OnCMHf zJTNPsvH|IcdL_hX9tD)zNVtz*4!uN07i(C}o`I2OU7)8xBhrr|?=^3K|6d{YG{$H) zLQesRdKrxpN$L=)cM&DW>LigUAtjP}`W-V5#6ouH{C^E--go{vHGL&9bf_kZ(+cUo z04NP0o6S%OU3uNr9mQmJ_=^^upxjO!`li;fbasZlb{nWjFmahri559- zZd$i)Tw18^1DpSdrK=TL_py2Y6~sxBLA+`u#%oE6STxQ0kbeO%tjD)0(D}@u3NjQrCsf5GPTLYpepMuKoW9fM#l{qBsWk zFI>P)w?KHrE5K~@C|K!B!0_qMSnJjXua(WH%Q^|_OsuzKxqk~9@!U9e14+elkTMwz z2Qs8oS;}5J^HpaBAtls$V$f*We1J4X&sz#D1=$N0@V6)g;g?e5Xp3M7pu+>kPjBjSRBPc*ev^ger{3yg4H< zXKXKYWN`gkP2zf{qVN?`JZPwQIRfV+WMR8-K1TL6qz*pg3C}M2rh*Y_Neu2D+KbDV z{c!S?K1R&_0%JznVyKq%yG`>ajzGPaGk&JOK;fOu`0hdrJZ)))pAOD|v8-eh5;bq% zD(pG(eQ@*m6&yJb0GY3MV3@rHMytHlXg;5c<1dsj_iH=ob&#PTRPDk3l8}NV$c_&~_|?le?ePc!8EsJ1Z4$noV2z;`ovCR|39jO&ctnYt)4NEU0nKFzvSTI6Q6g9-CwJY9=Yi4{`JPG$*&|V9BLTN zr+r>yOY$HbUhTlj6}1n^E=_($?H`2pH3w+s>ndXrcIO`4k6cAe2It0heNbXN6$fW@ zgAp0UO^<=FZl~dG;33?;?t?jhw!xfjydRw{Vch7~Wg?ouUqJ?wds$Ij>j&ca)~m?q zG8Ee5zrcKEBs1N3jk0QDA6fR_PP4;Y$2y?-F3kYl3^rgnpBro-Tl+=p{rnW5nfCpW@G!<1pPp7QLhr5P0S?yu!1PTp(uNgi??prO{0b69J%d~ps>-U7{CsiQk;yanuXF5&2}w{Z2TII=AkV&fd%gUX30 zxV{DpkGDX&rX8kiSqvvuX=;krtHSrlN1XdH+&jDroBcGQKY1!VfLi^vcjUQGaSl*Gu zmyxhGRDnXhk*m&4LdBy4Sbr=M*I%n)*!m5a%xVN{m3k(?qP;IkXC!in^p%C`ZPC#< z&}^&2**i59kKAwLPG}yUrIeH1&4ge?I=yIg{u*c|DW|HOCa9Gp6Sa^;hfmF-R*+_# z|A11Ohj7X*0sblzvHH8=m^4TSomedzf8!TiIe!BiPsX725CMQ5hE0@kE4sHChfn%q zi@O+xZQhIt_9hsjR(D0zc3WjJSGNhvvKrK%4_?xR#{y(;KiQbu{T#H1Wnv&lh5t1HYMondRNg&y?B5a!_j zj{kRh;6I@GQ^u;2B6}sz$q4pEM0^f9+F9Vq-UEn=%R{c+Y>Xg!(uF18<@L)dIk{Gs zy~I;bUw9L2%(DCpeJr|SuwmWujyBPpgXs**h5PGFamX4CfHm2PJ-g9`oU4crVrvs^ zE{(p8m^+8Lv3ZPWvShnD&&JQwsgKi(wKYJL4n510VAIL#j zT^kw%0kt#~sTC}aPf`te*CwG1Dh`)d(#iJX6I?AZ!;m4=^wp8)qNr|ZB66T6Fqe^; zhqSCx$ahkOBw31iF9L8WxENBF6UbngfmNfK{pAMm!+=%;LY5X5A@_AQ#080twk$ss zEoAZ#clT@^&^*$a21GVc0DwS$zlgXDZ;xQy3@wA|m@jE+RtC)sAM9GR55WpUpfq?Y zwte0MD%3W8cfOK9Jh^@eKYzXpkB#S{_n5IbvZ%KR_^&{DwgBK4{r%OW|sQ002M$NklM!gX9m ztto%R$7UkDr7fl{okSB63m9@O@zL`(_HiFPN3(?dbYwmALu8dED&%$0Plv`d+%z

Y~g^=XV|nwb=)#Uas~Y?6E(RH<~w*g<-bZOKKHH=UHI;m{oH zj7=LxLXD@dW9F0DSwdDtNeL?3vLEo0TKMj~_IBWD$6BCS3vCOMC|U6uv4yQ!=1QmQ zsOLC+A`I#dL(tva2T%r8ZLCX->x8jKqqp{7dEw<-$-nb}BwT}(i;2TS`Es3qMJ zc_jklfU%76zbu=~->4u&M<>&~&jceE%);bpj_9vQreaeUQ!UWE))#7vcSB?bXe#s= z9jVjlKu-WWy`1-pL>ubf(UCdr3A{p!@w`YGW2O$`rh*j$@6|ijjJF7AzIcPIlUEqM zMKA-Td9B9ZoB=9A@WSs7woki&7L(_~lq{C1y3~JcbWEClR&BO@FhkxU!nQt0uG&@>hbN68(+B(m}l%d@)tcQHP*pW}5ISx0s z?uAD{3If$fW306iY|RazYup7LtKwPe`wW5pFHo#&3mpSJm~$=@MyjbxrgxAbJc4TZx)O z=o6ahM3CiDB9Cktee|Uih-^bG#M+m0lb!g;9XHo3$L$J3BrElXf?+>=#~5c#se)Ez zZ}6BgE~;7uxqZ2bB`~fg+L*rkfCdlh0nI0_VMmAwrf&L@_nj?Bo_q)FHh5F>f30A0 z_jM#fy>7zQ)ffITtSm5{iq9w6U_i$VRJkt4ipy=8;2eN&k1v5WlS-=M)VVKzg?k70 z;2x9QPg|Q~_~;?5-q3)d;@c!dJ&opkvnep2F;B#_@6fO`tNAT``$~>SX+E-q@hEPG zOc?`$XH5Qz0PMy-j2pnmo*a*5xLt05L?tUIbnS~JG^py^mC0s)f6;hUbU;?WFGO>W zP|@+p2jeHo!f|5h7TgFbMFcfZw~o`rz^)RgDkE{=$0ax$P>dUG#^JyQN7(mNdnX%F zoyF>vCs(m*=@mSR%s_#XHdH6Az}L=Z7)L!&;a+KKMs2#C+yLjrAv-nz7k}ki?u`Pn zwCyox))2~|DTvnNb?;@8l={-CzQ=Lr$`x$*^Ce{aj)#@gaLgEK4lPQG$i250sC#E~ zA64fk(thJO*6)phZ(1ifZCVJ&F705@q1HVX!6dk zF#{BUW_clkQRnu_ad_SE!>ZrQG5Y(_Fj9Dp>htUWKG00Uj^GoL5dM(!_C4Gtllk6@ z9Aqk+Lx1KLtZ`Du0KHlhPrlCQD*O-a#_4hkv}oHNz2r-goLz=WX%)z;8^FAWAv$YN z=B9&C`rR6>{$AFiL8l7)2O6^_gdpybCtT0F(bQNLnH~G$%SBV*&>;flA#NfwP_ZHN z(QnGtXOq6%v1$>%F-0|66V6UW`I8(5>=&*8s~%! zQ+kN@44-}<2+d|&$-2(++=R0?BJksVHLU!3Ca)y}boqc;D!l$T~;}uFcDB@*n1EjUFz}97su+me4LX+!cA?6n#`rOZOBY3|)+7=5&%X1vnkS5lh z)tEGs<{q@szMu?DZfX!NE?tTHE&HI2#ld~YFYvfW z6Y4=5F?WzI92*6&8}u{5^2$>BkVFZ7LB8<4b00VEgd(A>6;%39!7rR?u9gziX_lxY-{ylgM`ckCiVJdR zXsZa>HZ-7qJBL(+pLqu~>reX%<7PQwoI(`*FS*0}X)a!<48yDu1ddCdB6Ron*#6QI zGL}wQwP7L*$Qo)d8du>uAsBxBC_H}IgMGw@V?2-ben9U8JF$WEf{UTPK%%XzkzlIT(8jZ?Q?;QkwxzW|!K zZlkYLaU92&*Qbnv2KuViCoVL&CO-I;@?)&Xf6aC8?t_ovl}HoR=}WQLsVnDN1&%Sv zf{8b2R(usZw%)`W+ix&vtbj{7E zA!aO?aL^85b9^<;!0X++!ZBngp7KvkQ}R0kWR^A>y7Je!u9A z*aAtqYs;W4IUI8KU&4IM94s3k7@4uq=u?5_JOfmWUQUg)E=>TW&q7cFmZb@^14li9hn9K;H@MW6*PJ9 z{8iLmOS;1A@ESJl@Ij{e7Z~Y02rI`Kp$#+7g4=Si&hn<-YI&y*iY1d_AjbxG{_6dUae}^+wPAKg@4!b9I;imsXpk4HC zn$CsacEz=A`w-Gr2L)Xnq0)Dv2!PO6q+%S)-cqit!|hTNL}`A8Z@Dq)RGS%Y?DUO2 zS4pPu`P2Aq$u4*sF2Z1L96w1qqp{BvJx6Ao`z4%SxC&R=PK5sOG1#=)5uNzAhAE5M zyGo30Nq$xzXr3z?a}~Mz3kn?;DO+_awd(5&{p{W+{$iirD6{#1? zMRQD07;fKx1<65+F^UP3S+>fkY^?xUCZO80Dn)`?fAKeu;Le$|*zDa3#?$7(h1!>6 z2$sDw0c&{fMuBFb$8zv0oLcCpEcKyx59niX{$sTJ>y?h!VNjHgH3vsWO z(fp8xSt?lg>s<8F$b-~_^Kh$nMAx3?m@=S?2mq~t!Gx=<9B=L(hVR)!_~v{j+7Daw zX+Sge3`6LEG>jW<`pt|uHiSgG!)w}#li`_G0jFz zlt`?JADpk4EV;hnXLvlx$773CSnI3}dv5#$Ifm*qcig?|#w3R~3e4wX;3!AT9^He< z13HucqX0w+hp4-FePc7ex?_xXruNvf#z}NP*1x+xI_}&EQn(N2zgf<*)I8j7KMDIc zI*~1?iMj&f5>_d(662SpxR`B-s4kz2Vw|>O6YGu26vPF3;mX?02yUs0%q|1awl7&^ z*0PjIYDp#!tA!Zj?q#&cJ=L!WzR1FWiJGPlVFQ|rpg%Uj_0(Y(X6$auclN_8)+%DrEGz49f=iFA}aVc4u4M|BBmvBy4Ya! z%Hz*AV-J;UN{0`z|d zG;^a#)_4%d<8r1RBD80SO3`(wC*0^!nJQw&U0J&Y!Qvf}p*;Z79FK1}xAaz{4+$ON zg*oSbx(qx!%@u)R0t5Y%kJ40zw0qe3%@KIV5inb@mun&VHmH|S`XCag7f`!7yfq#h zd_{Q{YuFfec$bW?E~765ujK>tHpA^{DPF7ffy>VuF~MAu)lp5%r5|k|V*@1@@*=$u zNv8CwW4UN)#x>*@lQEvj4y86t0O?%Mh51~=wiUm_FGB-T#ts<&?HBY~=^!#v{QLeV zY%KMD>9&fKYcJvUL$>X+#hBxuL%_6MgUYhEUrYd#=3xsigM~r(||a? zgoJ`h&JCQ)q-43TYfaCZ7Q&pBg9NWrxc}#VZ1Tt@+j7w5t>#T{|K-}7k}Mv--^3rQJ)t?)n7Y-~2z?1^>@TMDfgXCwH%2lMVru~bR{^Zu zh8qE|;HAG5)5`y3BC80yD?Z0BbIst?OQnhVvk--EB3XI21UE^R2E;d&C*O2SOp*n&g&n{d?q?E|3s z%ZV7L&iB8L7I`s5N|JokJ{j2$C`s|{M%>q)0~4pw*fy`XXnj@_v4!&zuy;L*n|tP=hthIP|3+sT^h{n@kV%lXQ*4*V%}PoN3zVdZ3WBj-phP*!&4p#-*axb zbi@y7_AC<_XpTu1ijAIGOpGqsVk|r8r~tY4GGqk44-_=|rngU4=Rd`pXRcT|?@xHW z(nJf3@z}H71(sc?X~Rw58~3f);+hZtQX|aUIv4#Z*59Q`O5MlK)Y#v+*kie2>GKOX zeE1qp-%CSUmyrw}ec?nsK|OV4N=tCT#Q?=60U3wX8cR<^YU~T#xqcfDU*;lOOdA76 zjlw|d?&znl(D;Pj*r`R&)dQL@N8)_d2>i5l44elVkUiN{SQ5rl(3d`qCl?N5>1Ad^ zN6vuDeCqM((7@v_%j+99rZJ#-D1qimv?wkWffM3wtLT<~6u###U`wDD`b?RGiPOkx z;-*=u{&hp$TA!=uv2@{S#P^y9`>{<{*6dWIB~cTcCD}Kx_#%+{QS!`c z_a8SE-#T}ND&;XMJ`rg4&e1`|h^5#)*?{I_&6(VU@2)WPNPeF5{;jtks6nP zj0`d4muf;sDiD!mc%FUG^4|xVxtXsn%c2xZ65?M{-}z-0Bs6H^(2CL|)oozdryG+! zl=Bb)^kgKxp!RQKCaSyBER7NfHR>9?&r=^#I^j7k`c|R4ZC{x5(uXbqT;fXq?5JYW zA@mN;eETCj+Kq;s-BcWyK|Q`Ujg@Bd8l{H0yX$>y-|-Y(y7PBqdzAN_`7Z*^{LG3+ zbg4gsz4H&_ktt);#D58Bu6m9TN~-)mZyh{3&x6@07wlWqN3@A+a;?%Vpt+nn=~2iI zIfsKMGq}alL9Z`9!?4bUkchqy-(!dI+d~Od==6i#)QK2vsSaHw>1Js&zZ+bZ#uw&f zA|*yx#NNe&2N5`SH6MLuk4K;WCa~Q|~i5A`p9A%V9j!0RskG!jQ4I5z=wZ=^F)_g?o<-={&~HSBZ#!PDb&I zFoZ=$BAn%S86_>r#^O2YqTYWv_0;XmV5&qN*m`lLMmpmPChmwtzwfqT{8%T<>DBb& z?pqmI_GzLM*)sXJ)01JbTH1Vv@rYfZzL^dKl$uLbRqr_ByYG=x$v+`RYmhmem zlkxL4+4*vM=xkww-h&1(Su2AMj1zzVC`IPen|OX=4Svd*M3bI@*!D#)2xk@gsCh>* zC-E%Y{plNOKqlZsHkn%creSzLeRQt{JWsAyM0(%1}ddMx}0KywMtk_4|ETXEsEKTfxpfMv@@V%kV!bYue>Z9{Fl zs)UW=Gq`T~1NWXpBeax|laf6~jB&uEv1TGOTnWkow47-zou5jP8skz+kaH0KD_ZjJUu#yvc_cMm^uJUVJsK+>oe(t0fV*Ma8h+?S|+ z$qI+{Pw})w8)J8^$1pW1vH__TT@->NbJyZ>1f{=>=QaSE88edondrKRAJ?BnKnxjQ z8q^72xen7wE*PjO+4%RW9plD+k`Y_++7I`x+`{S84-waWG6oLmhpF~DB9OUNt$<{` z_X^QOj37~N+#^I@{RLaklAV@FX(N3Hd^Tqc9PNy08eR)-)_Y@f9||%Um6>>cc0X=i z55VpOBYeJiB3vBJ(2Z3H4acr3Kb94R-Z--T2)xoYk=f%5Y@KZa^UjnsD)`?6&DDiz zsCc*&>n|7LOoGP8fo4`7Wxrrm*QxE;7OIYvtbEu=J;ZgHnXnq<)J!u#0v6P0&3f*S z2p?};z4jEZWmx^xK^na!f^d}caFhg14L<;yh0^Ur6uu6?$ql=4Be)bV!7Rv1Tw1*r=UkuTQjR$m z?p_F&p(G@Urttp-(9Gr*MdO**6!_?6gw6jL* zVKcGZRth;c_BIn}X2PsI5W8la!HWt_X#BYD6M$x#PG`JCn#WG;y3-QjV!iPF59491 z+!h_0WR5fPuoAlBO*ld+adh)KHGi{-8FZ3{j&d0jthW&%)d!LevpxixD@lBcChPy` zW^BKmgU}Xadw)3}pWEudO1nL)?1ZQyL^k1Q7-4FjZS3F1fo4|iaX(g0B2N*e6U3s< zz}M|2e%KxY?YS#NzBx0^6zknctkm*&d6534lLssC`ee7_hI_1Z$e-N%gj>b_NV&Ra(Fp>SMGR)fo?pD@ux z1s2UordBboWcXpv=UZ_lL=p+wBd}(P4NUYpu#!@OBoHRGC?P3b%IaeR&C8nwG!v(I zc9$f_ow(X*5wwh~nh7)u<5)#wbtej{J6=~uQwQDxUuHw90 z-Un|`(vCg@Btyz01%5kL;`r?hT+6q@n!{gVuvurCjo0ekVPHlum=o`o#rr5rU;#A;XDnGi3+8e(RQmv&C}hNXV$UDk1cV9+MSQNtQ*3 zFLlxGKg9iqG00Kq2CW`e=-1C0eT`Moi6BAqu<&|7GXa3}aK^SB<1u!y;YS$FLQ5*B zdv*+u&K8t7aeiL|rJu(fIZdJ zzK-;B#ky~P#}jcQ$e9nt?5V@hhu6C{7ne2L(h?>Oc>b7ZybSe5a9lB-W@^A%+86mM z#)#Ja^4|rT8Kb;UoyG2XM-bd?I&6m7W9c}Y&_t^5Eyyh_LV`FjyzwG1{cF4#Sz$Ggw{@EYbb%>tVFI)ZV?e}!}~+B1C&*0$|^63*Tq^e zGovo9p{ht5v#AzUMf`?~r>|l$7vx>J`5Ei%K*OZBNglHGE6M)NON~J?9n@nJXaE!! zPcyk_$~aLcKa;WoRwFouLaCx5I=10r|7}N$fadd;L?(C^i+75E<^`4l#PxR$GLYkg z5qfeLE+p$APo*zrOzs0if>-a~AC!+GXrC4BPu7hqt{itqrps1J8`)t_Z-E0_jtsx! zczA9P7M-E7VKy~syO_gf?sCi=*bPGr+P@cb>RMV4Xg+t74AOjbH8z1Zb&j>I`(whO zt}yIGeYd(d@R1Pn{|V5{bLK}d0VR|KzJi#n6k4#NsczHD_)|pBfso6%>L1T?cUhQ> zrd*buG9_E;Z_7)_f=RTyhD0R4rmmKd3|Dw(5Z?3t$W-1kH|WsSz6?u%W>*T2!4Ll1xtQi zgi#JX(CaNbraCVKufyH2o-yuz1{wZFH8IZ4TpI>za-tX~#*IyBL6#`SMThw!Ecy)s z-sr-VNyrvT%JA1-^7nye0hB3uf-@WTENDsY> zhr7PT4nI{C8#}%aGz%N7Twi#cxIoi8H^f*kz{m+h@cr1XqU2kP{}Iq!ofk%P9XHXO zbMK8J%DdN$#{ly#=&JtVc*KPHaXixCFB*?WXsOfzejJa*R{tiWncz@zFbNP_u;Q8& zQlxrf(ULLfO0NnPsdTg|evF+9cEdfjBeLwj!+x%Rtf-YMF0gjrqRFE ztWon{+`!{I?%4hNGqg6ffo^YW*iza-zq2AuMh(Cg(6bx9eMm}Iq} ziL?tPd(?z#p#Kv<^V@T5a4o`_B(0AF&5Xqr*-7hK(Be zmq4?i@mWCQ%!sh(h>DFz;;S6w7Lj@0)d;#OK+l|uSbiiN_lpcZ1e#f4S629%WS0m$ zWW|+-_e*$3w19R$2Uzyy{=>ou`f7qkb<1X5KktUFECEsXui`3Yss0QpK#aIPI`kNc zg-a%2sHG0N%GS*IRfWl9p~m2*ZvhQIyP&(JfoKmP9(4haZ=J%lpMoIAN*l*1^?+u^ zOMwj?xOWvUT@S^S zQ;LFpB-$r&4=8B&5)^*%A{>t%gwg=D99g<9_-t4&IGVHC>=S|JKQ{P5`={lU08D{o zz$F}c)}CW)0$cmuFlGY33oEyjsrMupBqors6!CX~8*rE8s5AeC^ zg`H2^!S=I>7&O=xL(C|Nq9|1(=_gbo@%?Y&1Mw{I9*)x>U8rE1pqQiLICn6yw3u!qOXsyb4px268#|b#U_&4~J_e7O_eV|#mMi{FoeUXzK zjpzUmT)FCt2k}LWn|(3R)(ZV8GiE~5b~!R2KD83#L!fyQ8O<#rEf5a`KLaxIpPV`h z_xmq#Hr)V|=8VN4O8S_*w=UxPy^8nj(LZr){c&6`V{-kg&oF=aD0JtoBP2JPK7{@( zMB=Sqar@dW{Ct(*KPy+IsQ;_bYbeHX&EDTwhTbUAIJ1z|SdYZWY(@p=?oe( z#&^V^AAZK?z13I&-i)LnC4~yoy+yKu@~0ei`nbPjD@@6U?2*!D*N+ zhOtVg;TQ{7$h$746ii7TtHYQ5h;t9B@I-zXF_j6hGo~TmyBF1+?UNq;2Q+_Dcm2Qq z;{Sl=PZ~ktHlzv3>p%qGxrGB8Pvb=^2IZdpFm(J_^dShROHv~E*tIQ5ZE5!cOVF=|u6EJ_8Eo{gJ)S}Tq&GPIc?VzLy zs$+3^)jFKJ9EJ1MBe8kg7`P1lYoIyP{}diy_!UboRYO$(nitN(IE_}2|0JMU_#Qck z@j8n;dw#;6)WPU6${DLyz6F|@)h)|Oz$>3WaLywWegtm^Eu4z}1}wFuoJd(Eua%aR zNfE9YYH5`Ouw?7PY0Td981gO~;XHmEwm55ktOIyyvRo}Y0V(&7;}|#B*PoWa#H2N% zpGP4&wGz_3XJYoWff(93j~bH8u=+wC!lVT=z#lM*3?PHIi(t`71T(-ae=hb9!2OjP zvTbTgS9|N#FwJWlQb_=^h-^=<`~J8?-R=0~biAfSOja(-=4eqaHE=DjRBdU8} zQy3en!MKA^H6v&=-^Y3{35;C35+erm!~nVqH#{mMui@F9QN->3yI% z>BS!XLRq11UjhK?JJjV?ML z@3-ax%{;HbWXSeAi0c3f$A0+SK?{-uny=IYnx~8#gzpC`qAG`* zExMUszvKb8AeMA0sY9!m8TwF9+N!%Yx~a*Fq;eV|$%*>BJYYLc5on%$C=M3$w_w}| zJAB@=0SH+;Is)sB?1<-q-tcwzz_pu?5Z$Ub)cTCX#vh!Z*G@_VY6~x~c>F8eu3W?C zdtcz61I<<3Tqg1Upy|}9Xgql>1HCa$=+~t^Iurb;J;cO?`J@QxzOL}U8;sjQh3K|u zEe4ZuJw%gS;OaceY5WGSBfGHaUIt#ZForIH^6$6I=4W+)R#RFL1e~fX3J`T_4bEJ+ zj&-*vgVBx2yNNDXu)q=^}4Gxa&KLMIWpC)`p?a?3ta4bqQW081i19o1D!`1Z8 zn6&Xnj4{=0z=o+yjJo+}!N+73c1|;bqlwbT?~U5iHtVO*&I+2kQb#s%VI<{FY7X8XEh&8>N3%I;3AAl8F(Z6#C35d=!)q{Q(Q{z~bZ1SW8W823E67_AjmDEBi4!v^()LoZddWw(io6{9dSMHJ&6K21eV z8#NeCm2iA=&Rw(RhR<7C}-$Sv2Q34eW(MG&!mrk0Opo0_U8-kW?nARnbit zkJ!VMASlAA4&S{8Gz)W%PlgQcseL7ybGm+1%A}EAr+I67IDz_ot8lv59#x$zFpshs z)J>x-L>eTDo?zdW>v-O(2Nb9NfVG^*jg%Pb@?J9$aGP=L4qWdg0A^!hES*ZZnkO{s z&Vig6W9%>|jJKuFhN2wWNwM;T*K*@)NN&cdZ(5KVhU73W96SUf1 z0jc^U-pIFj`xF5Zf*g^qhj8Q4W$e5bf=ue=D|WRaXvGrp{`xTKOd~dGno1E6Z$$vS z@%A)zhF5qU;ScXU`*HeNHBuzCVLreCw!LYjs@{f1xY>B=Km@XqqM8*14o>!% zMxTn?#irlZ*rzrw@;6CT1&we#8DdzkJUfzTqFcJ**VugTxxhhZ8prN2r2?@}!oVo^#?jLAfCVbR2+BJ30% zx*ZoKf)$4^#)zr)fo8@IdWR%l-3hNp2{@A02MgxfqQ8kcn9AedV({?vF`U_W7;ah% zxo7yN8qGqBYL8sXVZ_3F?|$5kAn8GdhMhyq(Or{TucG;x6$U9`NDcRem-izC0ktOw{E5` zyCh7q?qdJ0^IU^VqFA*Tto!w+ftxDysV`ieABVsIANcq`$K&vp=rLtFrKb90jBYi8 z_bX3?6C^3nr|TWI69~omZWofS4z&Qu2EY{}ux&S!soFgF8ba zI}I@H_^iKHgZ94dpQ*yA0udYk{t;ocyrbO_*=0P{uV^+_OOV z-Mx)~WC^tFWCBy0!8A(i2z6Pm$Kt(2Ag6mlcU0NU!0+=-DM=}T^3oENmzAL;H}^l*Ub0DG=A`$3lpgbs4jkR(w_xbB6ub4{CTuZM;|t5Ftu z8jr4@#^l}UFqqCtoiE=4%`GUOSU{FNtD0&mab=YYfI~R;Q zFIz*waVBQCm|>_{%^FNh=r_h7K|Yc?a3Ww{oH&HIC;^ke%1K}Bn$KiQoFX!MPRBtK z7ym!@t^&TQBk!IVad&qk1QL<}f#B{=OTE-(YunwnTiLeTZMXfZTiq(9LUAbW?hryq zh!A&ofA*buA&`)l5L&w1UFP;z^73+J?wPrBXU?4e`411KGW*h9{K5~vglHCJVk^{$ z{_+=IRvz}BK|m7C*Mz%RQLe<<<)2_T&GuK+kHiG09Q3O%#?Qx_;WHu%5km%JRAdC) z+#TU)YtBImAK;6MW(k9N7HEB9?@dV?`3QDI^VYK=H?abresdlQ&8 zoW_L{JMh9Ex4~rk9T;)fEx6U+jA0y|dRB1(dR(EckgL&^P#9SwbVQhT0@6Np)Kvm@ zP$U}bQ#NVCQ#T#0h0)lVm^wBJ<0E{uiQq)I;nh0@Vf&Den*3u}`;Ygr;czw%wz$D# z+(Wo$k;wELc=ga;hO$DPj&(1-j+JXtu&Q|yp8o4JOl3Gw&(;~@o-|BK@-vd?(frCb z1w7~8g9VS?i(C9nVAoBZt_y>X@5CS=59c1FowTiCE-CH}lJkBLm)!h)OT z<1gc!J1(ZE64x?yzE ztN7d6VjO93L*(2!46!$Yn;=#@HhKfL)@ES$_j{162?NIHCYm+Of~&4NzyZW^{IudA z&MBxU@e77ue<}<6$6#=XI|AM8X$!%mW%R5yqj!p_L7SFn{$b@dyu76rp`*qi$j=qt z-B}^1O3u@^;4HQzR=}9yzGH8qFI{gQ&1w!B4{pYGD)tTyehC9P0DE?Vz>R)+*?WuR_u;-6PD;#~FedvHw!Zl~e$KQ(sn>Yi{~VtaDx$oNn%Tw+kg(!Q zY~6AcUnZF$V*Z_&K79=C8sV-TG0+ zpq$>c&H5MBkyK(^^ZE6m$Lx@(P-aE# z`dx-FhGBNF(8{)=hRWLVQu+sTC}qv0P_#1DZZ(I##_p|K@%NSK2zu}xEL=Dne;#}J zyqC++=~G{cW_smn+A5H<=X-2ly9w`qdkSS-q-iuJS!W%F$x}FK7!ipf5q|Iqpg*}K zeO5bNrpJ#Aq1Lp~CvML=9N4x8Yj@MFvN25&u2F|;Ui8iL!g(UmMn^mW7!V%^Ycw6dCQuA#q7NGc1pC2Zz(9ls zc*EV!f@&xdG%9){e(?pC|KC?wU&vo07Gvn#CHTkP5eRTF>oKB5&1e{9=^3o1m*48G zso0e7&ZJTduboXrW4JSfxVCeYNPBU_Gx#eM%^f$@(@zrq6&XkI-Scl?ofw=4PRFbN zxRpwKCKBsnDRP-;Uc0vxoBf`|=YJT6IYZm~l%7t1ZQr+W_evc9`eW?a%yyi~=S9s( zi8_X?7gqx$f6Xi{V0-aBr^nvR+%*J#^Pk2YqXRIZS5lVtSB2}aW%)Y%?Uh}yfAl@f znllLxF}H!3^fhzw;lkb@@!ZiM<~qOVv>y-jUIWW_?72`Jun4kHZWnFQ~&e^oHzE zG^>?O+IP;0I!na+-J`9KAlshmL)^2Dttkm9`*-2?__>(++>^L7*Z_|7zRWtl1K)hL2L=7umkghxCA@;H7|E4+3>!cCiV<~< z$TW0=Z{!Fn7$;!tum}wDp?a&^Sdvrct6FnP)hjIaIun!f;{a{Q& z_{tjBaVPvrv>l327}=M7YB&9NKgAaZOORBdfrY&Tyh27HGI|(Bu^*46s^7!WvTyIA zOBYi>A;#LJOxnBeFixE~gENd^$*)tQzKzyweCCv<{_qHjz>v6M7&|l!Ax!Ay#3)OB zhCe5)TaE2se24AXB}ggfB9YPrZjAOYCBY;9)6-*E`JDSPcE(KH8DT;DBNZBR;!(2w zpLmDKs&>Zbp`xIYeFhgL=r?TNgI~4qHfQ5TmC6eB`nDX)*3uTBnm)2h7`eDI4}~>6 zf};_B=bv#aS>G6UzL}`TF5LP))~;NO7rsk|YWP!F+#Au%cb)KLO+TH4^U1|%^bCi) zt0TM}NebIAaUNC3hLy>eiRLm7#4dRj^IVExM9vuesIt+pAclTmcqMu(%y+?(HNL`yF3R~>_-dqc+XB&6g+U@LJ-Z& znXvD>@A1_u%WyE{F)X@e#;;5Uc&YL60|{BaJL+kHmvt@?3CGSLDLoBoY!mra;ypuR zOG#4F)JYprG=o8YhmQ#mSR=&$iG$js3A6$IT{gfpG61jbS6l zA$Ab$vOI~4Nh^}S{tLDoOv9&jQz#xHnMzVR{%jgbI6P?*ygi=|jh+o$yh9KiJp?fm zMj+Cm1g3=t@!s3pNnEERkI|WxG!by*c*UMk30=!8&cTffAHu?=yubqfabq#-#k*m= z;-5%4au7Q+OL@OmFtf0LE2D_T)_x9_U9TT(eFnlef-3q^v4{$Gr&8NeYoplxI_-;D zxhD=Fqo4*EY9@jVVAQXZ1MDoBmxIEWeq3y*yht?ftPO=?=pwu|n?B4 zoQe?5K}K-x#J}F-!ngAb45t0noezW{?N3i(*AdQt4F_WGoG3;wFtK;XAencgjo{4P zhYkBVr*HN9#YFR#Y$sF?y9*z^I7v&;c8;#QL^OYP0AHW8MDUV-;OW_6m^09}(|w9Y zszhVPJ{(-T7N5QKJjR})3+cFBsnd34=6I!y5Ap7t#y!OEXBB9vymg-k+*Ge=Cl9`rtZ3g|IGVFJisI*^IGI{WCfPf^L`wFTc4ytmIPa<#E zp&Vq^Gi)c8;qJC2^kYoJ-aQG(t!F2$F@%Yk6U=RyoXL!yuvF4nm=xl|@zXfAGlhwL z<|8U1fJ&{-T~2!;R85UgP#M_T&k{D?;qVIz#)vRyzI6DSudkvfU?%cvnMjXGCUkF{ z_)}h^Xl#VWh-ym{Gp!;(QFNT4IY;pE>T38-oPfBge3`OWy;88GYA8i(5tC^gXL13C ziw2IK&cqmj2x9ix&SFz8BYU*E?{)hrC7 zvhIav#=|;40a<6y<4jT^r?EEhkBUL$P+BlBE4e{^4k`+%227`KFGHZJs~gbD!FqM_ zL8PV9V=~(Yv!8zv3+GS5qCj2ih^%uZ6a5q?;OsH3$e&;vF$hIaOcY|NzzTI_*67n~ zad`wo3Ckr@Oh=Fj&CI+mWq%L~~KdfY8lYPL5 zJ8ng6WFQ9V$mpxeJ%XJ0UHIo#F$uhvMDqi>iDr_6P*G9NBp3TO@4)gE$6>ML|1fFB z6g(Vbt-ZLW>?q3Pf55}fZO1vYIQY(g6z?n!L8z-C!;4ap{@wFfHaJP{lOah4U002M$Nkliu7!Ky-nhHF>oRZ8e)Hb$n>lm#&pMTv zHzG3fW<=bGU*OEqBqZVgTA8_6_fJ%$m5&>jU1C(p{jr1kh4VuZ@v7byIk#OBqd{#y zkNgGPR?q!Bz&J_8vFi8$@nTBx(MxC6t^~^iqq>0#$xxi7%YYl^*UG)gQQ6O1om=R0 zjEH?|LTvy2THVL;VS&IPk(b99u6|LjB@4K;=ph`0tC6<`z3uv388XLki+46C5|#7s zt5f~}yjApR?<=R*TqTa#5oYA2MsbU|dT~Dmol4It6Xd(#EL};yXk-pdeV9D0Otm?p zzjT;Ow}uLRr9l|-d>L)+`L|KFh8BC_ifFg%orJn!#@D1}`wd23s^r$v&4y2uw=f?L z*b~-#C{1C~x8Isgh7HR`AFodYwye6U(zlM`wRgR57{YrnzOnceZ%=zs7~@Nk8fb4E zu8Vr;D7Eacp1xceG^~^azEw~%?^N9f<5~YcS?0`6^l2)K8CIRZ{~`XM*N`+M6!~Eh zG@cnNffdx3Qg&T+=lyBD*S7)i+6nto6k@vDH7DhL_8LHpe z7qA6Kx*aTnJeYyigzeGgjXt_YNPWJT%QylrTnMS!Dq48>BXI`WJU~a5$@d0!7pQGk z+L{U~`4y0uC*EZ(qZv^%E|44riGt3INFt+-@FYSC=LXH;d=&e#M+)W#PfVXo^+pK5 zT$|-~nX#_VN??>cUU2(FLjV-RBiJGbgmWe|nh0^wac2RmJa1aWNh+sKXFQi-?@WJF zb0aNCg_j#1>r8{p`ZeL~06#+%lU~#YZNT-=Yuk4;F`(*ZZU@8G7FJxd~78U%Tp#UGn!OrN#1|w|r13e*l zm+W_8qr(kBNPppVSVA@c$-4loV}T#5?aF5&9L|v&n#3%;B*nDONC*+=%D- z%_s*0Xg96x`LeW@YqkXBX~IA1;}3P+T{jd%Khax6 zV@SmV8{lrD9uXtmjBID@P9>{E#eu}=FP%8I`82V<1!YB^Wp83n%bfhmM*H#t4zXDQ zx#S-{!N^Enh4ZiIl5)~TYI3M^pE!L?e+194mg#+3`JHj=g78n0cb!R^+i{@1Q+QS8 z$ks8>@aVO!feu|y?5~zb+%}`eXiUpjP-)H;rV~!nesjaKaM`}^mk6*TC}$G*&wszG zh>s>#uGF|)7~%8DA%s!DEb=5g-dt_F&qD~1G*AGzz69n6Yz~??mNNwGa^}+{c6mUR ztJ@52xXzOr-m9c`Szu-Ba(&6GdO;)haLG$cF#_i@mBGaR?TnJ6+tY84QKvX$lsD47 zbQWdtY$rHFn$vyOY@m>*MNGSlG`v3po)p(0J6WVBg(Fo4eU-g6xcYr?40Q{W=Sxp+ z-Um~NiwIf_Ay@3|0lX3=#_E~Z{gwgidLnBtMXUrsjOsn$mrbdV78thE3~OA1a!`bL zg5`Ey!5&>^$JQ}X(7O$%eiH?Gnpn9~9-k2>3%s?XG=JRuZ_KG`qg@ScENLC!W0B8S zP3C7^F2`a3Ns!4IQ;Y(lbS~8oomJoLHxu+~$Gw{woh#cd+@T+q`MtSZO?bh;ryB#nkg)33|6(>J>oO+0nr9Hx+-f?IBZ45L9wVHx=hL0 zZp-QVmPq6J7m%0r!N*$L*u6k4-w{*7wpar`=nW(I5gA`}W#T=lqlQtS5u&Z@3EJS` zt3Rdl8Q-k~U_#)HYl9h+Yn~Ced6gZ1D=Gj*^u92 zur(?eo2QoVgZ))RBvRtc?$rOntF(iLD)`<9lk7`E7VF%#e<`)Zo+T_hn>vFoSAc+=<=4z z9CG=gyC0VEoFnFq`%0zM%g;>o+NL8Qda=%mcm185unM`u-%`c6maF2$D`ik~OhzT7 zP9@Z#X&$$+Nf;n#hr(xBzMS==>!-Z9DLoN+3R)EvmT*x5DSABN;xHuJccFsTX*^QOns2MW} z0mg7&y8%tz4$kZvo&(OMB|8N=lYl{dd>EvR@LCf913!11uM~b*vXi2>C*{5KUG<;e z$G^^%D+w8JlP&h;BsT~ zey z`2h)72PF)JIFo5a5>mei2N4~xo;ol1lV%Pz^%jI;lX0w~_cURL5=Dn2Hu8i_3JO9z zFq2F8jBnLlh?#X%4*-1wnO_tp4HKG4URRzp=?ieh2-9g}w+#IBLwu4uEpRJ!rdMOS zh}(3QVoDF0x4*>O58czc`7Zrr<47Pl_9XYcXYAghLDu@8`MOJU%*C+zsHEx zgA9cwEZpVzXeO%%h;sBJO-3Xa6@bV&s|qs6$h6u&c5}}++uW^>94k5v0}nwP-5n-} z!l!QW7;3B}BkR$R$F*HXGVz>rgJV5!Ief8F4OW-aU#+U?JPiEoiGr+3Dh`E8PT~mw z*VV=Wf8BA-Col~+hSd>cE`uxCUvLB>Lm!g*xUHC=ol%E#ZYs~KPpZt%o-+=;<#(v) zF+06I*x^BZ{#UFNo5Q7@%LAlJ4019^Qs+4}+wT0Xzdg71btny)62n`qFZ6=sDDs)k zFk)sXFJRRwizm%Uli{#d5XYw{8Bp$hqqg0*rzr3|Im^gkk+#4waoq0Ko29)GU z>qZsz^IMtyUDT#JO?FVY|+s4S2X9FwHt^;PT*Oee3() z=9o$f`MZrms)_QE6y)LFP9%SDGIJODvm;jWOY+3o0C>JCkx_1%(MSfn6=CV9(bFg^ zY~f==X=lvW%JlC&ner!BD^wR9SHs^a3 zgC;##jhwDU1>h|vm(4jYs&y%wpZ6annqN)d(;#~7ejCfhe#lbo{@`tj`At{#cYU7w z%BD*6CLHbD0-tpep}&EGSLvK_k=u*jx^lcL?JJV zcjP$aswLxe)1Upe*!F=xfu|@XXpUvZczBfcctM za@9Thi7Z)2a@tm61G(=lQNwxE5sSXJLnsF)sW3U%s^@aL^SXIjqF`4D8@dQe^MoNa z+Y)_r9;7}c{Ap+VL?vyOG%jOrm@+Jy+D|{o`@l3370J`lh-gcs*o1nLQq^-(eJ)zM zVI$k=B|zxoUFZ60G(UUSijfy9T7QVXe5~l?%^#C9KKR*8KD3~&zL6#)J1dHuXBNJC z3EBAB2YK_#V$>C{m|@|Gz3b;(sQbZTYbkBED#c!#gMZVHir6s!P6x+4-`}D)Tl}Vn z5($j^MN-SpHFGXB1Osyp-^;9u>Rd;Wxe`Y{LHBAWR|ME=qb200 z<_a^ke3{vIH%zYXY#T?XjyjUR3p0hb&*>MH`X0FHn~G#{_r|x^*t!;g^EQoZIw<+bwwc^d-N$s8;Ehywc=XJy-Q~mst>9~(su>N( zThTSeMhk$9?LkkoJ4`Fh3;LlT{q<*2HfJ~XN>~grn9II<1g@M>3we$utb`JvlqFY= zW<h-Mi^1%P5wvgt#=JpXo8*UwT=Cn4W zPUa_w{ky9#M}IBEM5_UkWj`P>3#OfH2Zh^=4{IVq#7{#!%L*Hc@ppnT$(Bl%vmwFP z^KH_ndr`C)T}ab54IQo?%BX4ad`iPP)&0k-`XHGUU<#E#7jl08{VM{DI)h0SnGa(7 zjs6(W194FK)y0OlqkwkELpCjTxMBSGVi0mr>8|qWEKz92K61}uPqv{phMjD- zUed2)*7C~g909+SMB7^kq&pX*6l7F27IoENWObY#4znC?mFe2f9c7Ss#17)5JYs2;~-$?hNZ z}Cv!`AdE` zRFqr_f$T4?6+94-7*YS?x+UaEdwuK8WH-SV6?tv)9_#w;dhec8xKK$MW!8<}*7SrV!u5|~D0nBZ(}Y%i zNw3kAcan!f16Od$hn)VAoLM8@12DvJxB;#R_^Q?0mX_syv8^bi(b~Y9XD3bucR>$y zTZt^xZn46130)89rx!omf7oAz&}Vnh2Py4vE^naA)6^|#fwo6k|&h~UAD@(A@AiBtuC z#^^6r>~e&By%NS-C$T2Rag@+YtQE=3htHmfDus+HGUviJe8rn1M4L8$(=1NjDCXeJ zX4M_;QfS(J2QzK%@?*ohWOT2~e5r0Kvd)7#@kaodsUnvib-D(GOSoy2|{cmk3)21}2O*e`v$3TXwea~a!x(g(esD!!oy5Xe6F*BjKf*p!cY7vL3kVm38-oeS3%S#9 zv0{jk%^C~S_Y$TBVbE+VWe4FTvJ4P3`#eIw zg@+m%FT}TTm8*++olkl4HOWI=pt3Ss-~V_9*U>fN?w)GGph$eamuv4zHu`)GHgU0} ztqGok+Tm_ZP2A&Mt))fO&h^``x7Bx7K*SDPXnbgWHR=>2(^O?)qw?M#<68K#9KAz= z{6VPxF|dF5zrUyll9nZX?YO~z_n6BrcwpmuKDq&jf6Qi0G=+%GWmk1V;2+Bb0-2PJ255u;y%V2etu<}+2Z-h_geDfk`m}y zN+?xPI`F3|ErO~LwZaylKT2*vaH7yVsZ?Rc>1;?|w@$Rti9^(0iv@Kb?_e@CpZTrX z7+zyGF!OM7u{14whWQ%`CJ*A@vLXZ0@F^ET+urNOa7oVazMugWCLi2H4^`GEPT|{f zF|Oow$hLR=OEZ=NA+#}wR}0e~2Syp^=bgxm;_c$X@TEfQI>or}-_rw&eGZh5{LRDQ z`wUm_Uinkm$T3iFbL;n=^X~Ktx-we#>Cn*1xQ5KOyWjmC8jHU#kKq3vL!r;s@L@)^ zt;}VKBgvGwH#&HxU-a#SOf$vjz=JWormc2D8AUwN-mr2sKg+m(`s3%B0f5XF{MGR- zri25b_iP8kP%dXA25Y{yOQ<02-GSC`J+#=btf!qar&l(FJLc;2vsc)))K6dK{j+Z1 z&C=sm0n+wBg)JBAkG*Xr{rPg3ZgbO1pDv=PN?aV8yqG&034H z^=g|;o(Ym;6$LGkM#2uXNpg}SRkZD$G&Z*6AA$8m-XVwsl2`JGPJs%s7yTbld0J62 z35qGTgp3{TC|difQ399bgn1hdLX@#~1St3KnAoJ{sopfUE6UzI0_Um}Z~Ubk1W%Xw zsXYEchQAt{Voyr{!oP$Uyy0E$S{EB}-`GWUyqT?HjyzN;!ri%==Ca7YL67<^j-MWj z!TQ}=j%9G;BamfzcIYpt+$UR@(Qf~cnbz%-I)V~+#s@vy`;zu}gTWQjuu*}dxP1gR zT(u%KKeDv9)?1aHK>59FG~7ADAFZTp%glp@5gBT3ofkTti{$YmO8q$&eodMhJRpTX zQ_Vb>&P`i9#<$e7+8KnQO#Vbq6jlF?&3yXwMc#?z!aW(+zkwS7SSLdDuEMuzxwPmd zFDCeO8Q{THJRj6qxE;~3&ovkajMUg}UnRC;a3Ho0`YmV?vF+dCtCH2KY?} z?+Js0xnMb^aaq4PS8ss|unGVUj^t)6SW2+0c9MQUK<0>W2q=>ngVuSllW0^G`slwU zMKzQ=i{ki;22nQyO|V7DAf`^*J{(s&Q?We;{^YibLd!CD8x)IFD4+r~-!F3_ zfQXzV(kyOT!Wg8iYpIN@Qpym0&0IRk=WSyug3v|^M^!Bb zuPBVlmVIT`L6~RPb#nedD1K1L$dOqDi8V`Xp2bIs`%#8F{5rsIo8t@mMGG=3Wpwv0 zET<;gT$Qba5QUXUbcB0TM<4IyU+VL|hm5_y`vVOdR(3Yxwmx`>6*iwu($P9hzI&8X z^uv%0VU(FQDHM{Z?t6bc5Cv+f6#%-slEk%8ssODJ6QlNBkqlw_$I7LbqhsJVm{vKtv>;>r8yq zL*v|m)`fa&F2{o}&Icp;lDncM--y4?ZU}$lEPOVjN6g-nsTu+5Zs|>(YZkOCL+1>(A7KX-=^&V-F>nfTO$| zIEUBtf+PvkFJcxU-5d}Lwv*<~EwLV!VCY+zC{*q-+(%NdXS{vb`6Yx1?FkuqJUfG! zMDz|vJDHI;!hxY;pB;r#HigfF9L8>X`&XuCe=-#gm}cFfv{Pe|%G-V_>(cZx>gL#} z;9bfL;kaC6$`!8}wS&}|T4TYIe1Ir-n)lOIFt#ir;%^?}=`h=`tWY9wh;~5l*sylW zUqo^IT_?=!{(d*})FYieIJXdz@if_lr47^+eD0*b7Nv}ak4hZ4<9bF!XfX~H!!=CC zg%*Dmp%6FdB$R*xe4!!ra;&naXmP4vv5WI|hiJCN@5uRU4mMamgXe3Fr={Pkp$8bB zdDV7QZ`FTLY!bf|B;d|B79s?IqD4Pzg4%z6;_-|!?Upq)5f&!jHKaQP1_M2+t!CCI zN~*Z+W8O7EZ0(srnxB}h zQ>wiq3afHB`?UQe3tSbiE7yTvDG^YF87qbX6x#WaDq1rtKO*XO5tUgHY6Kgjk55Qo zZykeE#IWJi6qxRxJJJb*G1cxQso6~qnQYOxsG3i&>FzuqUEayyzA=mtzIPSZ{xtY@ zPnGX*a3i~jhW9HsDhl{KuTON=H%7?YPAgAF@nd|5zlN~*;k0)jURzXYi4Yr9vj9OM;zLDM@gyyFoDNRMkMSwh$uEBQDpj|a``u5)N>LoejN{!qJafQXpM6Y@jx(UWVIXhRo|xe#66@heSHhLyBpsuMKO z%B_3iv(+n<$uRU3bHRK=(w`Wq>PT&yO*5HBf|uuV_E`CWlO2isC*#aXvr>>i(^_9! zZ0N|EFE=>3kM(t8d|fh!)eT7bN;AR{{|Z?1-|pP(#s{$fn9B5X1eM>s)ZC+EJ4h>xJBLg?4 z6a6e4&|ds@4l3gpJ7wkHoEJ>cUuyVDs&zt_Ky=@FrB|?r)6y-G0YZxj)L9Kc5gz4| zh%xm%CmLw{TOzY0v2ly8+4H32*VN%pQbi7#TX+QE8u!lg0OF*o|P?i(f?jKZ|^ z7GipqMcbkvzEL0y^HD1bGC~)nMLh$ez~<6o@nP&-Oz}m!-JadfrlS zFx3Wl9e(C<;b4wG{B~7gT~4}LeuSW=%0mP$#d$}U3%)+t-S2&7Y8*Z2x@A@O1Lfp6 z((CSbK;t->pH(d!sqtJ&)38^r{>Z@+{Pw$rfWO*F5_nUjwVE<2sf3_T{+vPuKjwnF z!-e#WD{ z{VT!4w4=eCTPQ$8qQUoA(qxi1|EvZA#e-oPkk8MT zQkxzAcu9;VWZm!+gn{WZ{AI?}!;*;bJ^1oJKV3t=`P2qsTjp?z;aIhBId%#WooB~5 z6g>k$bzxG>tf%|a`otNc3ttpMR#K0YCGk|5|Ew6s6^k_|-1-K5 zRCOJHZ?$)u3Nh1GOJY5p%UmDY$D-hG-eI7`(D-32EbcMq_{+-FHnb_>Cj*-UN)gG` z$u6{pIZVq+h80X=UdO$<_*HXErFG=d6JtObnjhxDxu+~Bjp4n3vc2vb#7f$3&37rr ztyBQwUH$ZPyg}^&X4k#?Hm&Pfi)wMT7rjdHTh)DS2L}I+L*q{M2Kt6Q7917fK6Cgc_U4($XR*^{Ic!P`E`8NFlcyHv6e-?w_-SGJpR zGauBG4DL8PLra9{*X|f?k3?IT;!`H4N<3garcXYt7x zguywmCXMzL9-`TmgM00EPd!(GPC*VR**6(ep6lF5*Fw4t7ftMa5+N!RZ3+ z%paoiLsP(M>5 z%I^#JIs&oj`^IxgSUW$zlO_M@yfiE*sd+yEUKmod%LXkaME2p$9`f6KMNbfkCw~fh zEatS5T<93Scwa&N)7sb=bJJv1HC=A~;56>Ebtx2^y>c&V|*j1XC8zvY;_Rra)6)K-|o%7Rn-007Pz5pep_ zfIA8lIB?WDsGG8o)cMBf_o&IVF@pIeeCT{0et+AFO)Su#9#kMZMU~gshTZwX*|4cH zHGlRKw{oMR0>@?P{d~r!q9rA!5~4C@!8S|IKs_TA?(53 zjA5pXI%#aUpk_Mg6>8Y(%J&>Sc4kMa`h$okpJL)2A-?S$D$#r{{4?#*rFZ5@hQl(! zlg5>~SK=<>z2qi_bIANE%y}=;nWvVg^n*yHPY&aoP!^F@Xb5 zwj8G&d;Ehr?p$T5EvIw`FCJ3mQ8cohYuT9vt~fVUQ#S{^*+5(}Q8J;qbRf<@d^0R# z^(P7OG9U*+AoBScal!c8Z3pfm$0K5ir8Z;C-{q$gz&wbfYSLVrZy`jr>+a3Fb;OOo zbElr6w*k+RlYC@Qj1v{j!a)yj^^VG1;Q(dw!4Fq#+JMDTh2K^_byCQvdzaAINfzI4 zh4RGsBmYNRo>&s<3x`M*;R6oRJ(x7%%KQkqWS&o?8`6u{G=rGneDe+LEAX&b(dvfv z5QnAW#`8ggU+wMgU7zoS{?5WUpisP#*@jLIko}V5DK0iALZ+>wGi@V8%V zS<0H3r=Av^bipZJ8jtnY3Qq=I=Y6V5A#F zjv$o5oVVR=KWqL82hVyVKEjdXtI;AhqDz*9ubT;}L+u;YqHlluxTM75;{_Uk8OE^0 z+ZxYV4sezAy*#-bnVRc+QKQI`~Q6uuVgGygbV~qV7T4f;j};gwhH1`Rv|J z1N}==N~*oZJraCOTZ6zozFWAPO|y(65iJ5&oXm|Gw=1cuF@?l6m~U6v;m)a?<^S5~Fhc z^_Ks)-2OS}|46k(09G>Z>G`t(Jn4VHrJd$KDCtx0Xm|Yo?#lmXbq61mIMJKC9;Ixt z|A5QYntxCdQt9cj^M9!OU$^nU{@r{g{){-B1m^Qa{wG`-p?*?=q;q#Y`0anysQ*EG z_uYS$Yr;r)VErdt_K|*4g4cAs7cKKoW&9ud`rlUfKMDL)t7_z!Yr6l0%Mh{8!ck6} zzoU35|H0P(|5b3`9nwFWm7IBX1yJzGBy?-1Mis>{eR$X?9tZ!P$S$-`J^qmXvbTli z{~-X03noXn{qQ5d1-u3y1(DO7l1p2wyESM1;?Y%Aky?97S1%Gv#Q88SwQqh7<+-b2 z2p5yhmc7JU@lU535e1ey6XzKWI+@SwjLLgVPoh}jzhU_@;~!Grz|i+_3h%x^0b^e> z-J_;^8&!U7=O&PvCrNeqfJhZE0u*`Se0)V%RU%L59l(xjw?k0I5BXzosW^sQ?_Zkh z3;Mcns}eU>)-Jc`Sb+jlH-^gR4W9SnMchCBz5M+5j~n(LYPaA2l8#MiiGS>mJ49Tw z9|Y+i1_%s^t6?%Ccjt~Rul=vN;hk2Ru`Vdm=ubKcw+NrWB zq5LB#Bw@B@f4q5c9 zuuh4-l8!INL}KUa=AEuVTg+&&gi1CG171ctsjValSgl zl2grE@l#J`3D!jilLiUj(83UY#V)`WX^!46#(!UOfI-YE*xZKt~)Q^bOTD4Z&_DcP4UgMk4LjO=#p>Lx)uLcrAaU;@f7?~pP zPw0#B^8q1dm{E6L#2J5)$O~^A9oA6E3z~C>Y}v44Z@PuLXz#}O^k+i`w{?@u6ZqzD z=CoxH33$Ic=dEoSo319O7C8aRcvO?;^QxMl%6!G=5FeW2xr!8=w(ciCA?TF6y()0B zRfziAE&CjuGhsyad)J?IAj*=Wjdm;`WN2A$Ph6>u6YyfU6Y*-h=mFAXc}-L?a8m~;yHhLdI3c#LAwS& zJ?b^8(Dhlrn^i7k>&jP${?$Z9n-*6`mFX%PFw)6IHQW=VFs}K!$*E}#DaeWUzC$)3X$7<_IbVgYNc7CA z5@GdTh}iwRi@;9k{}$&@s51G=0ierAzq7S-oVFF5R}M+)%X6e^$g@#Ppxt%3f|$x) zUhX!NfcK^ezBN+d`WWTqpe|8RYx55#JkZnkTVu_pdjyuUPPYj~jME42B2l?=K zNw)Ikskp2lZEaOHAXH!CS=m}-ny)nMJqm2L^18eJMvEGevB zjrSA9o4KzT3^%!2{A?t5ve^@sCIEwQ{#kXvFSlv8-uG0^m;9>tZ%m->55T9*cHa2? zui_k5?r}e^4NrU^nU+h+{rs%5Q_&?s*B2#`Jf?5m7$rR6{pggnPIow}i7i&Zf-_YW zOzFnjQcTiUyf_ao>|rU-u;(sLCT9y`aqf5AJZ`)@)0eCJvx{xT6QRR#AFPXw-B7Ue z4wUiA#;GKVElp|W8TgaNhl0;cN>+3D%OyeyKHzPvCYu`|1fbWtw$TH_zFc;#n4A(3 z)GwMT&nT{bHSj#u9MYJN3{DnG%)*1>uYcb{r(5W$qZNQYy0E#Mh9Ft;4Pj-}Y=0Ah z^20wYB1W_&mShaCZY7q!L_|K3826>_D6!zDsE{lqsYGb53kbKs9GkeX>TQ{#vu_KJ z8_O-~`d9w9#9>Pd)7+8%9!xLCAAT$>!>+Cjg}EQ1qM-iG^YC|VC`ZosdYSm3^PP2>sX{LP_ue{FLh*_0>Z|KeO2>6B+;eB5fzf89ZbL@vOUq|W zyu@22`mFjhl?nsrFFAH)17YUxL5^f(C=0bE`DpQ9Nj_9q zZ6jrGDB{4TFc@I-0#-$RSPgTRNB&z^!oM^`gh(AaK8V3LsW{-~tKq2m;)eQE;cF3rT)+>MPAykNOj+H}0?nZG09is4VEt3@Rn8fx_4~n&^`&G# zRpqUA-!rh)R|TZ(M?KVW@<&Nnjui9l@4c4fGuWKeT^L#ouO)V>R%ax0<5mD%t)4+% z02uiOv{(pPQx|VVX)F7@J@K?|5JZk&B!n08rlL5<1V&nS|2&w#Klvf7D6cT3BtO(` z)R~dl4g^t-KLG6jO5w!w2jd5-MB>Djg&FmBJy`7fx`T6bXeEQ0f)b3H2zA1&EGm-+ zZV^8pvMX<=$o|p}mp2LCl@`*Jdo<`z_$NFKdLr}CEcCs2fQ9@(Om&mtdh3T-`4(r~XwZH-WSR&Nhmjfjf(QU@uI|wRf^{QQSO6bDs!o@045WxY(FzZ?>SJ-*_zx!tOZt?#8zRs@)gW5l@A2q@Z<>g<>F1;uS>KZ z+_v#)XyaYM`)Wt2B3z@_>)AGy-H(qVp6ixHHUFCrk#dYp&oR4BZ7~hn{q*eB>1eZY zT=bUMU1g1!qHug8jA>|R?^!wDCLu4bpNARzNt`Q+wnggD*S|yjj=TlFA;p7$H^ zDX&`tcbg!x5U;e}8Uh)0alVFDpwh-p&_QJsAsT%BHIYDxP_f&Svk)kIrch*+ z>g^Z@oZgXVIevkYEZNf^d`YKTN=DGhG zeM`4asJ*#!tDykl9C+Ief)o%%qGFL!I(PqVl<2=ap6Up%Hb(27qK6uvHx9nfd-2?B z3v1OwhAvKh<7o4x>zQ0wmH<*UKMI)40IE&NhqDMn*7J6yb&8X>n99?hOixC_<0!ji&mZbnA1Hg%$&G#iMdM|lvL zTWohKY^)3iF3mEbD*h`tXZ7tb(50jM6{E_Q)X(K`GbM9diwZh1tz)f%*H?o1g$D`S zZqasMdp2E`&OZMfHw(;-=B^u9JD@G1u?y-r(y+zQQq%#VU;yq1SCsb=+*lragvrFD zE9Aticq&ByqQuF7Kpsg=pOtYT@3xhg#o0VLEReGtG6M?jI`{cVRHq7nSC1X|Uaid*K(Mc!gMZjzz zurn17e8I5qc}=sHmEc>`G;9tR!K|4=cdsNRS9-&r!X`+qQtBZo?#PA)I&xi$fN%{a zU6EeYWkGzA(%%nG4X0qVYhuSQ&AneK@J1v{bn|mJ?ti zo^IEdVKeW3i0(2jX@&#Wj)>P%4K;f{D!cnc6c^(1chq$=Z9w3K8~MkVCo zNjgJ|S(0lj3Mw)`0(U<6f(Ta^ngt~_5+iuKbiGgbhEx|6J)BGd zHon@0C}rK*GZ<}txkaM0`f?nY%FYfw-E{VHFVE)oOLV)H2H~!4qgzG?(NPm8uz3Rr z_jcp={dMsC6YzqC0Lm1JOU*301dPFajzx;?LN7)~ zL$iCK2rh6fSu<@bk{Y3CXaA{>xqBJq4bu6gJbXXq#X>@nazCr8)>wmI9>f>Un3KmY zUO|c3#lxY-zleet(yH2Fs9 zrhqe~SMyo_Rwjq;pKPD*ZZKFJE?5ZYAmblQ^mejHQhyL!yvRlpVu+un1Z*~ZQNoic z$oqUORZt^awv+()`=P`N*7W*aQ4SZ7GAXe>BJbl!%!1f~^I(4=YSdTWv?{V&9U~Cs6xQuW(B5 zsKLLEVYm2yhCG~!Xbn7m6Lw{Am|Vf&j(ObquDVW&R_3XFyeivh)x$r0xF9nfagPBl z2?bi(1pmg|?&!VotAD(y;N!jXP)ue9nNzN)RC+?5NU~My{0`{u;zvA47o9tO!7;Fx z5;M|6(3WC3(SkC$?OkCuU;nWx&>IOiSVGa!511$Wxb(z--PCG#Yrry>u@}9V`hTzh zmfc~<<1ChPiI_0gcZhoH1P4i%H_p>BQWnJ7ny2?k(5byRj1Q2lFaOT7?W(d*KizMe zZ9;lY3Ihb3Nz5jS%Q%;j1%4>K?7mJpeyto3##CPLiy;uK=%$2zFq-a#b9d6Gs%Q{h zE+lC4Em@1~%i!GEqnz*C3+H5Zy;r4!v)1sqkGA+kbOTltHD8@-6fSDxL=|_vkct1U z5Z~YbRNd=EFHImN`ojgmuSxx^fV5hHnj>w}GyAdzB+0dA_(;F&{zGQFRCY~&Mgz^S z9kp)6$4zaHel0i^(NbEjr2a2e7Deo^Cm$o3emoJ=@0iuOxmM5h5`tz+C#XbkTmpUp z0z3Su zQZ&^cx+q|raF-4?t^fiT1|U&N>{bhcnL zV7?H182R7D6rO`V!<8?P=uce@m|S%(WasY^s38J(Fk@}bJZ3zligi~+nq9`Qu}}6{{|wH?$&|IR_yE`REw6F{!3?)m}eN*IEvi(fFoz>aLq)^Th~ zO5!%FLNsHyozV#0IksaO*_kj~p1gzM=O zU!$_<-eZ?G=JRc+zvjyK{bq}V&-J|w7vg*&)}nq{>ZDA2n;)EuIPbP?zAcdYh39XM zfzbJ-bG(24_xRY;0~duaFrAD{_G}%j5=e6I{>#@4U$ZI;4gD8m>Im}WYog25hlK2h zQM9Vl#F?%O`o{sQq;cR&eAF>bS&5Wt4k zbV@e69VjD;==c`uEEfxSZh)spu%ulq(`7sbt4a&+awDPjOB>W6dK+8EYMw6AKsY~E z>W=mE41J<@k%t;5$jhW2mt}j=8J`g=*J>gO544L8JDQ2i=I5QM*Fq7f+TWG|@@HPi zSg)hPtBiTD`GSu&o7c>u3E#wIh1sqpEFOB!-8nPY7syra-LM(f|6lCAS3px;*Dk7p z0wSOwBGOd4bOMBqfE1|;O7A7~(0db55RoPbB%vuyq}R|pL?9s)5kjv)dIQ2@VLW7XLBbZGCmWw0#{fotj zsqwz}eEH9OP9B*V>AQ>HiBTYY@R~4XF{DFRV57Zp?Dh?5>2Ue>{NkQ)La}Mn=bBQK zJ>#k!GpsZpUqw%H@*=GP8@I*IXbcbQdPP!unewM_iBe1jm5~Q=BQsA&xUE>})Q^&%gC zzYMkTg9#dq?Ml0T^X(aeFbIB1-2H>3{J-L%jQE?{0qN(+cwZsOS`8 z**&R^D_m!J|6BTP{I-LsSmQ~De3eWX&tn16#AFvI1E4=|6`%WEQKla_c$ZG#A&?g; z68&;J|Bl>OIwmR6yhO6);y_TPql&EgCb(Z}JJ!Y^p5td>wy_;CiuJ}$0qntC{SuKZ zebT_E>Kju>3oO5xn4ZzF=*rbU5rw+Y--UYI<;h1{PoC#6@`{@eItw`KDBxQ(^YJ@A zDd6V_>Ys0n&Sgcs)meLiH!jLm8-RQKV}aIIz3jl&Wb5m3#K$n$;|9sa4Q#Ak&X!lK zlR`m$I~@hog&kwyNmcH+=ZEE;8Q5nk?V9zXJ8L8v;`AbNo!S%ArRn@**qWLaE&VJ0St>_nYEPGM@z0U_gn=2!Jsv%FAW+qBr6=_?OA{u z*Dd6D(QSM+!KpoFNDdW01V~&EhEO1R9#uUti_9nfF?wK4~~-gFs--kB7CX1FCk`{TNJ* zN9Vz_kkj9OL^T?uCx=c9$*8+NM6T&v$tH32KYK^mZczE{rm>RI1hdCt+mv+ z1SGpdYvGs7mekwp)aQhWYo6{88OvUV&UZgb^a5Ly^_rQg(?)>$`E?F?7fL1(iB2Fx z;Dgb4pP2JL?dKF26DLv@nW-@48fmSmoxKfNkh>cyC?$0No^B3v?n^|bG~i+JhQar3 zzk!*8mel_b$o}hc?(j)=c6}wQaXGXkCVyvd+T4<1H?)gt6-g8};mhv^nmjqII+|kGaT?hF9{qj`;fr^UOXV37yuElF9_kjBixs z1rb%N#kCHG;c|bHg-Q6=*CV8kFrw;`i*!sPkif{P=6_c4u8fe#W$%eX1$?Mik}f-=oD&xB zOIZrV;0=3Y0KnH9v(aXW@1EtwE*_X)D}UR)`(s3rObAuK%p}HcBDUB6y8>yN)OBnf zmVaYs`qmaEGK?>rUowW^7q8(EweRh0TASc12k8kbF9j)k0F^0MpDNOX+^FNw^{dFn zAT{_0BvgBTJwM&Y*z{;lcbEy#h*7mfT-7&6MaXn&(8mHgvDCWOQoge~piPg*{XZ0ENxdAu*xm)g{`8&R};>~}W|l_vyTapP;^ zO~`kVsaYFJ<%X|uP@cVE^#AL&|J%6g65fPk$ybvvGHiLIQneP(8qHQzXA2xAnwjDo zLi=^)lHCRynYEjLMVXd0^+AdCe^m#)1xmR-w%t@VJ9`WX**=YU6g)9pkOEl_TmZeH zX_;27gUFPP1_-b<#Dk&tVjK}cSI)bPC2T;8*mLm8*+=H;X?q$`bf?B zyZ#&FI%DP5fseD$hs$yL%r-xNWP0IbLN%v!m-20hR0?)4<1Ye8 zY&I+-Kj_Y~AmL3r4%W zI~>&bWi&Z}OT%Jxd^_#<>{*K$SYhW_(H`*o#`)gPQ==R8G)n=Jviv9sh4wFe#r?vz zGN+CsOwV+gCYbh;&X5mve)n7~#`NdCuIvNF^Y%5IFf z&8E%J*Xa!n&z^}~o`@S>U+khl)!@52K8bmf)*xMho_!YVHTbRM=*=FV|i|E(yGw8dPX$zNZQXJ`nFrG zHh8v$WlxK8Cr~y zB_z=)0B&K=ipC)aq$^Y=AK`g3`*Q8FER{PE4!>56X6o}CFk*D2LqW>%{WsH+w-*WF zp#F(afd6Vprolw=Sk{_SYwevQ$QyX|LzqGgYj<4cQG?Wcf*^#6yvJiv`_8_gC^Vh~ zHRq^_-ATEw5pte|+-ttKt<~)3WOGkWc2_8-wsB!+a7B7J?_LJ$cfU>v-O3J~bUV;3 z&Ptz-cnuaQ2wIP^qiQ>gUVKl?jN#6^R@hl&0Nw(SScq=z3g3L^?1;Mpyiv+&H|E%b z9DfRRHU7HTAXx6I4;3RTaIcf|CYllu4B@ch=N7CfJ^WNT@N?|k^15+6;ZDdU=a_1i zbw4=OCO!edKT|;aBYTj0voh}}szT=kT#+^-iGtS_Zg{MppgmkzLyaIYbStJC%K{$y zm2UIGA!&6tU^#_~n{JZKwX+-B&n;_$Uw3Cq*vUlI4lF^U^sS=7sx}43Y|T>4t(L3~-sp^v#Vg^n*THT6!uHGu zUsxq#8*$?Zs==ATw}$7J%IC=4rW*?{GJeJ343IC1kP1t@^YiA-hrg%72WUSk6c)w?CLH9nTJ^wn zA@_G5vX-t6oDn^kv)yaOVJ+%;3@#6%bCo{lqlC9;jD3k9=$^c(wD_r`mqP$3#}=wj z4>bF4LF8?{%Qk_}8^=9bvLWe;xhyVgb+r_-80WwD#b@r4{9QJRcGD4 z3WM|*MRcE@e?hTng}}zJja8E}9}2RfL8I#C<<{F}6i`h~>?vE^mT~`-5KWq7C{qlO zg^eBYjstkcqUc(m(Yk-DtKUqmW9s4UJ#oRU*Bn|fw0g`3huid;NmxDy<}qk~>^e2= zy=Q5yESZ^LzdA_3=)lSdJXM|hvGtg+7_VV3-1=~VmO;B}z-4t(cA|PdEIzP4=GN<* z&`lcHp@-XuG)L3%4Y`tohccDE%a$569`z|#1%4WMg6GFDH1Oaxpxud z{pg)bDJOmcR>w2Y=&a{Sz%AG?IRh^&f`Wp+ExSt=^S92po-{bhS8eV*#+RK#Ce2a# z^6~lm+jK~%-G$=I)H@9W5r&)pEVZ`p0UeR)U(Zg%$mbH%CvTjo(a zc-}Vqzd7%JJI`;JYd4`hGUuy|6CzowXxVNrr@%M^BKstf*riPEL65nbCXqV+cQDzN zm1uRsdV#|}SYN~O3snqKv)=4n{|Dy@f7-(%1@9ut>X?RSj_bqjNmY@(RjBI7@| zc5j?B$!z0Z@Mn;0%~zRf-iO`?e!r0SJJrb*?0@-^0;fdYl+H8zWdV{VlG%mM)J;U7 z5Z%;W>~F?M=|k}d65lU}(H;z`kL_iF9RT1U0AMo~d~{#;Q-p!u_SG3C%$s}uo&z)@ zFv*^LW{;s3^P~ML0`JUc%_UKp9}i?Qn-p4~4A|GAtp+$a-K84Ez&0a2OwL-e%4R(w zag1tK?bvJ3sZ+Yh=>REhFT7~g&}K6{zUW=JX_A4Ut+Z?NA%G+(ka`;D2yR&qkOc1K z#FaU*gXau1Uiv&xsdcfj?zrm1jAUg6CSXk$qaCK-r#xb|&ErB2XbHT&1$~s&#M-Eu zwkso7=DCniP(Qfa=3Py;n;X~s;;77Y_}-^-XX$upJUJm_J9mWXaJPcBG!Ds(J$Tja>|koO%eSyF1~-& zaLH~%CJR}D!<>ycHMuW6310}nEd+1w0@u?EULfd=*BSmp`?$1&7iLf8J^f?JLo!s3 z<7|tAPVTf!Y_>!13MdQ%9Ps0w`z;|{aH!EYN(5HW_4Jil;ISQ}hUREQ=@8@I@Ag2~ z_-ymWfSgFeuBPk31#eUla0#c>RdT=z}gbFY&5#Pfwq zAhl`gaU4YPkaEAXf0736iEG65+ZA*8cfQKJSht~6!hdNWg}b??l0J?I-r09+&je1Y zX~a%yoLUhhrQi>&-!D zHb6-wk_@?VepDy4DoXQ&QC!c;kq+^0(Q=>>xACQ8IuqCVgc!BrY`iJ=?b&54WpB^d z`v@3F&cmT)iSHU^q~8HX?thqvHgE)G;vcY-9oofOt7O&}@SxUjKrv%jt1xa!(|ZY% zPQJ!5+&`uf_-@a%-Sd@=b;7L*nd3b>-F&5(|CYi28xoF0|3E^Co~mBQO|jrTVpsHY z5LVXQM0(PFcKr}=ddy!9FDU3XA?cq544wM?ppz~peyNV2$NDg5@}}d?TWU#r17%xW^DGg zh9Alq4bjl<{v#{&cWB%T}S=WaqcGAu(@!q z&gs=MO?7=NOReM^Lb6c0_P5<(y+6pB0!D?q0j+h;bvk+PXPNe-k5<4ID9+0^YlF+nSFVGn*1xo@ z7rsyH>Xn#fW+`7KDixi+B#O*DAdfk$PgzK=8#c@ahUL7Zf5Ptx>s= z`VfQ~@VMGlfr!PfujeZpgVR=sp2k_UL_ZBAC|edNBOcz80i#(A`{*(%9qu&H7-$h?3&M#AXixQ^Pa@pG9z10&ZWKUDNBM6u8dc0TV^d@rq08Gr zE$mU#EvAJdIQjFlXPgs+yu=#sM*yIU)le~?^JH4IdFwe=*=hRseFlVR#^aqsUbdAT zv)X}lL`rOoLDRt%Yv9RHjxR&Q{+0qxE7&2iIUSihjhxQ({0Lq5_qspAh)MJ@n&aB8heY+;>NsE?$yN`S($CTUBTEma#8D3^( zPBWyAe)V=21cu-!Ihu4s7c>MXdB>NZrF#6(!XQ#;7AtJd_?}d`ba?E(cV(nyWczO9E?hYRImxLy5;E z>;O`!n#;53R)j#KS}%A!iZZ^n-U)zSV2U6gGeqt2Sw)I?j`_vFu>c`}}lzGgo?5)Jwvky-d&%V*FBHJNyRbDej z`DJQ|RwYIXqpsT{O%kY_$L~MfViys4V>DA*mznr5RmZU+A=R#P`lqFjXUQ`QLw@_r zrrroeKJZ*8JaPn(&mTr7%l-Wj{+a^H#|{Fw#_F8-y9RjdS?a?zEzGoYivg>Nw-K%;4{wtdQsGXwMHGmGW`U!fIMQhpckNnsZDrw;lN>}*9uLHZq8Ma?GUDmI+nKSO$d)Y66r2zzmgdc@G!&p*Q0OFwu|oe5;r8)s_go~pYC8!9P^(*nlRWNbG*@VHshJY<6x0GI`#V$3r7}JjprbisE+dEUWPHW%47g}*!I8=G(5nFJ@%;F~nZ z_-)Ac)zU1w2Q-q0E{_ybCGINHC-2fu`fS-8gHL$3X<{NERqKrX339V-<;DTjPRHw5 zi;mgpqVb&rS^~Yo07E!fk84sx^z|GUm`81 z8+Llz+KbuU$-1N;@&Zr97bZk<{QGZ+#1(69-VQwtmad>$EqgHPA&<*F05dv^rTfk> zJl`HCb=s%GCX+y!V$%Or(s!AH)21^VUYP+hef z4ff$GvU8EvP3jey5Bi|X_t=T#_V3NO`uyF=)SuUjO~sSOP}YF@Z^=_Q8L{c`9a2ZN zv9O5W(>bE1R_j)mou5S~rQ@4&qS<(%=#F}I0}%hs%>hw5qFYfB1@)?WjTFHFUim6x z29n)CuF3PI<BDPO}4lNEsLkBJZYT77x7sViJuEs$POYnjyK%DKAU{JiuSbf~SJqaQo~c zZA^5WJb6llrCcU<_&9M<%HktPMD%XDhQzhQHxB`<5+g_p(Sxs=QlCxJrO$SD@SNiz zx`MhbIb3p4)zu~+_&~L8wh!DrI(xgLvMpSjrcAZ28`bIrx-3b1cxrrm{Yt^|dHte? zNN-_PRN>~jqa%0Y;+ie!g`~tn(AM3#zWugh&yLxnrRC)$V+*{#b5^s%JrY0?rklc7 zfTgKfX2%UMn>7(H<0Vox6zia6cGVxByP4j7N5xKm>MUXx=g~QNu++nmR%9VPfH{?x zbMwH7Ex!)x?

?d!PPwMAo(=qGqxj~o(BE2 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�Os!`$o1GQ)mAoW?gtIVp2-+j{RM6#Vf9qdr__# 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%0AdjlEZ`-g}X)~4+(l|mnY|FUWWVm~msZ-O1 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

^5I_C7n>G4OJuX&J^h^RlK@lHz0>f09*WD< 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

CJZD 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%0D*sKNVP|M2t{1C7NuMA}Zx!fiv}Mgc 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Ƀ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:p#R@5v0y~Iq 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�F_EXAQOdm@P+4dI!t((o$ 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

(b# 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 literal 154107 zcmXt91yod9xF$xryIVk{K^VG`E&%}n0g)1ip}R}EyFoy@L|WPbk(QD!0f!Xn@NMsV zkF{L)E(7MAz4!NjwWD;jp5o)s;Gm$O;H#=A>Y<>ZsexbGSZLsP@M0bP!GEyfD#orT zC=?I>{X$jMV><@FN$jR%4BKNZnX4`X;~bt8xQM;F%rl%C3z6-J&4nw6`J(7rWOfZ7xp-b6K2opY}zjYKt?BelcsRhkwqHr@YR z;+%cf?z@Fz$n9c7vVWufXtANnQk36gZ+@o?wF$YW5x8{m8Fijs50ecuxq4jme=?1)tf`Y79 zSmI626d=%!E?^v925}YQ($t{T*(bscyy4*=1JD`siQ0GTv4v zOZd1gw3&Nt?o!V`b;hAIoCMKJFMN){r}^%Fmo&RRyr#Oas0N=1d>(SvAD8fuxDX$s zDcP^uar)1&noZLm&5kySV;LB|PuJs4Pu6C~@h%UQLQULeiWZcv`kR@DsxC%X846K` zydzqlX4^Ig;a^mo>rRABg&&OHnByl^T=Qg&wrWLw-5l6k1Um8Sk z#s)vvzUdM0w_{N@d~5$mzG?HlXqS0i=vl&h-Ao+g+RpDrRFD}CU|~YA-^tC?=CUwA#=WCQ8=KFFto8%m(NGLtMg33R!r^H5g|=!Tzk3=HPny&f+OHG46dwXRn0gk%h^Kq-Gdm4KIEN>24u_s{P? zxcE9dC9>SP--DiuX(N7+V4H4Q+4KmL`Ker=zKUq)tNU>08xURtu8kGAN3 z(RYxmJ+LA|aV~lfM=!zYJuiAl&{Y?U*5W&DNp1Op$ra*AW_{*@jy2|IC0t6ai&9is zS$#L^Fz-Aon9=tNf|K=u=6mg z$NTqq?Sz)iLQguJQl6-l5XZV!FsGs=Xu2d;Z7!^z6eTK&cCL<9Z=86+ zn}#+|F7E;^E^Qt}JBMmH9{2|eSP+g_J`W+sr9DPDd=O*t5+Yebvd6%HzF^M8J-9Xc z^PYQT`^;w<+P~C8fcX26W&FqfDA%DBhOzaGLp$@w@45|cgL5<_S-+S3wx>SGDeYC-Iv={w%e$=Tg`4|RfY<(wN&j)UtEr?*rW8ClGv;$% zY!RCKh~3CrV-^jE$t^Gyg+28rcX5P!E+3oOBtb9iD`ZjQptQ8Fp^xSVK$w_8v@dq? zn+&WR(^FdB`HXSOF@r35TCRi-E6jo`E0ftyl*JR8I}O$4lHU(olf)@Mkb{q}SR0#j zQzT^IL$-``Q?LA_3&tPFVTH>he_`_B=fD0yme$hf34OBM*@kI5FZenD?e>W{j zZSM+2j8e17+h&sxrpvo~9Zr32iU)Fuud`e-)?a>2co0!UJEwC@%4@&yXpJm5Voo+5 zGgPi)nIN)1glSjfD_+&qs4`vAqjo0wzl{`Ps>$uku<~iS9EOCQ-CfQkl}x(HG7VBX z#G~lGvJorHp3}}^&NCmWG&On#2JuB9dtx@#$~k1oBo>W9n`xP)1xqI1n|8(D{L@xe zU;pI#AbO4!)3+U(X`&S46B6vr%dnGQZ-&!FW$@#9q~#)BnT_*3&_G}L^667b8z}?q zxW=ApwnD7i!eK`pf*jJMe9@BiW8@`8%YK}radug$UWEn!Jo4<<+4aeZ0K&PLGbBzm z`90D{nC3w!v68OAj4TroPWVv1P{Fvn*>i&@RCpL@sQmSVR>6)9yde z%GD|_nN*qnXeS($XbeFj{QCLb!3LI)$HFJ1)$W-XnwIb@Y!j__1H|Oi= ziRo&`O(7vBCeY9#wQou2ds*DeqqOP0vU$Qy5p&`NmP%NYHUTkgEl7s*04*2}1OAD8s6>k?qEXeK@pw#$mxxgdw4c0^7hqC@Z!Wd3t4PFr&gof(5w^2y8HN z{I|F_`}^E!YDKu+s{p#tgL@oBLdP=HN}=&ckB;t}le05-CL-`7*1|nKJ&QT>b922U^F2EJ zeSK-86d`i&2CddIz-w8Cf&EZ0PDv1%GQ5=_b5Dg6J~lSyC(S54q*d+;hbNBjP{s_F zYZsJF+Y4reVYe7@U*Fw3G>hkqGNpFqZWPB>ew z6ek>wNSr2)fsSs=MJ)3tn`$!KBbvB~eoH&H{&l!?L8zRKAa&gwEf&PQ%Lk0QhpG-O zi&*@dP_6XW@tcKD59iWt44iqk|#I)q}gTR{p(;O?zxF)X;jW|rGq5`OHIXb z62(0NHjsIpXpW{F(a+O1&giP3NzoxxlZErN$d$0e<#UPCd7_Isqse~Y z{(OoW2|gW797W8aS}9sa8fWxwMr;cF?Cj>wOVXS-11$8+3_1DX>CN2))N)pVNKKMq zeCPBZdfZ%Wxq~+X_mQKHQ!eQwex{%vCcr@wWyGxMMMbM*);OFi8XYI|*@ zJafVLbKVT}U=*(xY$T}aH@~XttQw3umSMs)+zKoR4qN_!>OdwU3yE;;RNGp_TSk@Q z7q#CVm%cN6hR!=(ocDbBdSK~$e1A0_=yCPSPufYCX7hw;l6&sQQ7Z-K)`{A zuSiZW^vkKANKUxRH1do{bal@S8Xtq$j4qBvy!v1Yj)bzj;tK!)>u@h$UvQX6qZEZ{ zlyK|c0*HFbl4Qg!k~5k;YEv~+95%bAmof}7d^%v*b8e@W5B6$dw47_Sa>=CK$2p{> zo=#3kZ((I6_@3>r0Hk_!gfR7d_3`zUC>X~KMYYlA&3K|&HHR$bESuh2aGbRhmSoW8 zrs(hKiFmX`R@^@`Gjn&>1t#2AtdK;t(jNbY%IM;ru%T1G@@cS8DZ}s0>n?8YjJPQX zA`zR;C6i#z1j_~;irHmH$dVfvME0-W-TCuVeRuCNtU8e_P+_xaGVU^ujMmWIrwxLk z!UA;y4|-m#wLSgwAQWb*zFfCe+r#6FeiZy^>-1`^FeIIyCQeazAYhFHeBhD|IFv2{ z0Uk}CONoUToHl~Uozp9DK1XdrnBGs@cSdm%LVlO(J=%RCT|nsgC`o1O)H{qILZ11U zM2^U%jShzU4G3Si=oh9yZ({VwGBIKGIf{@fC2h6>bV)$;>U*(%5u_u)L<{b9-QIe!>9|FHo4OnDKfolIEUF4Vn`u-?ZG zT20)7!OfBsw3(>%a(eU`7G#1rw3ms(R+t9xo&h~}5^Oo5!m$li!K{Hs4fd3E_&Xid z1OzEgI1%CA>D5Fvn<@Z(ikRL=E-bVS4zYOL$o@$?w-IEP;lbNLwZOspowpqio=q&A z68xM)g*i#f_h0^9HPTaFf4(%jk+tYp-`RjIMQnzhh7~W|US7(Hvq3+P+vG*?i?9zr z4$^SHn@3`7uVMM-mePl9`(csxaT@nXD^gjU(QLT{*H*`lvw{;pS+voYsde-Cn4cu7 z-g8hTw5U^hqDRA{;rxy`{8h58V3bZH)o+?&XRfSXp_v-^Rj_SmG8G9!@>i8N z;cvsgTRq1gdl)g1imm;j`ObN{KmEA#^E$_F-n*@5?5NHPe;M8R%fgor#ATKT1JYST z@{SL=urqhd&PW}yGEr@ZH%jefg}qG?s0RC8L_`=Iktxr9l#D=mQ5%fG{JoWzgM$p5CZ z*ZOg;sJC(6w@P^)|IbwcqOheuJhkmk6v_>~{B&dz9hB8Su1ZXvr=RcgQ%TI*B#yrs z^s(mi#bzxtY>W2`Pld6)v($;w)~GL+xd8~`uedW*YvbN8jF@Rcdi49 zd@0$PNLTK7 zBvC!Ji7KHrKm<=d7*0kN@`KMtu516#^!llWhjM?_OrgWYVa*O58@iyhO zW*-iI{Qi$MhcnX%ls8AT7L9pw0< zS|zd0SD4kZYq#u=*x1v%UA$mxum6v}!O>}_KQ~Jc7(B1U{M{MU3)hHSdii)?P2OyD zW&ize`~hpjJd>{l&nxHO^nMx4-F@QGbKCzX_5}vP@wesX(&$cFfs{Lls<~l2!a<=a z=bgRRn1$M)l_=hy@czB2R9jCa7asH>J$Pbx&Dv+m>Gp-9uO?kGxKKuA!xzoJT0dfC zN<1E^^Tl``kBD19>9OJ8$yhk9Q7J`=!`)wXH=YeN#6A0hoe{UDNh9(^>FW zys!_n-(bE$F4aO;+ilAW7KTltk@imfA!#Df?zC>J-$hCuZk2HyHXo(2n*Jbj+9GwH zM(c_5j+8PX=;EW#N6#ck6YBbj51(#Dlo>V6aJ8bgI`uxp6cpg!G5^H)wDBX4 z+bN_`xBC%`i5OKR4_q~Sk1XYz^;^WpM`2a8F-!wqc4@ic=pD*ltlNoaZOsxLmfYbx zR*v60U|vQZE0+%EUxXH9ITuM0&e8=5j%6Ri_mgZ?sbVxLD@ADH!1PH#?7dzV*!#lT zYw~B1fS55$knN znio!oM0B%nId)qfMo5y}H{srR%_F0`P9%ZAs+wv4{8>5OTyt75>)ZM|bI60l8e$xF zHv%QyMD1i>yO&dOUj;uLqiraSy~ySonPRm6-Hr-V6nZ1VJ+*5VpRytLTx&;ww=gr7 zsw`M6_UmC)>6+gClsi2%wD3`60DZehcfLox%j)vCWte46T%DgKI{$0_AF4SpSy91> zVu?rz1721pI*Zc2ou3a#lNbFr?nm)^!nhRus{O?zjVb?}of_$trxtiDqZ(yQUZdT` zIJPWhuMlOdMRz&VMe1cqjswczC?Y`W#X}9b3CT&m3dS>2}DF~lsLiJJWYBPOpdGn(xtBPee3SIQdf2x7=B!;6xpDSbJ>Ys!2v48Ji{dHe{ z%K;FUUXDgDd=xSB_$<#6>hGsr>EmGBaKT(7n>)om6iV#ONz%oNu{X!U6?0L?L-o= zUz?@{Ye8a&B|p`rpR|%LT5wOvm>=Pi7JC2r@e#%>Q_D`*;EAAR%bD0kJw@wlr}-ic z_E54X;LT)d0NyIg1a#dKO`t4`Ih9$GtQvMtufX>h9v+56x1hWkBbw#U=EEst9S^tgpdEkPz##K$m#df7CML+^Ir!CC3s0ug^gVd8DdE>}4 zLoV{}Rc)U7e_ZLpMdcSH4Drfgl_PP$^Th>JljbFn3{@Dhqt(}wU490NDpRfXRci19y3p>B5eEt5ut%GeKDnNaSiF41hmiXk$+83^ z0brpFSP4|;ojyGOUnAiHa^bRNzF|t~(X+C@MahM(>_TlH@Oav!uGCuclW~7<@~dC| z93qTpDb$vG_<`pRA`<@ZLoHc#TQc}fDOBf~YT|FYger>Kku;UGJyTJV`Qft;_#&F4 zz0upBcORD?bHpysn>2S_{TTHY&>UUwbTVDErlo0gAvXG2|Hssbk=LJ;a2bzK$yFn& zpt@Bt4QHRoBC^?PsUNm?1)I<5uTyWBJxsy>so-@Cq3LSIt(r{u(j=F)SDKfd-!qE+ zhQF&2DgHGq**|d8N*4B|sj&io;M-_*+0aScvumpIL;Gswd1i7Q8$MygApJcKlJywnUzmD zdmK16<!O)y0}*_4j7&ScZjz z#x0r@47$FHJVvxFBu~fST#idWrzqXbFd1HqXuL3pE-f+n@%yWD2;QS;d7r2>u}Mb7 zpT}YMHk)SdX5};l!LlgK)yZrP1v$~*=SRQe&nID+%?Uhu{%tVDRa8>S@k*=wo5{2z-9Mo4a1+@`G*Qs|jiW=NyXt04o8y+FFdVWlp9Y!mY>v9H_DH*-5@&J(zPZeY;V);^>*nHGToy-u_v?zrKj_szO=R1j&R-QjsyD!j(3ec8F&{sFa24}5 zvDw@uqJCUxOiV9GWpm(BsYJZ>ulL)ZmYIH~)VJmEu-@YBw4(TupH}s|7Ah?$;#Y$DIFrT^G#Ov)ziI*2d#Stay7y-)1!oD z^~naSsvg^s(;X1j!JnX>}z2PQ4IAq&IEn=ce zmUGn zB}wBuIa=j4hPb6t*e7<%Y&isx3i%Bc*`Lye_%M`9E5$;;Hz9i46%+ljFFviqUo?o` zHY6i>m8|~755N&R@Sr6o9kn?Hmft#@>TLbp_+&aCMGSDZzefu(i+Wh`Q~6yU7cCwX zL(DpsKX-Ok&mk-HaU^4@f+{OTMMeMK?#S$%dW+;dF;ky&2Q7w|W@d59tYk=L>FDKR zB}P09ncL&g@f~&c6!W^4>Al-O(_nhbw@f7;dt3?0C=$OZ?kn6W~}$l5-=a7bA$FoREA7X;pWi{}AL+HCGk z^AF%o>&yp#DMVfm4k~r80{eY>@8jaZ%F@ywA1Sa0vou0_nacR{w1BtfB?%Z;OG`_E zOdvtHe97bolbU$MCUDMIZtw4Ia&l-53~1xNY;TuxMgs?b_2_xk3=)Y{WRZK(mb{WR zp-MIgKKqp)L(E{Od0jE*(P}pskihjHGK`+f9{mX%bSyU-Jd7EKv(ugNBo%968W06o z@MBS?&WH@rj%gjN-{GXA0u@OAqt4nXPiqJKwG%H>vcZCYYt{5^eXj86DE8Zd23sLa zF>WT=^lexR?pI)rcPt;u{Jk1?6UYDalhMVlGtk{Vyy?sJgXM4E#a|J*Kl(O-@Xi{kl^Bay}b81K2YvY$i3cGWRD5z`2W4CXP{*J8fS#@fcs3 z-{D4Q4PH6KBVl5cayB-Ud`SITB@Cu4UNYv${Yf)J7f70g~gz#l^oj-G3E$;|Uq1w_MtEIIVwoJ&FeIpd9VroxuC2 z$?qLS-ZcAJZDNG>o}HaJih#%@w|7S4P*`GV0zWiNsKymN!K-Ewl}w2`0Rfs7bI2z& zasB-Y1>>Src#67o1d-r%>T`7k++|x6aUNxtMHNG-@C37f{>joRXGq|niPLM+NnYvl zrHoT(Q~+ zoQUK|0el-|fsP_MYj^Fn35ZRTn)6Z7RZvs9xIqt=t(yzAiIpsFS^yU8a3kI$T6-nn z_DA*u?X-zV7_||iiQCWz>GP!BK(tNR_B^|tS0-_ii;vu#JDJw?qMEM!o@h_n{~Ge^@5v^@q{bs4 zAi%=|BQK(Pui&ub^m63@o3-3Hv@xar8 zn(@;%)7k~czMPTMTdM{OJ|>_w0TLM#g@uNOo_KxxvQH$f4I%?Kuw`~m#kh$;6v&VO zP<6pEU4^YDVZk`ZW1@0VEXHu`SaM%;S~xKl)&i2J;>izX_PrK~HZN)n6^bT?hq1>` z!H5*()~YD82Ne)trHkiT3tr7Ue`#5@tEg0>OU4qE3bbU=H9TUt6r2)=W>emOtJS!} z7kCvjzVkQ@xL^V#Q5xmXRFhefun4@_Q})h$0Nms_eFn|A!c95^h=XEcVw6jW2_k9Z zlzB5ql}?V2b#zm6a&nBg4S6#F;kMupGX9M1E%#k9(2AU#rlM%F z#gib61pv=aOY&nhRXX|u1I0mkK|r)33-BqX%e8lQcJ2Zg zAJTwe7X+h>_s5v2o+zGAb+LZ&QsHfApsj-t2817y3>Q~^oWCitAcgQ2lN$57$Y1_z z;dcSu(|ZuKU?6>uEj7z%%@grCu7$u_?drCle zfd)IUi@MA)CmViKVNDVd1uKDA)|B_fYVP1Ryl}PyC5XCUQ~mAJ6)w-5rUt_Mn;TzUvVXl2u`CT6K>%?e-gGzPc(lH| zt3oz7X{XPX24YSypgTH7wq3v+s(0~~%-7)rfF=+t%LE#OUiDxIvypg>3bXCoG@G1V z`4PPa;ZNP%V=|n8?!N#>igY+LMouEgGZa@qo&|Q~s0}|=SQyucOZhbTiA6deOnDVX zXN2E^KEUGJOni`IN;1s31MCOBlEN1NEDc709f17=ss`v_cMuQ~mQULQkZeG$z|mC; zFD`LDw;Tm10M$7j0$FguIESiU)l9Yq`#Svc`Wj@jAf5KRzx5WT$tx&$US(ilpkAuS zL===?Z^Ug;B$qS>8UV87ylHBnx5Mcf82GYk=JU;A2Pk)x4E)R0+FDzo($c&XF^NM0 z9fnog7dLXX>-2!*1c$N2c*#G^TrWJfb$fpA0ikyp+dlbjSPrzil2Ve2Wj8I!*ecA+m= z0neC2MiqPc`O(EG9~~ZIgvyobfs_w`%o@m}NV=9*RzMaHVzdN=r64ex00A~(Rkh@= zTV7sv?=UkkU?%qv&jZt%Q-D+ReuS@kw>`kehb935svYq#uC$N<^fvNp%643bpMYWx zTf4yFYo}K#RBSqAr3;SH#a?c1-;XYwU@%#VgWHD&FLl9Hmw!#MM~yS#)JjDx1fn2vN$L_-1*3V2lcBAp6-G%@g!_qmU?z!?#y*_BUz z4|DN(*iCD>5vIU>4qEr;UEV2Z4DPuZ408&rE$7tj5HJqs419pYqH5~AO{C81P>!;l>N4u z-%+-+_48+p<~x8UPqH)uPIFd!+(43ROi#wZj#UUcM`VM5v3&%d-G0u{iq=nj&FO#R zYLqoeB{w&>XMzi>?_xDjI}2^$;Qn|W{f~)(fn^V0np)Gs{?DI3cXzQqLE<-mo?h*Y z7p(j5p7~HxQUcVZLxBCVUJuA9DE)wW3knSm&7j$}eioKm&8gOGqEwV-(N(Bl> za7Yh3nACtu|2Q~j{CoLSj)*;F7`#@B^ulFW;N`JgIxs+E13)neuoB9$IV3(tsB(z_ zLxO=pS9`n9#Ua>vN8p3|Nh4B$hN^mA2l^j4rCmN}I9o7U14neD(Uo_jCd*<-k1qi% z(Hr1_z(~71dBvU*Oo<~Bro)*!YEv@Fxe~cpy$nNGH2|D!)T^6gzR+#TCjJfj3=p{> zOHv!y=(`*MkU+FV=+IHs&E5Q4iJZ~3_h4-6TDfQp~wFN|bjS@D~)>hW_M;I)T{$@H!L_OClF-ZSA66izjNOB*Kz)a}o>* zGJcm#Sn`q#i2ZpplpO%RfcS^7S#u8gzo&yocp*=`AaXrSmIbM7*bc(qg4k4LgV(&x zi69zTTj)f2M%-ZP14L+91Uw;2TmX&9An_qV1IcRgOF@&>cK}Zk+DCNdKLp3N@6lDuW0o*fZe% z_G)9JSf2}UL;dgW^hr+uSXnSLF@AszIh#r{VCEkyccQ!p@7dee*EcXQ(9cg}I+bnj z*Dtl{kI(DO!CJ|I-d-O#CEOIUdm|(2Dpa6SIQu;10r{c7k}giTJd}c= zg1o%#Q*R~R`cUw2iwEo^QGm<nTdKOu;y4j{R3% z0^^NrP^%oB3$^OVg1@!3b!g`lj5wXjHChS?>RH>bGjYWoPz2~35jH+%HclTO8?%@W zg=3V@f$&C53>6Jd2ma4HUq(WL6v9kL@Y2+@rx^rPrFy}%pi&~oiKD+D+5B!Y~rc||u@sn{sPradZ(vp(-9|7%DSn*Xz0E>a2mG$ zE%WzO2G|dZSEjs}WE-L}( z1BMpUV}3ZXGuw1qnSHYf{%z>a6<`+eZMVUka>nVPXF;a~PBs0*Qe-DJ#9-k8uZ`&LO8^j@1yKw0Ya zN-||wqkI|&N58X~=et<*f7kmlvF5BNz}~co%+gRMg1oB*MG|6``by7^Ww43(#WFc0O0rKxpxE2nN z?|<#El;JRPz|4jLRJxfV?7Ga4dV6~j35-Sl(Z!iy7sbZ|aSME?)gwfu9wMB(mEit`m1oD?x|y80dZ zSzXY1!ISe0E+SFEx+zC&qUNQ&ZZod~npHOMH{kDsEGfg_BTm=_AzBN?7!rsZabMck z&EaB%)(>om>Fez}0OEZV`1fY+WuO@+nO8pei54cnkTC?d#u*5lOnE*1{mp6&z4I{{ z`v!`xfZRD>JMrR5!*OC$?nw&!hB?9fDP8xCMG5Z;=2&0Z#%9-;{Tin+-8Fo#f`-7IK1SgXN@g+=qK-i zW73N{CFQ3+Q9!gT`<1)0{2o|QU?ZX8f{y2wE~_$a%pjzsjeC9VBV7Pg zV?Z}J_wccWjG`%BVfJrPJ^Js4CC3#Whjc@38jZMt-GWBM6zZys%BsVws9UC2p}~&M zb$WiVn36(ToOvwK*ZEv{{34pNd-{LK4T?k*<4~i0#Y>F50e1Xv;(hGYLVcm;2KrU5M819qF?sHp#zVC z3N=TD#v8<$eR9=+@NNH`zedS)_s0Vsan~T~mr^#Ci^^&~NYR z0>)uztHPIm-P$Nci|mxw)LY55@Z9`-u+5^sRO)m6J$v~#+{?^t&bh~O24WQ?GrHpn zIeSQg?Z`reJ;o-_MOv^H%gJo|I{3B9ockIr!a0Nqh%Bvg;35(c!96@8yod^g9c8?z zLwilut>tRNF$#vQfn(X)y5@bq=+s>*`}!74Yy}m~QoRL74}Pki5tJIOCyirP-^i3e z%e=Q2b%V;12oU6d!=Qh?o=p`@%J`46OrSjXrrCrg>GEfe6MRZjV(j$xU(xPlYikSG zHc&dir2!n&SoMNIRpd{Afleg^(6d56#Pzu+C0Aqc;YTviMCo|PebMt9T{1Y_LAn62 zrYk=JTy&O@!h&Q1^(ASND3xU4EDeiQjxofPpap{+i+n673o0?5|9;xN8rcQfwFLzQ zwt`u}*SNa8%#5c<+&@1*2OWcg@lZhuoChCO1#Os!ppufew^yz$i&Em^A=a41p+Qn| zM8cZrPlGe4JdL^1zJl6^+taW456~abOXly8D+Tj^puNz0BtlXK3fJkQHbAVhr^uz= z!l950phn+eusc%)lTSRcObM32nR=Y9Sw0Q65>HR$^SU{pH~;G`^5;w}!e4yX=rdv9<3BpDQL(Hgt_Zz$3&9pvWD z*5M!QX-rwW%n8-RhB)b3_P1s1t_EJO*M?@0DLcitC5FuxtCfNZH@VXAw#z?5ISpB#%DSgVCx?}|dpaGOx$fdc~MZ6ONP zxT6Rt!TtrVic8y4%WA+Pz@E$XNKi2TcUwVHUj6HK<9|Ux_usoCW?DJML3Vkj{z(O$ z1nv>&cOXbEFR6&7p#Bqz&eXN5N1Ls3FSlYZ&q1_-{a091fenZpe@t5l8!h}$h0O(= z|Be{WXb>r&Bb))t(V=)&kvazHXsD$1SYop?Y5RgvpWXF#Tq)xd3eSu5ZJ-@gd1&JL z$}oj+YvCj*py~{$aVt3>B*c!`ifqq;It7}}*4B0+w{fAmm=nBxz(ZC8|4M22xp}P( zhdv#BV>)yDH)i9Mcm>1;Hcj3@md6aXFCT)1$ES(lRxG}`y9dJE=$Esu4_NwUOnK0Q z0YrVCM{fDY!6%0$J@1kYmNE>+ER$$Ecfv01yfs=d3fk7#__*b66N|DW1L#xpTUutnW4OM(ItAgxCHOG1 zkw2kY>D$BBA@h=o&31&R5{Vip3Zfee(D+lIE}#WchGnn#*Ev3wEgozfZ<5Au0)|@~ zzs8kWiv&BOk-OXb6yqY8312o|E{M0f;$2KH4>8UB=;$b5L4S`UWq`|I!4InRjSJi& zAv;%qb16JFVGU6URm9TkUVZ8{WpG1kn1~{t!?Im&1LvX4*+B`bqN=8p5MR5B@;m|PG zs!6*`zt3O{d)kQS0YeTrZVg7VB!lg}SfDmvL=hCsESuB`l^q1~5Pbc;2>dJh=jY}o z$R_`S&Q0hYFu|7|qC{~dt9^fzG7PHa1EgOY@vfktU!}}?34N&g%cBFB=m3!K@9z%} zfg#1*R}8Lzu%`gp+6k1dJi2Y>>BK+tE?Y1|wka1Nh2B;YNG&cX9PzBufL_=iwe1__$mAZf?jpxCAqW z1aGh25oqS#vv=F=fL(NV162duTc?#yz{i1C!KRL?UgWHn48im()f0GFESTrByb-kH z_>0OBW!xh7dz?D|#9L=tJWQ1G_)REe3{a^LZiR~c2K`mws*S}3KxCj3=nz2GpLqGX zxrMbFW!A6Bw6?V|Up#s*q?&9Y9Hf-<3brxm(7Ea*(DKMMJrWpHZf=|FT zhzMWVX<<4gJ?Awj=*lHV5pkiZQ_?n}C%KB}^^FG`4S=NewujyoGB`00E<-KIf;$d$ zn}ytoyqa$q@Gyk3@G?=Yp7r_w;f=Bql~yt1fHkR{k7XJ?B@8Yet?@tSB47HVFhoZa zoWcaU-u%GN+bsmN*lZID+^hWj{63Ea(l@W2q7*kSFI(r3Gf4M=2SHG(ciZCT` zX{R46gm&0l;jNEkJ{%g$Pi-~_2s_aMJ-8Wk;sw%7<4hZSg2bQ4ZH=kLRHeyK8jDy^}E$Cs#(dTDm+rWGUw;g!tOikoJirZSVy;Eda z4Y=XC#0=wMK>%|W+*JbhlX=~y6^gJV(~G*fl7R=Pc0H$ zP+F0oGqQWW2FiASkB^DIpyTAIl~0Gu14a+H85|0x6ObLNz&Kh5`op(E-v^L?@jMKJ z;Hc7N*d`4+48G}uGFuM}N}sONQ`bx!;Om1PnwvP&fHL~e|L+2n@8?PGKLR-yMk#{eqsIkpn=eVF@aYHVL~7QgAz&q~Ero1m zRO7VQ?zptcGJ&f=KnI;w?UdQ#+l>rT~>}U4c-tcl-o<`=}g&Z-JdA)wDUE;$t1*H%AT-LQ)P=a7@hJ?sm z+2c1MkBL+VmHRt}^Kl_d$-@NNoMI?9Wgu&!hyhS1F8<98Bwt_<=9Sizmn4naICnF2 z0^>_A9iSoBceFbtCcKM@pFrc>Z_3<*m?V<~cTKZ(SEXjh`yL?b)|R3WiJ~(dxK1QT z>dv}?$rEJs>f;1ch5uT8nSEk(j|!0_c6d}ZJiCk>Cpn%FHR~%?P{`2^#*ZWUzFCS?`a99T8x0^jWF7k3rZ{Xv&+SgenQn>QVRKd|Q&6 zhA9J(277|80jM{8BecYXZzM+1dbT*1r#PBH{SgR6z2%1D+A#|g^+zmfD)_53q5Z5@ zfxdqfR2;;I{tT$!bOzUErDS9NW+x( znNBD#^cgE%idlqEtuAQ*t7Gu15E1OjJuF?mOo=H=ru-C)h#trXABHk^6bUU4XK?Kc zXkLuaH3)%IBW)4w6SrOLy+$`QcUJzimG^^T@AqGk&`;J{zV+7-CzmWTGInsNh@mu* zCVL(B2C2ON!Cg5!DI&>S_8W&s*d~6Bo?Sm?< zggdJ<7l+k1OB4?o#;GBw46ot_juxX-L~} z|NDxEB(fLZBndYu>jlZi91tr|b`-rpyAsh+tiOYq#%hXn-kI~BFirbO#qa6G4-q;z zB0YIRYSB4XQj)QC8PW&^L+Sq6XPja5;072G&zO-$`p3y^SB&cu$OA9>O-Ks2T#y&AU|INVdqKVqsGc5mFn zW+B#zV)5FlL)3dyZAxhDFh(|KNA+tnNxs+r&~(;ORi<4Vr@KW!N>YxBgi=z{0+LFH zG*VL1pwf+mbP5Opf^>H`NP~2Pgp@S+ZRT6={4r~mvt~SUp69;zzV>zf_TsjK#5&9F zN5|8))queUx}^HcuThEedV>uPYItc?*hmD<1|J>{zv_$i(eYe2jHaTx zpTnt2hidy*5AP18vssVz%CvJ0S=G1OyV{d;GpjQKcaCBk_Hw@JGt-x}5D*f`dpxS1 zztg~RY+52%^emG5&a+|-6kFQ|eC%01&F@z$)e2jC4DzYZFq$$xbw};mClkC)r0z|= zD!;uoisaO5rhB&2FCRX%7Ta2Mi1njZ$NHYw+aGmW#=mEORbeAft~;%_3%#uVXKeDy zsmyRQbPv;=tt#NnJEM6jZiP<|c@)C<{R$&dMkc|NtgM-Dq;2#lt!}Z-!P-I{6O-1m zLUU`6$A>+%z5s33*8O0s>hAS>5pG#kdTwUzdK1Dy6#iGGZ`CU`Pm1o)QTiEqr#oTg zDeDFr#>jj9{Fe#+F*0nkm_X`H zIi{82T-SK}P_|C-LtR(IWaJix65{Xh)@M_ zi!uI(xep0*la!x^%p&P9t%g3tsk5`Oai=+d@mN7?Gq53KVKjxSq)d~VL{gcJjVjfMDm@I;P)KcOcOANtONv}pq`#!bWk;PvwdneD?)zE2?m9`| zs>wirs{{-suAoL{`%o%xfb@BCVO zL0k?6dG^xrso-3oDT91qlfn+i{Yb&jA)d@M(aM3VWM5R&K6;^gvVP&Nc^4*&6`hU~ z`Zjl}buqLEHQQ0dRC%OL{#B;r9>x7;4O&ICu&dZNKgt9Yr)r+p;QCZp&rE%Lpl@Z3 z7AEti7}@Iez$nfxCVlS#xL3Z9K`KZyMwzPuitJ66V9z3>1y|avi3kN!?gXVuFtk7- z9u&zMCT&}1n1?vk>Kea)6xR!F|KkKOsfBkMkrs;Hwo)dAvHsfO)5jSZBoy}i5DR;6rm`J3VL6(H*cI~XNu&~q=jU3TBD1_j);An^0D`eqmm|3VpYN@8Bczhhq5iitc zG81B&Iugt5H8o1wH8)OU58YHcOnQ6XLu2| zSQdZJ!Ty=$7OsveF(=^2V;h*s*$J_rgnaU`B;l31ou~Y5U za`>|YLH6K`!Szgww~5M4hk$2^k=51FfjBb=?<)2WNO7P>?0T2Sd7H3^0CDMWRgv?? z#l;0nl%#vLi%WDpjCwW+U4)(a?#$@icc=l>{Xy;N6Mhsi1EZYt4H92ic7q+i73+g$`dI6H$h2UG~~k(6r{9f2RHNOqkk z&;(zE8~v{pQkle7iS~D;I&rp^#7z0gsuE|CiAg=jY`(kA9qRc`C{C3WK~k^GMFjHt zwGTu}p4s*Nwj=aO<-4eU^~%XDm5R);L`DmlO!8%_3nMm}B;~R;&jfStqe))uF7Bq_ zih<=7yhge)>#+p1Rn|ShL9%lL_3j@gBM+24lTvuPkIT#Ke$Hh{*?%r?E%P~zS?Umy zDpm~Otnz&`G6y*_?6wb$?7<8=fv+=l?AhV~V*>7KW+K?yV;3nvJ$f zTt!1f-i!-+Ijb2Le2tJA7I3Tr^TY6t9A*h?xD?|kws^CoACuEjhdTFe;I?k*oGS3(>PRQ8)ynzuKxQfSGWpSR`Ty z7z5tr*|X=H!Ac0sCpOTz(LjH+ zY%M%^-bw?VYve{oK7#BTP-8 z+>ao_f=3QoIh7#^X1PzxXrg;Af}k))F>M}uq8w)+j&FP(Sd8lq{M_yL(qCeza7L#u z;`z`zwCQsMz8p<%W}y>mP7G*tO$A-L-}Ez2%(w)RFFd7kMbkU*_8S%l4$~%P12w}# zn9x|Pz6}&pd;7;p{lEhGapT&q`SA~ifr5jeXo4jEWsa^z5kd-xU>FZFSAjNvUr6xH zQ;e?$Fp#yb1ayM(5NJfYy1MBtr!V7LV~b2I!`sp-iV@a_I~@yFZ{IPd+X!l{cW`HJ zJgN6DEqZz6lp1*$5Z-n$#b~r6R*5!KNFB7Pp3MHhQ{MvT<75HP8O?@x990b)3ZG5o zp@(x<>`Z7Ue!i*A?k(18LObL%B~>`gu8u1kW0*FiIh!W6e)e1&EY{k*4`qBGF_vca>;CZ9Cqh!33n(_Z|Cu+;20IY zutdYH9OdQY<_t4N2gdEsV@CN8rI3ARTD`fQEVqi8q~8e-Z12%Y39yni(<9ZZ~r(=ux;VIsB}wSEQ$RQtJEYjx#lu;+h?0>AC117r_a z$$X4}mr*wT7N&{aH~&R2iLzW#y+t|9YQP-vw1!LmmZWmjR|v<%lY--#_UR{a55|tq zgX_vPb`UB)JX83_7@T2EYQM~9Eu2=V_2|uP$KdJ@Eyf~%;Z9L`iRm4@ zgRR0wsUMX5R*}+R9fkZU777KG!qCg#{x2NCwR4WJN&*Grr6l|_h7Y)O>JGarJ`jc@ zhtqQxa;YZpkA)di3l1(155HTb$b47!BXxaGnDh1}GH+dLy-{lQiIk!oUSECAn}bUs za>1y%OP8Xj?$c`$qQ-FFftB${cLW$B=tG_;xIpa^lyYdtKPOvaQdI?u;H-7r`k6G- z@Sl_P%5*F@n7y4`z+;%y&)|TfyH)c8A)Bkxvd$5lnvw#>R-6eKy@0KWnn2tdXgk3f z01CmfY&Rs1yVXPtL*VI3n}Ak&>VS=|KPM0|0m4jyTH%i$P!WT@7ff&PzX$KdkHSL0 z>bQfE4NHubY&l*DU5z38*I`kicmGz6FeNx}&h$ByRBt;j!MDjoLQh5PT5ZQ(@}Gqm zj1Q0lfdPcS_XWK7aF4t2R?chI65cJ}XR!{xEm~pKf0UkLyK|7<`+D5lQ2->;BHN#i z%$UCir#h$D-j_5>$knyA8n>5LR2*GrY^r}+)ULDB9Itkss|q&SiEd|J4_k9heW~`t zkL01c zi_9gJy4(sz)aD3>7`&OKQGFeJh}#cFg)pjVCr0fJnhR zd+%S(AU*w$Gx-S$ZkHKpumQdlk7mM<507n ztIJMF8Xpn(-m}luCjV6z~q3A-lVsn%WdI9RY^eXUJ3g>5ZJ=x(Hr zBa6LQv#C{fAG_;ie7~u*yrEVk3rC}1v3ay}A4y1G@{1?*1zgVg9*3Mw^+A1ikdfFh z>sqsCj>KM-W)h5(Pkue9iCpM#T@jec-yv=?(9X3Z{qk|nSjGD1pmj5^2J=Id{L67) zqy@R?g*2ls#u&2VR{n71=cFc>bcj;N%wzkn3=Y@LE{Bh$6+q zFwq2Q9GrtbZa6UJfH(m(Lqxuv6u*0sQ`b{MJ{<@WMimdv7YC3NXycszOBKm-sxn#4 zY52m-{65-!#aMjMG#SPcadB}`OmJClI^>!EIy-YsSMntIfxwY1dc}-o^}tuSOr)9& z%z5^6pDp?h1+#3wGRTB~?GQ>>9hq|Jy|ax!GSfs-(ws!)r`8=6}DU@mJV5dpXM*w=HS~KGJ|ow+`Y^^3l0j_xL$yLB=t{O`A6{! z4q7JiP~f@Mr?SGYf*lJgeS(0yL6t&z8={E~DSqQz_s1Cw-jA#*%{EqZR`DACOp36X zG_0_yFnf6Mrcu|N*5R4LiImD5QiYihO_dIQ?Hnh&eNL6Rg*l_ss=zi`v$ascI~MWJ zx(BMW!W4t`AqK0aJ3sfzKN}ggyp6l0$`-3|#;hIa>F8FH(FQ2m?CAmp;@^<2aQ<>(VX}fn`vz<=m^xMkmC?;(0FI#Q}YT#ZNa-3a1ET7&1*oDln z13%b(Eh~n`;a{Ke%mGRju_||Vxan|y>7dKk|n{+Scq@v;+8u?8t^XQc- z>kG1aOl%S+uRlv?5q4-1bUo4!%;U*#vy$-fD4u@?3pF%TA8@glNTvX_c<(cW-=?fADjxyYO7TR%`CXDzA1TI*Pnx2fSVbo; zF*{Of(e57PNOm!D(YR?|=L5SorPAn&VdgyCVs68~3_ExS8ed%0?p^rJKDwB5TXkLS z(4-lRG+FB4Jyq1 zx)r*fnGVrt(s>~**s5fNe2gcYT7ISxXV1(*aa*_)@4P+qvqlwk7X79W1A*Y8Q|+dM z8%*TCjlewdHeo<3NFDqs!!@roE%HfKL;%yVLMA!1yzeWPZp*jrz^}^|2Z+%RPxr|j z5`<^sqKot8Q(X$mJnsmzGnP9WKj=NL$91pOoe&h@#EUOuG)9vuvres-@C?;p0b5FKe_scJ)j#-BBc~72$GQEDYsTRqZF8p~ONKPC4rg zoj$@Z_HeVG!kNBHOF>`E%khAf*tF+)S+JX(=`p82R?W^^<|g~&EHUH8m2zvXSIVTi zY~(Er;C(TEEXu_6znW5oIEKC^v+!+`FXz~R0h$703$)#-V(tC zIUti}AP9e2ISMo(cibroVl$O-!nbts6j^L=HlU}&vPyAXzeg9$ur!7)3gLe3 zizTTqgQ3%Ehkpwcar8p#p_Oaar;`D9{4KQR9Vf7B0nnp&HMo5VB#p}tjng{-T~lg@ z(H{5;|GI&<7Hq@`kjewDr&BF(9eI+VCxU_v(4@B~nS!M@INKRO6jq2+L5mBrdMHc|uLy9x{?Hh1O(> zWYTF6@6`~`KI4j?VTb{1Qs!R<4l2qRS7*HCvZtZbe+PICOXZOsk(;M+Eap?=*$h|t zb?@(5%*YCs*~FKRkdrDM`b~gg4V5vxm^S>)Vu5DI{Si}R#zxO-=bA0aFpAWbll{3z zO?px_oml~kwq2i*r13KlJtZ@n4n^Yt2U*_Ne`2c*fdz;oLH}~$A#7&uz3T*d0T|q$ zQ>#FhFdu<4o8yi}-qN+8FTxkoTkB&-DR&F)I6SLb9yLB8;STE(&!4JP!Ea^OJxek^ zoMBU+XzSWT)d(gpJuPZ~{qAAuM_tPt;M%?ZBU{vX;CyF~1^uw$Ne8JUy_>}xiZ>NL zg>gWB;z@%J8K04-_EAF-qy`9_3Gqx#gTn6m=K&4S^Q3Bacls^YpVTUIT<7K=H%Ejq z6z%>93yi9FP$X<|7< z^?l>iVSd<6RbHL%r+s;Z-t4BrL!*2$JStn|+BEN!=ymO;WCpSaWVRl zZT?l0Jz7@yZ{2B7z5_~90cYsYp182!6;>Y&{nVk^i9@MY?c=ZShciET zd>$2K468L|@t}TUO3_jBsEA!pd*jnFZ^$$Dw6trs)|J5-0YcodkeD3H^xk33hT6sT zUCe(@DU_5GRe^PW^?POEM$|6<^8%3HJymMZCNwh8>vJfy%W7Sc+GA#9Is1rDi`#0X zZv^63j4XCdXUk0B-^Mh|ZQSRT*Fn7A)AWAhgcPY|N_xWWZgUPU&gRHPO2HXvd(BIf zL9Q{oe5up*7eq=|y$`s0Nx3fwr@h?E8s@GR=3g|L3XQF0%=S953)uB_bl0GJxavi} zZ(5Bv@!!NmIr*?gE}NN_Q}_|RsMIWN>Fd>ADoVF&kSbc9JZEL#g_F2+lDMoS>cQe*4%D;W@O!5q~oME z_`&$rAGy^+#H`uwIK~(z1B^aPnl$Yag8#4V` zFC5lbgyaj(I3dr7N|Pt$K}dvT8*`5u zarGP|c>Qex>hFl$;Ffx1u{+i+MMJc$beB?M|_PoIDZQOy1X!rqQ*>vc9Z#Dx52GFp>uX0zp8^?7!b4Bi}uW zRp}u|9@c3DHow1aJC)(a#JvNx+2$h8zG{@IWh?K1k?B8+xg_9{2P+bch>4<)E!3_X0M#!1X_;YT^@LA=!;W+4-wvKY)hmB~< zvG_4fC?(aM`Ss~}Lp~FX?6`hn4C)vWZ5r8W|DZEE4+l+^tCie?`S`S{UOUUnj|DF zY+X4WBmvlMSzaGn&{lI*02BRRc2+jZ0h~$+a=@xx&Nwsog}!@Z1M)TdyeG&DxmhpG z`KTeu(V@{e-`LOqI$Rf_$i?KJ6G$BV_RPrxCN7A?-16WTHp5ffy_Lny7v<_0qhKxq zVsjD}nk$dQvL#eiA^T>c*|BfKla!Z-lA^;l0lMPZ6W8D$k-)=3Ju|0AnISzbe;o1ni3Y`kSbXu35Hlav8;{=M zAuv#bB;pm&`p)WEJE}`w31rsbdNBT?s*q3}gCMuLmT(%R8i1Q3n^j*gbf_wSF5*nf zJ&Ta23+5m(HV<{5)AMSUS10R$1-!O zeH3iSM3NpH_=|z@!QSMmy>j}y)WA~7KrBi$d&P%7+3-Jugady{EK{yl!GF0Gc>47@ zT#hz~l_T8hUq6VcsHoWZySxm|L}_txvF(Mirts6J-KM3caw2LK07eGmB9RFxF_2sJ zZ77%!`jykOzkZ38oI!);>UtjwC2Gx+nM2_;!lYW|!Ore3j<@H1D=RCQn)q3kyliDO zDOURqsVmtmNMix?n~7xBy7F})lq(vn*^fBO6Ic}P!uMRRb#<8bZ~No|UM;XmnwG|o zKAy|obmA!qJ4}`ty=B9;75Q&sFM-qlSId787W-O3)MX%stHF*4Uf(MZVMtzrwxwZbZ;x_6Z^-NtU=|RCfQJNXrUbcRO^j>B zZzEvcWhH};c4I}|cflI~_H`3h^8|R}Ab8!-a6OWCO^V|OuS>4VugOWI1AVJY5-_m; z=VogJGZOBv{=&n@3%U3g8c|IrYra{y1-X~CxKu3npWmEpZf_NV^p2FIyw6i~1cMBTkzDL*vwjl=XxA`c=v;;%qzhz@ps>guxi9+#Mag6a(Fa5aL6G` z3lHP)G1<%Si?)#YFfC1(*y>-k2ccXl9)3(g)1g(T+4FzC=K&Ap{ySefEG0%r*f;q= zUVM6I_lAUr<|C@ielE!)m)9wjAg7|rO+_1oV)*es3#-sQER<6OEs*FS#LGbeCT9Q; zI5~*0TOvp)!!QmU#LYz5Ua6`&J2^#le>ErpP{jE;S6vU#0wg3@9*osgSr?T4JCLw@f@f|GTB|?G_fAVR3x<-?2WeJ;04XiXBKgv8zL7 z#{1`1JgwKk8Pe1=`GL~LK(P*mpe38%A9z$Jwo(Sym-is=|YFqlUs>fn=1gSurMni?HIsC4-~|WN85$?b>+=q#3OOhN=87 zR2(gfdsaNCnJ_m4b3c%xTLTCoHwbChvr8q@m#T0~{6Al!a@tQ}+@gGGpRW(~KcgGG zS&*G7ocKQW8$Nbnf!d2$6Kb*#Z;Mk~7bBkg`)PYF^@V;8Z&upJ$8jI3T@WF99w-Aa zbW(M@+F3eH=@=Os`mI|=c>ZMBxF;HHAM}AO#PlB(PjLpLKHh`P zb1r{JY+LX}GVz^&TTzkMo&q78DsIKNeLeYbH7 zOSdZN>dtiKmsvW46opP0V%){3fy@y6RZ(573CF|Eap{{uxbw!x$KkZSI{ynW8iq(v z+91?GhYWuMD`hmWuOJsIcaJfcfeiWWCn?5&)^vm{j2rAd&BZRb=;)DN7L9B|Bl**( zn3WWB#gwC*nCM*;eL1I>}?}%Vvl8-k92JA=7 z#}PzloBt%m^zov@0yMZ10!-;LH_#l2yCo~Ufe8`q)Q%IxjEXHoIm0{Oxwt?W55fmP z+fIG-!L(CHiY7;s-RNPf_;^i);qqZM5l0q zh5$2|_Fw16%jZla&5tvVZs*ew&FWXa(cxB6Ei5Z9FCW`+0$<4P?vD%vxKz&p2L*Vm zyFzX&`!2?sP{IUf=;-P`r6vFqdXK-~t(J#?ici=)*V~*8w~WSb@Da zDSstHvlUp6k~a7-7$vDH+uRl%*a=?#kLxFid}mG;UJT+6w9184Qj)&X{WG)x7VF(1~-g6KdV9W{uo)xo1VBy1%1~?DGwK_k*36r5F z_gqIN3i4gB>qUa$r$nUNqZEaQ!3PlYdv9dK_@5m`l!!lNU}$Iv;uJ@)9s@87lMhyh z$k6B$Uix8cr#k#SPLG*b!WJyR4~7dh6rsR0Ed0W_AZ?A#4-q!3n}n)03v(9m|~6^k=W&Dh=-hR|F{z4J~b6 zo*PuBtYmkeeZUjLd(9TSG&MB^ewEH;z^Y~{Bg)|uz&I{-6yhOYM>bsqcj32N!oeaO z!;HE$D$%6@|MKOi8JD**O*p_CDVQ*Qn$5U=&CU6H!YjcEYQ@ZsQ*#mhy$o)dC^}oL zoX4V)Kz)Nr`mn4e3UQ1x{^?=i`|%0W@hjL zS|g>uPFVqa|6u2S@WJ_C%P%-acmyplegk#yW$w|&5Wq1mA-N7V2Tdp;j2}ae7flpF z7Se>wF`JS?J!+67O&{0OY`2_a_}IY99eDNsY*X(biO(iF2O?r*QbZa<@VH*Hu5F{JYsj@ zG;n@F65r7=rr)%f&zlZZQeeelM8pE6G)H~kue|xLG~u$MoRGW{jfEh7cfGBV=a)deR;p2pa8T?i*kJqmWIG) zXqX@kDS3mV%gW1}-nm{3xa*RFbi4~`2uzwd5%r@kqlGf+-o zRZD`LSeK6&hJhs?wCCc|*rA+CsxnbGln0p(ir_c!pESa8?ESJ;fQ5a{<$Y$Ow{`!x5rW0XJ&fdaxb577e`EvWh9<7C%<_zArC&E4YVpbwV2 z2RGlC1-g&uu-GZ_gWzF{a-JJBSXo})V5$({0T8|i92O@UIKt)nj^$WT^j1BYiA&oR{X5l!jga-C>alqh#cYJr(D*PS0%5X^Th18Q3t$u2 z_wxC6RV{@Pe4E)~qC_ZOo*4)Z?^XpAmIRP|B^VHK&_ICiD^>V?;^6vF?#L#zs!+a+ zSuENPyTIYVs{WJoMHa~(m4t+HH2AUr?$g?AuTh}3m`sG0K4k@GAq?wj!)5lo{vvVj zG0AvquWoPSwfA@1quVeTi|`#oL=q(BfL5PoPdsZ17IQ&(>XQ0l!VXSUHBwPRyRS$% z(Xp_r3yxdJpN@T3A5{Q+(;M`Mz&(OaD!4zR98RV?L#zOz+%%>^JeHu^G5ubqw4Deopqn)?~DSCQpivEHHaH&K(Z0drvaLsse-Z_>j zgX^dG0xHXmjT|Ke9FFj(4=mV>HpB%739us&Nmb<|7!=0Q!aQ7wrE5_nmAO zln}_#t@n{o#8I0DGM?w?Exd(unXg@hMJ5Wa%gJ;^Ff1YB1P&2wue#a9^mSru5le)R zND}UuRr{w_aH*gd11Jh=4^B-NJS&ob6R4V@pM&e2I0=Xzy3kvD>e64@b5WWnIypMR zU#F zqg^eS+fC%-bGu#1t^ire9O_!Nhk~U31xzadRxAkyx-B%& zVG(^cK=IxL0%MCJQVBG@bRqVdQ$){80pAMI85Hx! z`?nv`e&+NCs!Tby9I-5gpTN_U0r)bUoZtX^m8$}P3!qu0HQkZnYpTSnfhG;A1v(;V zpUgBzQQcA$SQuuYSD9=4sCR>|5a8p>8MD~4JkaiTSo9Yb8S2#IyVv~N$=Mm+O%LG{ z7eOpGq#y|_5+m0CoO}?|1tLgIH^ch zI*?}nMnUwU@a(O(5ZNPf75m!RqPt)E-wdXTQr`#7MHXFHdEv?ZnT0DCa`vUJ*8IFp zRZabaGiXE>?W?c18n3?tD21O!{}WeVl_tr&E`oJa9_)o7%i^aACA?td4Oku>q70D)+O`EJOPq^W&h2i8k?cU}f zx!Z#*O&?1toW8U3Rt{_=e96!aeyUYs3T*mt062(Koo@ZWu-2KIr?NMclfp?t87;t* zmsSpuHVCV(H16d}37S?`R|jeNKw?j{z@b{~k}r|;<&zejJ4Q<3KYszJ>f*Sdchzlw z$!}r{gg98c2dx>sD|E%<+kYA^*l5E)&*NIYSh0DW07XKTU(N#|Vq89UFFcr^L-`7D zxX<0Npbr?mkVSu@Yux;hf_uwggETCaf#DDm{NrV2vQcAe0k_2aMA2bBA=PJ-y1^!0N>8=3m$^=r~PF0!2>H z3zSIxHi^Y4g6K1hL5A%G&abc^V2C2@4XZn~Vdu`#I0z=@u&)V1jVy|>NVx7gkbPPK6NZWFnReg4S<-0&-&(+$2HUJ3Fp7=&mb(G);ZRX+DkyHlD z1hjL4ienbHe{2^NJff}uwI7PhA&iXHZB*vgKPQYXBx){(RC^CV1Oj}4CoH&3mCb_G zd+Xxl)RYVpNyN7WF2lQrBsH-}1IRX)jaSo+tA}TB=Kv$Hb58Nu6>PQIuLOijdk8rk zDk5qwIK(efEw)n3(@PjH>t%G4M70n|?vPXs>-unRUHc5~t+CmE}dOtZ1 zIu0ffy5qbTzalq+1_M-8tR7oFK>-1PxBQnD&mcB)F~;{78>E053{H0_SwM@%L08)Q zkG52h_q(G5!~Bqg*C0qgSP){Pto7SPDY9)M@lXrf7@tDl!27o z=^X|+ayGUuA;18wAJV_ZC{~DIO6$O;HmN7#WN5=sJw_w0@=N3`{a(u#|6JA(GdYJ49iMNE&&k?3Ol^9wyy`)VNne* z;BN}&w7}LyAZ!6CZ$0g!8mY<0kDxF_syy#F1c^RK0%QUh=p1j~$&i9O8jKD)Y~+_m zPS={b z>ZdLT1#k%iE4&|~e89aV4QXFMHFD!8;ZRN+bar$cGvfjh0tz?WtN$U)0Vu~&%CT$8 zKcL7K_ijVdUgidd9;o@}uN!DwAU%Ejc-I&8(<;dM-D1ON+%P|ZH5YQQbrGsXU~rxn zP^vbfA}L^e?#SRlOY0-TC8jqhgBN`9BROM*NlZba&s*$+x)-8O!e$8h~YSX z3nq~cNUGkw@0^{<^Fi1V4+I>h2vZFhwXQHP(1}+---NLMvEf%~gKD4$vnJcNwG`Vy z*#LkPf&l2sn*>Cw;LmTTxBx;sWf+5}_6_zsBuEDU13EfZHa1T>puz|z(lmR71)hae zuBQj*BE2fAx2wD+m}Smk;Dd(}&mzVsQ>2$c%VYoA*tR< z9ngy%sE($^kaOFi`gNiB5RhPzIAQi;i+Z?wtIYqZz0tM@E?2aVjJL7w-42qV`_PQp zQMH#PcTb%B0TB+e;sY*>s5C%7a8T4$Re{r)PY0`?53pim7Awyfmw=yJhA)Bo^vD4x zf)ysZ9278%0tt#NB)uQThK`PoA3hM|Dyyn8ZHZP*U0;959CG=0{T_;J(^6ZO)=qGH zt*%1V`Tk%9+feCw<`69B<>Mo&?%f7;-P;P(fnj)H$=P4S;R%e}ql@>fz&f(zqWq7> zI&8unNDM~aS?dv@XNmWfD9;l*m#09N!SMOHo!%lFwc=E}Xb%lF$Vhe;@H$VjI&e`F zfQB)#H47{nRE!5bC80w=ZK^sfzMq!|&*%y>(!s(S#@-EnNAUK*09-R*>PLJcdQLX{ z?lCzsGj+B==TVQ?lY2p@xKxek9xZM*b&^~E{%ki)Q7x~a$ERx*kcDpgYHK`>u$g-$ z*7I2I4Z)QuNqeuTOWPZq&V^3G?SMA=4XKY&BKtzg%s&+X0;BTK?7@FS*QmeA9LoUw zh272rpUq@hy@?_GHWT3EsTj+jjXB`i60DQV^^-i~POchd+%?iWgHVG_MEotgRG__B zb8K*Nafo6D?kqHry23~pP&?Y7oo)*e|HPr? zs0FwD2%K9LF`ZmBo`jFXlC?pep4ft-o)xGnLp0abnkvRX_}vP+#fUeuu+`&UUw1HTw{ z7fWUCgGO{3@Fj|2V}8epaj|Q67hZqAVq$4v#6n>$Nc#;LXwv|chyILz8S7K$80kj; z&n(XfI6*QH;QP{9@*jP;@yTPGZsu%nwgpu-3F~jPMy_a1d&ueQkwS$fR{bHaE4GMi z7S^ngMiDFip|3AT#yL7%@}$%6f}GQz#0y6)v@0hvUt)qYu)O>ylbVwP4fIFOM)Ms6 zgBXa{%w}TD#2k3bA^;@L-2c4w)RGwor(^Fk$utbPNHGILIS zB*#2^bDp3(hha%?+Mat+HcW7Z4To^djv}1|o$mLq=B6Gc{Am{)h(QX+4Sfoy&;xeL z^a#n+yRY>2qG2wxA%yjo<=;$+LBxy^tYM@Ve<-9+%TaT0i_8db>1hH{U&$l0JKfad zAFznn^Set}$Vuq1m@&eMy^x>nSjAqk;1_ayx|gVKG_|)^mmNSk7mWomDju8HT=IiM z=<^T$)i3hxWvsWLO4^Mw9kC`|s0@0&`TTVTiv!~jT^5&Pp<+=)lPE0aC_ponvFl9+ zt@S(hy}z^Wwa!GbXa(`%{Ut1oWDCqa?#Wn>q8EMgo`3ooP#a=YltN-t`s4vh8*-GV z)fF2ybFY^6l)!1H%AnMv#FxpesB`NUv*@Q~@9pU}&JGIFs%fleY7I>3BNh#^Sv(uK z<1z6%%$v>I!c%0%F=e%*;*AWFCgTD`UgN!6Sv<7b7iv)-oZB^NkK<X{p{KF4o1>TjB#c<|y+(Akn1 z&2x;$7Bg=XH5x_mx)zUeU>MSlRvF0S6~;Ti8kybv)k(Hs^7gUr*b@ZugE;q*pCO0K zp@SS82g{s{G4I=oPE9pmJ*;%3-+NI;QsjwCtZ#xVsHoM|RODt%yHfjQhbM=z*Dc^G zLG$(4n8U3a(YcaFw>1{>Pr-3xtnsnU0}^kN8!8n`@HXCjAkGoH=iFS#vb6LbE zHr;N@+-DuB-ZaD35qo=@ym9AT-^}+eMt$wi--g1)7&z^>(DOsfnFH=3h2hABOvWc5 zG8rF7M-u77?#Q){ug}0O%7qT4Bst@Y+7)nyutZu%c8&g=z!TApf zXFRT6@6M4BkCN{fRNF$RG^g5#*I*KBeC#nh@O2PX@N9X^>A1Mb#HClijRW_OKTp5N z+eAM@pQ9S3+NFd&9k-fw#R(w2R>7@Mqq9)0{*(j&SCv4502!&WTS=1IM@`n`j# zCC>fx^H<%qbuFmREJMg2DKOdzJBg3PwzrOl&Ft1KKH6U{V?3TFJz~u@EjHC;bn+(r z6X`TL)u4!3xzRG9`6Rh#nm#l=H;KodF}h*1wsPZ$ZTW1JU8|M1Yp%~~nXImJCOaX_ zJb8ZXlIgEWwlyVr`@lc$yAQ5)Q~MqzKb{Zt5}GOWsE^`23^%_Q&iRbzTyBvier?w0 zEB=elKhXy4Tmz;3ReXx3zpy$eym>eopS(`s^=_OudL)`OMRE0FAlyXk$D;e|uv&JG z7cRKZ?K8(G8Oka(RFCaQe=M2>&isAeYQZkfA|Ec6xW$}t%7V>;kE(rt=m2xZ;}~DN z^d+%QYr*1(>7%pDFPlw#tSd&p`!-B45v^n+BVL&+T+DV~<{}FNt$P{L(z{|f+rnTc zjeinvbSv$Bj8pfEt~XxE6=u{i!x0pIyZXcVcB1i*wUqsn{8)OtHGk+B)NC1Dv87s- zYf(HZQNi{0jE-8kRrCw|Y$|9SMRt|gF9LckM&tH4IX>Ol`;zzDj0@n#WunqmCi)6_ zsSn6ankbB<>trfRH@g>YCoRg+L}40@xNbTsl0xa2)aL*ZPZ~Ad{FehifS@J`ga?sI z?!xAgxL&{f!zxKn2b0nJ`?~Uq;Sw>;-M7O(Dw|>R!EES3Oz@ZIALb&}hCd;B=1`EZ zX@Y#xQ}-^@6oi1r`A*pNA?b*6;J@V0Zp}P0W@^mxO(o*;t-?$w+H2&G6ho8#MM<%Tb zS6n-BCzroAS=2VN=`+5ifuNoxbyrb+|L^(=px+~}N+1~e-;x8@5Td9+z|JYjk^c6$ zIxipDl<>FbV*byp7FqaxdIxp^KyMoDpO{S_Nn->n4z3P**>kWlNnk$!uo5HCBt{-4 zap+1$n4E#VL7W&IAm>-Y9|guZOti=>yhzRpjkPXiwVuK52BkDuomuaGLSP9~)gVC{ z2{Q0oJ%fclE`oCwi@K>OFx8X1m=C1h+6Pb zn?Amysv4=PIZ`gu0gEj@9Je)@ zQg2~03S41UvPc715p{KWIEZNNz|4mRV0_DQG!?QCF|w<`#9}sz5V<|22#bC2Fxd4P zfTLa>rnI{Y^Xk>tak}--^N!L!UIWPzXtz))!f2og6k>WG=BSvQykFLp+(N~Xti50v zXwek6^nFmm`VTgc=12rbadj=wQK4qvZC>Jqoc0wO6R8~;gryF!MO6i51-FAuYfHzIgEA_?>C zlN~4SFN%^;abF2RF&Hm0f0N2YrhFs=SnTFaNqdcw9~`+|qz1uORVV0Pfy>AfUk1A9DC6$ciJ~w5UhcGw+A~!Tg-dQnJM`s|q z)IOaHOZUTE%{o@0&z;G`>b8U$?52R!PV!3;h2 z4IXC)5dJrzmX;KMtf!A6TgUKi#N_5YfkZ9l0CRSXF?co7A7I~7%U zwUsuM_3;!l0xyq`c^*6x!G!|_oCY>F;caxozJZJMTY{UNxWy?)wA(3?Isz4&=jSzA z1;TDeSm<^j0@S@`mo;P?3G0g@qRCmbMaYokO(PXU>pd_)@Dqs|{c&(!9zgki|9+67 zz!>0CIwi=@-?M5z593JFp+6@uh6H-7H*`HWDT?9bhtj=rnAAAd;LxN^E-if$&v&1osokGQAM^Jw3cA=%zu206(Z*=$XeShIcR-xovUE+(HZz zHaMvK9kriS3$#?W&PDKm*qlx$%@FzJK{l8sp%eNB#yfg#&VOup>|gPQlv!v+;)uFq zbAJ2X@@kMUR@U>kj#;d^O+hgSM<=Lhpck8TK*JkB21Mg{kw{Lg`>-&EYnaE{#>U6u zNk>G{UHgX}GIyvwgh5A#nbh(=AS9LgX+C7rMdkl5To`^PSEo*{ z17wkUf9gV>R+%toVhj8EHr7C}}2!jfxQfa31~B|Ex3%Um9p>gEd79(LZTA%KYM1%w}N=h}!qy3=ti1X5_lDgX3kb^Lo>9$6J+B9T_Hd{Ysh4MMFh2jBNj@T@Mnnar@VB{zI zmwU7qTBy9d>qWwkNR3SU=p-bocjG_m>>bYi@w)on3=a>C%E0Io-kEckkg}P==OJu7 z0ihXY?xk7avL5BbcqHB|b&WjtDWFv*=>3^P&Zx?Nix0mr<_x0YBZW8H16E!(9?S)n zi7=wI#eRWT~A~btF)91JO25w1p2BKOA zGH38yh|s|T8Q8B<|N0vfX_^LTsoUWn5W;_GM#=Z*Mh8fH)}tgh+=PXi@+J_8CjM*y5MWpp8*+f*CS= z2amd-jRy)oKk+{(KWg1QJaBx2Fnx#iGeT%+xc4>S&uvuv0>cKGtWjRWR1>MaWiEr* zPJjY7K6n<_Yx}z(&;qmxplBz>1_Ca2EwCM8XiPHp9?Xy87*UIU6HxI054T_n$aUlB zpwJQax)!Fu1)ON!O9Zbi15UBG{oTf8<>f%`xD_G}uF*7Pf=3*>t*}b5$HQg`&wa_A z%%F{1Jp)QIDFkXIg02a6&`%EzTNiD_0@I(T0ia$@ji|qc2L`792__6Q&~V<8VjA_T zXTp3n@uR$*bEU<9LH^D)?1)}h8~QW|ktqwr!MqbC_+Y_c2%$B@WUf-FErZRDmH3+& zANn0{1^O_l<(1xeF*$L3DiK!{iB5Y$Kkiq#Cx{Xe#y4=5$NhlpkVr#z;oju_B&~Pl zStH6ZB#2QgD}3e1Rd;{?UgA#i=sJ*X$A-nsmtm?nRxBti+-J-U5+}6;5TF=KMRYxV zQw&g0twQKTiOu=DMMSOT>BKMuLaSFL5kfmM`yLLY&TN4R74R`hIiNKLolEmPq-;ZX zA^}uYfIQiAZkX7B3P(wm=*uPaFHQFz5IR9^)=l``)jKrEmEEP2bSwwzsRi^eXVAy= zr3Ur~=i@ z^F)}y%n_qT=%=y+j{xWm@{Z5W9uuJoKea&j3~nK=p7F4?MQoDSd4NKot`Z~wQDvy{ z7>ziLsF;VKA%PYivk`J7wlZO-k<&9(*uY@m5r_mqXND|T8i8-%AV_(NMBS&# z;HDv`B8?G10^&R(flktJn&}UQFwBN_J&X{<@fSkm=?GhBsXtC^cwC)3z-xn;YHzQU z8r=(+8?c5!JmReYaf4L{bMh4duRjtN(JqojNqT%I$jg8^4w6GRH=u4WBV;Z`WimlO zkzNmK%75s`BUj3wGBE^qxYNH=-a`#QfzR(#8bswANt|lH6RT}rU2)UWwu9qjhcj4U zT-Qs+-)yn>rN3#jg}-BN)z#8Oi%c9J20EZkZJzf6OF$&wvF$E<%>X16b#``6P2JDy z6l3!?`k=K5yKEofe~AW%T^W))L8!1IKg(Ds0n!>#CCm$;;DD84Vi5+`$HWamnU4+l z08F6{PrLW~5zuLY{&8jttY*yloutTb-~lC)4_n(>&F-Uq%brcl3sL! zMJF;Z53)hTD5hcH@Qx1_Y};C$mG|I@CNlY}!gAIf&B*jHoEmqoV+5GnS5Fj_(XpdV z61gB^6uN>CMi%Y{IvH2jJaaOIryz-rdJ5|kC!W#g1nQojAQu4Htl}t0V5O_SCh8I9 z>J_)=1|_{gUlALk^d_VLg`I3v&&odd#TlX(uW}}rKrL!i3fYt3qb|q+@W${{@P=p! z95O#6lpxPBqP40PqQLc&hAjscU5NaF5)U?5u0(|y-0f-wXg^Ij-EI<-j58aqdne z@r+81oCBn zgNyuaqhB~rC**S)_3stu<@vQD6s^PBwqTG>YEF$0ssd+3FE`dPeTS)5Er7Lc(Ixq$YdW@;8|D*Wa4}O^OQ?* zTIP>WgEWv1?vyof7Qm?7xlTz z+_0HHQxA-tsS+`=4UwfWoHeiv5*p)MCcI2e9Vf{fPB>BV#Tw}A6bNXIYYC6`z0oI8 zk9>4?7TzZMYCwNt=!p)GL=cH5eT-zSb}s9uE2xFvNej&6(uOTHgBlss3`SfD1Hj;i zlrxnhScIFM_k5qtg4V0s7<^L@z1BcLRE#|g<^A%PjC6YYN6eL&MeU*fzj0e_GX?!$ z>JxY(b!t%Kw@i&5;&U6VnGwJA=5JZDhH+Pi5x})S1L6Q`w8_ViscRslxY z4!SV_uL$jc{a2NUgM&j-_1EO&f{4g&<6==j`=Z!ynnZRk&Jb~Wydh9`qBgGP<$guV zNVdt#19wsJg#2AZoQ`SG|H3?e;|TpJez7t$(QEAS5iqI1UigFG{jtXL{RU8*1C>in z%?r*G+}V&cf^v#}9)Q9$I`%|Z1Ozz_;snc>?ofc$kbw*5c5-O}N`pS^XTN0KAuK#@ zMJfwCW0An27-o7~)BYP56<3g*k`~S=gKMA=s?>sIu2v_U3r=yWb%>d0Xjy*=8X)m> zYr&?~B>(S<<9~{N`(bsJC41M93o}Vwrx)n)JrdR??>AVH3{YH2W*d&c*X}C`*{BPD zpmPD>b!X?Ghrv{ms{U_qwJ76?;EcCq^-tCY(_^0|NT`7u9{0mC>=R}n>be5%S4j@0 z@9t7ZC>*!I?^az+^UdI*Ly)a}MqbE6(j~1{<(({i{AgVZe_RGwW zrg{Hc+75vYExKAzxz;4le{S7=*l+1IV_#l{KN6vS-?7_wJd)gYRXiN;3lnEEUV}X_ znn(myfmY`RQdvSnv7S+lthO%V-QaUVQ7rb_K*s?Nu5jEBcL2vFd_Pc7sW5GBZ^P8? z*tYbEDiI9`31;<*rN`YKG2A5tLkCnsko^c}PA-e2I7qhWWk;8&pAA&RE*%C3#D}Lu z>zyWV^13EBj#`ja8dJH@oa+ANS(el-jP3Qph6J_kziDGhye|Z;viY}DyAQ9thtb%G zv^XYw2dsGUbVwo#+eUR>1aI$aurf`X@kW_Ej_w9s5ja)BfKf#kvSh|9A%lBr#zfR( zS`JY*RIGK&9rmguWx{5*;JGsVWZSw8yZ&$jDfzJpM2Kp!DMp-qzw^uo>P^wBb!AwD ztmm7q-~>V7EAP;~M;eD-K&C})jg3$RLe_t5M)CP?9BnxMYhsVvF+Pai!)2_|H78CW zAtk*0rt!4XWWo3&tQR`8F?6!3F=1gi{y6ec;Htw+`FKnza}DWgfbP$a^Q9J(22hg0 z?iYI_Ix!vtUC0e>YJ!4NTU*8_;P3xpr+Wq+0{Bg-#YS6FQA0ujI13mSq{DLB5otlqKfQN{npu zLZR#j>g(k{i^L2l_A!ksPU53kdTQeuxr=(%VHj;7mzt&WsNO+@ehA{C7E(FD1teCTKl9uWqM;kCb!Nm0P7f#VnCSadz`B0@nt z|GvI%UaKS3jLvfB03XF?saN)ow`8ge8V1KF-utgh#G%~RU_D+Fy}ADu9DeZVLfK6m z3icD|Wadl!A>szpYRv&`S0KPF)$o4~_nX+VC#Y*}j1Kzgg2K=b#bSE?Syb&B$+(44 z^73eH?(;W-CXO}|&aY*xuq1%g`8G*1TZN1&eTJ)Zp$%mM6jeTA3zwG-fKgq=AHe3c zu9*i7V66_Q{)cpvC@ub93y^!tCrgQkf`ra6#>+++tYUTJm2VOF$1RmldJx8ULkbU^or@IVdP+=*D z49h<`k13#06E8>qYa3y1X&W(@{Qh~11FXsAQ~ne9Wn{sVAGuVC2oAxs1mBbWzq_FENf;b>ww1Ve52;ASnLs?Z>JNHWXBoda>OVB! zfi{WGdTYvhv^@)1tE9cq`;>8L=d?kw+1dn25r8okZDgV#U(>JyvD)t`E<+E5FexxV zq_rucKi97vv?pR?RAEw=YE}h!?7hoRAdWgH$LDm!s`_;)sQ^oJ2|m{3`IFCh990;^e>uG#go+I$hv4#%6LlKfohJkYI^cb+yEN4fH(qSNUk6X&;I^jT-nEPL9~Ktn2&FKT zrI&E`gdkKQZpPAKpmF7lJ}bDluFY%v>-h&K2z?}I^Uk6U6BD3dP$a9kfY3EkY!Ej+ zUsuX`gWUQb*Z?`skqtyVq(#Y#Hmd{QDW5|2m#q3kB6oOR9#j}%K`@9 zphE>>Qwa>)5COrZ&yZ>X>Xoy76He0aDUS=D0;Z=@3p$IKiT}hvqw5qD6bpS4au(Gy z(Dr}@8rJW{N-@rVnq#;njDvtF^-C&fuuU6mB%r_x+C)VXLxl`v4UJAxD`mLOoR5D+ zXKXykd(_h|=C8&sk1SIDN$IU?Op8R1vL-s|E5ay}+CD|iOJl0Xxp#^a_snEJBzdi{ zb20E6Slib%vASvVeLU+jIqx&jU^o~_PWx<&x`VT$<)zFd5r@Jp?QM#o!(~Q_!t`ZS z_+eS*Csc(|MG@m>h8)SpgG_|c07B2VLpt5dNLwH1zMGWnnwi&G#-$+@hYq!cxGpMQ zyLoE6=ykNW;bZV5aQO%FuUVzfd?r8`kTkRAH1gXr%nfu3?rk%`Vjk?;7&z)f_|C{H z!jQV&X|uSjwHWwqI2JEeP^|qSUpaPleBs(VzNcahi$+W|u}mRoQu3}}7E|L==5W&2 zt#s9zJ7lmrLQ%meRD!2WqkElk^t(6y?t*>|0@$waJ6A;Pl(Y$<6H8Vj1$}S#P_mWv zr~1T3;mYQ07A7!&uhL?6Mq>NxR4p@zJgRh;)20R2OuKBPd;z=~EVm2pPFcSoTaK0uoD( zsTG0_$aH}c_Or)`qpy%n_(%1m+}|XU z{}O_0=Wuy;kN}P}Y&!occ`T_X;14NPiRbV~c@Kn(g(A`}s?a-NuNFIiCHk9d>tvMILt!EL8Rb&Ua;c?eQvOq>L;Wf(cz!@3(~$S0h9y z33JFf<5lzPe{5DSxI3ADHtiaXPsSY-=1XNyzz9hZ$9%xT#KLHmhWTsaC+p|2LO1@t ziMp(S8bzI+f=%|<)&74ki?O)E24ZTznPATp46^-_4h`1cya=nPCKnlel7oPyT|SS+ zMxx#>(WHrn7v4Dag$NISJF?a@vKEUEo8Xvyu{onWjF-v3tXHe#^dx+Dy<}rCouwFe z+TD}hn)AT#1JAq=(j!BDk!Jo_;U)o%2uGUf$$67hdyP<{%XqZmsa@LjP^^`UfL#~9 z6{XGooXEda;u_lnrFi^DF>}^CPIu;NE$vnwwSDrR58zm0FNZGHNYB+3QHE3+QvDNi znV5{d*xbyiAQDc$09l>UtwHx?M_vhw8vW1BN5r2beO@iUt2#~(XDcT_ss1Qyz)z59 zWJS?Dq7bi`h#h>0`M}3yAlgVO!(`*z`aS(`wpff5k9@0hIvp3tYfq{I|0`G|M-xB6 z_tJHpMD|y%_C|7YTl1O3wrZKf{`t3FVc~^>XX20c4MZ%rX6bScS^6|Hr@}2Ctd)WQ z<>{Rzshzop?zGyMen=NTb%m7K-B($(L^d+^_pcf0hn&1DR(j@*_|G^W^l>#gG7y`&*`V(D@Nrn~{|G}i3;NO6o28|HG=V)v7L|gs0yMUrT2^A#WdO} z4d_S9Y8>~E+NxXT5Plp$fG_m4qPjCRq#$NmlHjrX*;-_H+`6VoMw%kSUCp?!qu(Yr zB)4}tcoP4;Uyb(Jh5R?xzPQkrCdH;*a{@hnE%C44Vy3RfH9mPC%A%R7xl)vXk^;3m zYcFVwOF&yc08=<7qMsDo#0E7k=WK~Ww;#rxZI`WtKj*FrQI!hO3ZB>LQ|8ziLj5rSjiJ=)lef9FQ09tfa(S9RK^S zdD=n5;|z8{z-y=yjhSjY;)HVr?wc}c07wJk3iI-qbi*)>hc25EweP0gIb8uP1`JA` zXdm#k6yAO0!B)b`KY2AUHniH-_nc4vj8?m(%t@$SyQ+kRT)&tXyGU)6+HgAHvb_hhY24zH7p}K`Im78!XoW3`XUB>~m83b%Bi!Nsq=i%#Ur>Xi6=Ct)B zqlsttEbc9`FJBUUpUCV_w))wKy^~`xu%VrXTd!`_&$u;<%UbAqGQMI3Bcd;B66<^u z)F!_ga$i#Guc?g6_b*XQlkmwtsjRhHMZT%@n;H(oE@%_e%D z%2Bjt3V9L`GrwWsKWvh_S7tIz-b{0!4efTK82g- zOZ~}7Le3>s&kEf~u1l=%@R2WKWP5DPE3T(smA_%2<$o(aS~2y`RP1FZO-otij7~NO zZm;PV5BJyO*sG5cdj02;8t)5Mwm*7B744>Tkc?T@SCP6p+DfiU!q>D~z>Dz`~ z-2={TlfsXiI2Zxnm}kkTxS{RfhS#9i{uJ3p;25XY9M;Q0Wcex59lnYw;?ivn$aX*~ zuvBzjSOn@Oy6MaSrH{VydFN{M<_oRGexxH2BgzMQ#^opVSs3+gfTBs7I5BYFdZ?j!>jP!DDKZh zuetzTf4v5w^ZQfDidt3p+p6Chun^o>9ruzA-EVbN~u{<}LevG-t+cmy)%St%4_PY!BlD-hS?;hD8+0PO* zEDij}=F#~cXu20t2A_yn(S2I?TddfV_S!Diuv?AtyGr28|1j0}7+J#9q z5;iw4)()dsu`DmjO7?>y=NjHk61y2A2-Q^B@W1L?#4Yh|oN~8IWLuH<@hITdcx}aX z&Cxc^CRj)5R&x-voe^ph$CVt08R=Cq%SMp{GfKE&%{e_mGM;+8qd)4$>t`P zP`ypOc7)!(b;m!{H;pMCH+}ZOb#5h_{Y#0lVH(w9HZOF}$K32M&}GZN{q7}GX4MHK z&f>m#R=!}_)N6CXK679azd?>{5$;gW$hP1k&B-p$e`(py@1AZYQmL)79$@ryO<%*_ zRr=cLt1WrWc_ICt8=|CY5Zj{jd<>d6-AkhE`5kC2M-8D;@@|*?DEv><*G;qNPPW)Q zQAM)8BG2HM(fzIEwO4f?SMRf3ta^TRv?+$_qvQ+y z-}dk#c+;Im4|o+{5M|#oh+qE;q=ZRf!^WvOFBxp-v3@Y-NK_z_gmE#9SWGGY;q-3j zBX<_$g}vJZ%?_di8;bw8SZ(9Z%QzB>;~PO5E%D6k?9rJ!OuAv@`6La?R|R^Ol^{(y za2AF_AXe@%MfjdGnB#6KIU(RPz()s1Ik*F30^lcr8n$TDXUZd~3Wx<52w}&fz!Xx(3e-CoC&u@x6;FVoLxAFXghvk; zk5G4Lv_g@<#0QrFSQcO%yA3^oI2tfragvU`tpxUMp}L7CP}wFn#Cd1_Sv}yzN7n(+ zlS=^1eBw~(bc*8}e6SZx@eym;IRSIDni}Z82p@V(X?h)aTUOrBY5rH#cID0yJ&%JO1< z4o`?j$j7KPoeqNWx__eqr^kEV#X=(}JzvZpMi8f6!T*|tY}3T+TECAFCM(e8LT zKW01UKMZXeunrb&Zaqx30M_Y+DMBQjtT3=@Xm1h!@(j^oQb6YVr&5K!91Tj3X7l_j zqU@P10F=l^vGzupB(4Dj4X>P$24w{q1s$KdC94CjckWMUE~1K);(iEUM)xhLtW;wn za!#F}orSUvCg~Ji9m}?#^ax+ZZbyHQHp}v-QeyDvcZ<$hHok(mcjaC-@!)9fnl&jJ`|GyPGFe-X!(VI0 znJ4+@^e#fr4%v33k{uWTeM+skU`R$yFH#zfk zNs6Qe`PUuaN9ZMdR(6Wle9nTrNv?@*R)))$43RP=A1N5R^osA?V@vK2l%V-5FYzQ| zR@$${w_BB%6s{}+A{AR|`0=XGO?~cWu&BMRlP(FtyzL!?-DkzElr}O2yu2i568vXr z%yHA~#3)2Ky9TFaswYV94(S(3MHDK-s2M6dKV8p^r)H3bp1Ft_ygOSm;1NNyBcN^> zZRy*rq`2EK5mBFwpxh-;D_HZDT=SAeGX3ry^xYVn3z;h-ebz7guR>amHcEX$-nBNM zpzl_`_IY2`Q-2*DRN37o%H(!2qKm#D$@vHo31xR7?$P`;!PLy(-L&i%H$=)@5mEx9 zV^vZ)^v;ziUqo*|j#Y?vBb#qm7M@)7vH6^c# zjK)LBU@cfU7Cdby8-W7#{0tSDs4w<{l)=v?t;GDE zEeY~I+m^}2z=s=`F~YmCA3C2>&v#()kLUuhU*34$KXkzEzL?aDTsfFOKz7IpnA|0B zQ_Og^*yAJF!|`Fb9x2tJoivS}7w?2l6bhxvWm;kKAA$wPCTN{o(@5 zK2wNHfFLRMyP3ccP?e+`xKQ}$Oh{fMDvla!0))nwntB|p!-0gvm;pdq$jgK897gT4 zRwR#Gfs_d2`I(ul3=R9Gs)j%$_EIZ^ugm<+n8-NM|awDlOgY?08a? zl5@Hbgx#=lJ(mT-4M}(7E&qRAFq!$|f+8a0 zylh06KwAWL%$~CX$GB!(b{?}_Pnj?&Hj!j&qae|4moa=U(98T2G~0j)n=CA z=nSWmQ_Q=V#OIg~*_RsTv^wl3+p|KdL^$mwmgLp*D&E7<$v-`ex8+?KV1Q0#Z z$-+F>^VI`2wWk>>r5e17cEVIJcIvVPH7^FuZQA__$o4@?`pW=#OySGWDrjtkG291v zYnYTwy^}a8W5s%fWm5?vSje~%CxN~NAQGnDfo#8xKdP;2YtNarm%E6CG|)%eiLQp3 zoVw*9f`~nO(V?&~(v~F}{@oEyY$lii7C=st1Wh|9zKwre1uF|M-3T$dZ1oyl>1CgT zXA>pkB{S$EC#%Bh=fLgl)OHVkTRXeE(@;bOIcILTmQQ{60_Y_)wx8&@7A!c*zhovQ)6B=y#aQ={9ys#=Y|EyW7|9B4_` zWBt5rNYl~x*B7k9(lcB$6Yn0sknIZlZ}5}Ja`3<)OG(0XNhi5ick**Quax|gwa|wK z*8F}Ss>(v$^jrp6sw1|Wd*ZZbD~*phlDb}5A>Vu`aK6rT@m+{}Qut(?E+N#OR&*i3_c2`-B1y-nsKJwnEi!S7`$RTWuY-JsuB$8f{gWP} zZcL%?*4-KIB*SVo=47EAe3)?xZl3$3UUA&B9U30@W*=NueRJ7mjZM6`U&)sHHnpSs zg#iOg_EQnN^El!f#)4Bx{tj1`w|ZHddhJc#?y~6O2GmLCZ)2yo@4*{5lMvpFRMlG~+;kl->h}ywtU+3IrmGuBoa*Cd|PDK=sBx zPMx@0g@n8y+l*t@u!`Cd=%+bJ%hsiICxBUgda4oi z#k%PXjik}s$_g)CSoK~AJBMRRMjN=KKqqu{G7loKnJS!iU^mE6cO6_BaDFmLtjwo2 zkOMQ*YO?hwo0h6A$P)DQ1W+~{dX@$9@}RDR?3N}7ZS1Ra9{Wd6)o*YH8I*Wo4DULN z;Y10P{aKKtZ`d<0%v*4q*u;@hX>MMgiDq39O9*K1V8G6+2pVJnuUPgv z=t`=x_WrdOG~kJ%0|J*z%;O$XrvEZRv1^}07Mm-?Q2Y-H1!XN@CBvjZ`In*uTIPr8 zBuH85%zwgRPE64ioGG)xs|bSe_Y{ChhtsydI!ZM()1ms*A{DY)0Kf;(I!FUSM+wxY z?B|Jz=B)=976z-1jiNNuF9s>MZQk1H=~AScUo>imok?}k;F)`Aqvqo)svGEf{H8g2 z&cn^**Oe}9nt0y&nA$_a)8T7IP)me}$8lP!Yr!-BL|YZ+=?BWA#w(Sj=PT!GA#5EU zGtN!eZrX444r1I5e>jL03m(PvYb2@qB>?Uxz_dRmm|BHR7=T!Gvi2VNdDB~tz_zHL1wRk4Jh&49$OAj( zkKrgf)#Jcx?uw3 zVVXyW$Y6MvL_mdgeVHls6Fm+cQ^(DD#axL8=o6G?X20xWlKjw-H_ z8vPTogYJX_rQ=5hh8OJd8FMu96oqk>pNE?Egz>)K<6Z&2N^Q~1JHG!E^ItM|7)37E zz;TPT6AT^&DJl~+>sL_eVIM5d&81SK|4{(wC^onowd3e-dnX`E1>&f`1aOjTD=P!O z*qc$hex*i_GYQUw{s^cG_fcqy7*gsJBP)I=#?Ua^*7M`hN=}S~WC~_Kx+c3+-hS=z zJmmOOs8<+=)^xiIO>0f^_oYl--=m1rmtih>lMNhNC4Yt0hNCK&DFR!GmjuGilYFUF zUOlx+!i~pAs*Sb`nszgw&-@r>q9yY&KyVL*4b!(p>&f~ZiE!ikzgz-mI6wGB9&2;< zMyOw&8*Oe4$j{M~2*P6omE%TTfP^8C8&>7i)S%76!Ojl8FYd(ec7yd$4Go*YdISqj zXVn(yVyPk&M;n@&)YXm_7Q90YK)mw{^;|z|7)owl?B#igl=vnkvsedXC^)0VnLG-i z4Os$b8*^+XOY(Bfp7SM$W`IyyTRZi(5*!d&!~cUuOp}*=p*6=%m^bF;!p@qtf@6kf zbFumK_4SkRp4X+|Pc@P|yIAgle_14+x^ zKOese^P?tq3>zk7`1UdnV~80ubtO${Zy!8;#(a5JOx~Yw{D_K;Jo&={)XW0mSw-}> zy%+>(4Ay4}3%f%*T$QDz1YuHe@qkRkN(|bSq76{IkOeN*v1=gdY7e!KtRI3kk*Hp; zn6Z~zBXa%{f`7mv0jOG#ogcXlRs7i=;RW6WxOQy>KKGjf1P;YefI?#YDNfNwQVsxt zb}^lkO*BF0@)z7Se!~vd*2UnoZeNtkZaZiIOCA#hbBo>hanS$$KDX@RX<=aj44gYV z3-K{o!g*n($H$%s!qeerd!MRaUt@e`Nen2@e|_@P07kkQt+nGf&_2Vb1hizc zXNn9REY8a4h{%C^jpJ|c2iiJ2pUzk}f0U_qtqBehmC)oY-(MG03$$KmTkZUhBY)ck z%IWBF5Besqspej0w#{YyC36#Qt)sorM#RamlDi%@m|YQ$w|x-2yg|9tT~@7aw2DMO zDk)~Uacwm88LTzjiN9T%#^0*G&=aP5r54+S#DLlSY?SwrTh^4ENNjx()(3?I6x|8P z9l(FBJ2B3yNYnv`8qa1gTU)rDK!2PNHZR=9toO%nJ{nb4XVKqCGcGR=TG6p`nsE-c zwuYo=Z0}~OsL^xGLFasao-(xL#iY`974v;fs4I)?bE3iq*WstK$?O!DzPOJwQ0MY^ zYg<99BPk~@kF{5Lu-*3Qgo9?>q8cnyaOliGRCx^i+Mcr8(|r#rJ7JQqf(NThs@L^p z?X3kCsb9qH82h+efnnK`YDaqF2*m0Lo)+BKm16je zxtg8jZ!Sn>kULym=Uj0_jP0BP?rd-Qt~PDf6wcL2_k6C#vU)jk@cp36QqEQDrjD3# zcdTMq2nE}zW*QuLg&Trp=Ah91XJek(;wzGWarDChB<`6^a83h|`rOnMoKU$F;jKbx zS3F^hB#l1oaCLfGGqKTO1nC|?uy#p>L2&poz>NSW3>o7|BJ@)I@chjJN&m1DJCTgO z#SA*Wpn8|ef=ffMd7?C}-DOK8d3L&&m1Dl3xA|c~4;XcjDFUqsJG*Fn=(B^AjV~E4 z0m=YxKk#SXzXuWL^KWDDq~L;;p3KVk)&wr#arlv~cO7Hd{CKkJ)7P{UAL)xN`Mpmf zUoBOKq^^{y2x%U~#T}lwuaoq%tQS+<=yLvN@vgDh-XluPEg6FX9?Ne)w*!lX)8GD` zI(S8ZO4VRX@7TYvp#YL2bOSsk{qw*W2|fcWvJ!$oH#Ie|J1;I~(8QDk#&o(+j;k+F|a-fbqSX(;`sKxk7bab8@+@om{)7n zsG7-(pLlG29k1|nD!~8?XYoEXS5az2QJyij8VhH`yc<0N-BR{D!p`?5C;hx^R@dL?_mZ%+AQ$*&(=Ft;k6vWnY5AM?@oU07 z);n>-xc>%N%4^U>SO2S%yZdf%{ll+Xs{19?&FzgY_~%0J#v-I0CX9VJu)B;|==BFl z*bo#SKihuMpvyJ*`xVO2x37niCwDfX!Uw+10@2puPAk?ZVPF z)U7FSw(}-rt$Rt&3n&QToM%4>$?I^y#PD8s5OZ;!n9Bw{TT+e!5^^;G|#C$&6?u*n{ ziCTTaFj}~wNm@ZSLbqtuB&6t6CRhBkWwiVadr8JD%W5J0&2vqU@j}%R4jvazEra*Z zDCm=>?e(-hpNKC@IG-=?aIyuL;H9b%{$>!eK&*eSOH%w$7Kv61UqV=XZle9Qp|w8c zyP2O3DGGPVpWwVdsIYzUn4hx})wxe2VwF{;gLuMAd8So=i;|D1q*^llL$-`64ne1b zt|}3|?B83!IZEv>73@_c^+da`k*AOF@3^c^*R8v18A#OPl}uyThW7INPrb7^>B(w4 zr?)+RrEYfQlzdHcAcJ*E8J^1>mPY)w_^E8wdu6+Puk<&VO{^M=9=3>>Ufrm~4Ku|> zs18OCMa4K%P*a?~Hr=cEl7>|BLe%cCW<9c}uVd3Xg?%-9Hu;YK3&!$Ho^tJoqd^IO z3FYcN{#lDnWulFvPqYTsq27<6*Z{P0m1REcU<_pm*V|6bb`)!W|AsK=`msX|@gYlj zGV$SoY=+mF5e^%IE+VvHsIZRbESHz@Ml7dli=C znwC>h_Guh(0lE_bIMH+6CQdsc^D@L*&Q0hVh(?I<`%2y#1i^DxQc;g{E?>9Q|(x9>wyBjrBTk#K8V^tSYJc z3FnT>bX5266w>(H(cprrPZ?9sTvJTS4b9{Xvt!`QM#c#u(*VpC(gxSwKb~;7C)vI7 z9zD!jQyo0^N3NiPmT3# zw7ztaDd5%|)8IOI)qFLSvSP7d9^NG3u;QY4G2Y+Ncw~k>_VHtuTNU5iP$_PdzZX1q z3Q!ok(ELO$JXWoZJH1%o;_YW`Z&SW+niNCQ+*?(-RH8x|&L?Qiwiv3ujyHqzzI|4l zEnB93@jPmFzKieVU2e9wQXndx+r4AddJ|&O8^gcp-A^ZjJEgMZlHd5vOMP%NJLW5I zc{o+%nf^MHNuBv#>RN^;1L_{oN8h3oWniNNed=4isDTdNTTXh0%*={-~v#D=tZFDnO2( zr^zU1JSbJfS%H_zV57*=+oaheb##`*=O$W2S$%54Ug7ef{!miXO?ySMjg{GpUs$Yk z+-;@0s&M2!pDmXe`e~P9*KaAYtI&wBE8UJeZGQ8w$x^129jlF=x}-)wdhuO(vq9xa z_3>N0kI}sZm;r#%?xlFv{sG*N(1-_8u__T*T?{xE1eD@d{t+%2qA{Y~>CU zWGNYx2rg`7#$5kB24L_*AV_jQm`Y%_mB;KL3sVT`fC7Y8arWH%{WwRALr5X?TYW|>+IZ%2s=a+TVZTtWz4H~iOF-JGs3yK&VJd^ zzNty@x$$LFx~|;lr1Ao(b5gx%#um+fW34{z%RS0A2i?5t$KiLOr&tOeLrA^?@6+u# zuxpviz00r~rc)uA0^!l1&)5^_vW@Qk4$;a{=rl<=nsGov`7=BafBl^U*=c2=_c|)S znBMh|B&8UF8B0din**VMRdz&?Ye$cn5+^>GTPf$2rA8Eg@ox}3 zL_b!3KCCKuRY*Sm*jl(*s%8|0DKxb3{bSelAM@kvdMazv@1k(hnim8gUoGdv zcaiXJxh&0-Q0Z0LJDpm4joQz79#>TiBxdfq!ktg(9JDG-R)?@DsHbuu%3b5o9ygEP z>Mcr$QeHwuWe_1uXDY{dy)ay@s)l@g?4;t-T4mW=rZD<_{zq&7K|}BNYAHiLd;T+F zae`FxKnRfoR@A;S`d@uX09?KNr3U+rtyxkEqS&)AOJJG#2ET|=o^PLPpLa7Dq@ zEv_AZWA`ogz3Hpww{v31nx%W9Z{iMbD&$zDmAhT|8fQK({zbEF(SINIQx!uY=?A0D zDA_hLFV3U!YS!MiK3}c(qfC7=acv&B?VM&BE(0WecB$$%4^cL^b_^bknem?QE+SV_ zf_oSAMt%3Q#|rpbasarSSz%f{oCS%HROGHK*Qh;PP_m7=+y z^D{whT5B(MTr}E^nV?rA(jC$5^-X;blS}8Q^>kySO-YMRL;9=GW|P<;`q)h* zYSP{Wm^6>^H2+%b?&==1di;Xgw0S$u9_}u$I~KIiP7Z#S`H~p~JkTA77|34ks7S~! zGp>L>BQO%42OE-xWNO3(D9FGuGU~TLc+p0rjKAUnP-`t`z@Y@E+1taMz5IVG`7Mgo zE3P_T9qN}I!fhYE1|AKJTb-9$sOcg-S-aUh|6aBPiVc%}_Q;#!Qqi#!)oItC*A;=}yJTPl4;k&CVX!T$??cE?R1e zP}a6kpf@u5fHYDgtFGB?`|~gQ>$Oy*Yr)xjNcf{(YlUaobFTU*SW>U4I9WMeA7-y9 zh%P4XjQC-vu6^^^iA!5(B_qcvHu(A}pw?Z)T6L7(L#ABpu7qix6QWzae90;|=U3Bm z3pe`v?2lc*oVH{Hkj4bd`)YL#YZ=wkx0N8;DF=nb*Vlh2UQ16cNt_x$_ zD!D3aYuD?nm}8%-9T5~Q)VGl_G7%k$Ssm?%o*-}ag{%+npi~Se3XWb?ZYu%Q$BctA z?muFh3-b-fgHDIJ9Mu~Z{{MttIP~bR3BO>o=s_OL8Pt??v=yCtto811!+Ly?;daid zdS2H2AKk&bOqNxauM1sytF$q{!>k1>b&XyYVwQ%!K@g1qh0?8m*WJx6dCjG_OlRUA z3ZjmjYK#ig*v`q6wvNoAqP4AH_-j%$hK~*;;Gb6RwF8&T!V-VyXcZtcNosWr1y#=q0c zwINO1#qILsM;prdctY{|D)-;)vzD?eH1z&+GM^T3Rqo3T8pZJ9nD!h-t{9cV6m+)b9yT9Z%T& zXnf{oe=7XQ8u~L^cJf^?CE)nC{=XNb_XhIt{ZMCx+{ARJ=8lFecCnsQ-? zlLOs_N1F~iKz=|v%(;T*v`C%b1R^CgwUXBOzmj=mgT4aTJ@CT{!$gZ3&YUsaYr@4j zs8AH(<>FI*gy;xJNa}gigcx)vZ*La=TA=>;PBmG58;K8-Y3ZN6YUKs22ie*S(7nwh zV!`+P=w)@1R`Hxv!)0CD#OoJ&i-&ZI+fNpD71m#)%^EfzPhihGPYhoS1Wx>2{yFFBNL=o0 z0s6Ct9m7=$IC(dt&gm*vzsCQjpOM9E?qA(QpX&%MqU`3|Lf{{`=!Q-A*r*Gz^=Mo)@)C?tGpBxGtEZjFxjaMpo(YW*S!2#bFSNn}hO5z2hms)_Z zG~y54yX|ZHX~-nH&3*pYy{2el#cLirj!GH#)q%Jitb{CGt^YQcjeHCfTFP|6+=j>A6XPt z@x8^@$mlhw8sJQFUK?hIevLaa5}G6w%XY0-4Hb~Au(dJ8W6MuTKR>UulJ!zz!UZa~ zAyWV31lsrzJh<`!N(fC7nJJfxsU@koI-xAQKF6@d?YvYL2)eVG`Y{3e6505PmMR-0IGjyL~(ecEyb z<%Gs5HJo1D`u5+XW2U$0sAFcu4iNfaYxmfnmgootQ9QI06sofpemlh1uARA9Pceil z^Po`>lVWkqO(0T{yUbZ$is*PX8Of&}y z;&Fu{;m`R0c%=Z^s?k?pKbR@pj`hO+ckts!AvBTM+as`ir}+U%`mdD9c=7sos2F@% zO{)84ogsngdjI;e-C1z>5TtG(rd)$;r#w~A5T_TzSaxmT^xXVQyUmgXM~26{v<2tK z*z*|vj(;SO55Co~N3Vgl@J5c(6;J%GM>XVYjHa^xu?1KVAi5Yh6d2E^cA%|5%~Luh z#dZ#Yu~sgQYx$TW>4@VBoDogUDA;f#c*A8rbvpl<8SryRkMq!T4e z_m-g##fG-dcBct4WpTQv;e}=-GPCB|`TH+`79uV#jDf6lxOgD14zkB-x#ICR+-bCQ zP{6E7vJOONa~wf_+0RGq7x`A{=DM`Ga(NXj3XS@Uv$Ofktd>y_3kgtL^cv*SLR3c~ zM?P237G*H0g~k!N2TMtKxan>Mr}5noc=c@OK*b9sEUHpZl%;0dAqDksqy7kDUcG=I z%k2o#2q7p>i5ao~HxHKqt_ak-Lj+5%n2{KOT4_p%iJcjF_J1Z|8{RAg)(?d&ZOum~FVgT~tv2R?MRfL?%K|5k!PFQomdLk25$@1I z@=tV+uAo=#86+lJ)It}wjZN-DmRCUpY9w=eC+@;mkm`0Hhr(uU{I>O6T|P^W=v&YL z!P{)r4n=@{Yf$+_D;+K!ko;=+N;BA{IGo=I$w)bOmuFaiAn{-+;qtMOJ^t;~W5~@b z)r^oWNH%v7%mR&!%KIDNAlf?L(6{o~O8b z9{iM`eO|7C7A($4$j1g(6Xaw~xoByjYXv`H2hW7)6FB93uc4{b_xkcL7|@{eSTYa9 z(gRF&cc-6$S$OmCgR%>k%^dVV!%GZ7%6qE?>;t6@G1h^R#l zXhIgSnko$S^&!j>g4uQ|+e0MtAl&JjkipdBG@LHqxjL$MJ<@9MR4w=Y@pxMzc@Mek!%#XnG@VFy6q^V)$sWZ9TQ zR}>8n2GnjpN*RQ-R6)!-3_7znZw@d2{iXc?zs%A;gsjU!kZOpzz0SQ5W4~v$4=A4x z!#&zLrd61%>X*Mmb9uqj=OEt?h+2F=N)fSWm zL4*@rOVC7v^l0c9gucaA&}Rw_fx`85bw(7n^>g_wfIw?a!(d-*)#ZeG>(q|0svZ+% zVime?%F4zs!2b(@%u)9gu?W#5rXX%SdvudH98Fq`69=mw4n18LGu@D_bcLdCw^f9o z{YNy1CQ23xbX2o%%C*?jXZSL2pzo?sY5GoTZmAB4ONGM|4CM=UDFjaL?hTENr7X9T z24iutQK-h7JzKBruTSh%)&E_Jmb`$_Y7}QJEqGU;c&w^b8S!NUlgj_N_(to5j!uGF z!U7Hkq$!Ed6+3ozc7oA?w)ku#E>Do``Pnp<5> z7eT-H{YHO2;+o;A)$s5hLc ziz4QS1S*z*$!`g4Eh_jWP)MarhS~bXl)}ZpjgvexE6di-E5IHciOni0vtdbp;vE|u#5&FAc?~1Eg%?xNkEWmtdaGLZgurY)>hjl9q?^;{>TFYq88r@XDM^VLVu?I) z71wTpRT)MY8DdS85g>A(PK{@Trd=pEZEc0p!Rlm9E#RP9Up~qh21p|EF$E|D;h`C5 z0fqhMq3_jU&ci`CyG}_Brhpp47rXEqR($cE68opq$LCU4#MztG9nd%M?_}?Aa}*>Y znY!~<8;Uq#_lI!{58;$oMnF!oiXGv)H#5q~iMSEjIwUx@FrrDojjI}QR7%|gS=UEk zoj9UtjELO~{(^cHx?7j_yQ@MHml8W25c8y<8Vgl|u(g->V#ma{g{crc&^db}LA|kK zKnFLpx|XaHUkB{v(II%MeVq+3Gmt<2=lQ|c9IE*MTQZoKKI<_OK@b_FhQetMZEFbw z9W!RHUzh(^5DQX0ZZ)u$95k>*fYw0$C*)j1S~_qCpxCT<_$j+S-3yln#}E^igLbFH zEZ>qh0~U%ImTodUeb4zPwRkAtc}_28N{}xZR8`_yBbeQ)k%or?qS5x3+WQ2;*7zYJ zKNdspBk7U*)+%@+jf}>hU;J5Dj1gpsSp6^acjn}zZb@t_86dlE|IM>bBcy*BIg)D2 zIvhmdFuTI6ChYG|_*m&1OTF_}m!43GYua(o#*{&od9Ia20OG#fNIMvB9z$`O0H}uG zmmms<@59$etYgwkMvFm65YY_`5hIa*3TVe+YeMii6XC#dMteX_`TIXpPgxn1^i}_e z-1JD@=x`UW^vDj~XDU?N^37*~VFTzR%-P*h$V5lA$c1fis=^TNto((S3-;xD8bn zP0;P`@p^|q-kk$KXj&^KtU;O7V?r3=MIYyfhU#$uRU}ZQ4nN=fTjQni1Vm6?vDjZz zD3HoNR|oZj@R5Gq0Z-vKEJsT)rdI(j*xA{MoN1_Y5>f<^B8iOu=sU=^*@c7Bp+m)i zGE`l*%s93*8!rOsAu=V37dLc3N2Aq>L+s@fjHh4W`5E85?aC<^0C;iA;7z_Vl*-$@ z5|TsbMB4eYOEEp&rmJb~ZJj#!87y#o&SUD(rzIzZSCl~)kc>(wQvMrxw znseL~!7~hf->@J{VtwF(jX2qSZ+}0dc*d@tE=b)!&hy>!kfGKDNceKZn;pxL=}07@ z55?%DsU=Q<`*APR|(#jiN z{)AB$ZmtXe4xnT&^H5TLNkl&_7=Ae(X$d|7#_1TusbRljllad~q_HS92-!bpbR6Biv{Rt8HJ*ze&<=V$Xd z+!$V2q4PGm7u7o|c8zm}u?~!E1wjMi-+cn^`O??)TcW+V%!VytZ~H17rd(D&A3-lD zlvJp^2N40xM7>QWk}d#n>KuXhKHC#HZ zFw){-!=u@RR`5WCJ$DE9$`Hop(7Ys1e@Duvm47>QYT&&m#{oAP^e3|q>KK74@gC+{ z$Ov=;tCxO4p=w3`?~xPjvqG*z`+PbkRCUDJ0l+E0@wQyNG#Q_Y9i0UfE2;`gVnq@W z1T1@_+a194|6f-$E!MQX<}8y&U|?CXhv3+L^@Iw+@+NQ0aG1l&|v5`<3`q^!og_BmOv-j zGSkt_wKPimuZ+9%A4AF@Bq8P3Llb=p7kqM5=(#8~U5Z->JgM>OX_b~0FaLm3VEicS zKWVZu7WkWq*I(-7m(ctX_0||;N__=_&r(g;-+l*_J})n}0yN~kI{Y`SkIx+`B3Mz; zVG<{3{PqG%C()7~n<4TxxoefH4UfWy>Rw22fRx^MJ@h3P8GQEO&D3Y9-T+*P=Vm2n z)c-Y6uAOY2m6LNQ{Oqmyc5>b!b-1*9_A};$va;;#>@IVP{>?+N>x;)hBaqKXNSCA_ zF%yu5fr-q|1zf>ttTa9HA%vnY%+5kMWmCl(7N{Ul3x0Bnc1)8rJd{fPcclYkU`L(p zv?+fye8>Xv58;tn*}9@p^8tBPDDHVnb4Fp zK+kv%TwQn7A^X~c0>kmcGEAiiVajvGA|jC^FputKD;0!7PXi};zoQN%3_x{|_>AoK z6BfbmMk+RHq3IqeO<3&_e_24M8$S2=PNXFkL_;PlkP4XDM}NyJG6jv8`TdS~ZtoClvIt^95+a#FFM3Fmf`@fS zWdJQWSoT~2{dE`aeex8C#3J;zSpe+?+}+3eQB~vh^^mS{0*;3yh1|v(_|zD%B`v$g zm=4<2Kfprau!=23a(nw0%;!*L&xHkV9IX14zL)Sc!WpNNz~9t;)#GvsXOQ=Z(ldB2 z#e_}lGo&YjuH7C793b8i~_6J zg8_FNTUx&8qP}+e5-S+}m>kF9DN6qRix?vjC>&nf&^6*LG775ZXKnvNSYVY-*(VV% zr43K_=Vl`*HEiVmTv{w36gPf0;oclYNcZ7`-1-vGB1a7x!w>JrEWyw%ao|V?&R8kV zpr1&-Zjxd7ET?2=oh#}78ck!CpPHL9D$i>c`vP+v5VKQOm+g`p=)u7~q0 zRgN)Nab-8?SRB+EvK|CvyIM!E&;wX%!@DCE3~t=7hSe0zBFdQj zX#87tZ)w0TK0+5F%o&LYp3d%_0Ll0*0Y+;H0=A^vsa!wqS%1Np3#>6oXU6*}qKHj} zGC?EYda)b%#V zh01at28Jn;W zn8LW$zQy{Su7rwR9mGHm_By{EEMwehnK?NRBJ!;$TBKGo1I!0r%5^jtOOB3=`25@B zbD-tF>!g9o`h>hopyzV_X{3^}YkD#4BH)mo&KwvwxrT)VhJBrd#?r=#tn5c7=3U>b znwzm--{)-}bnou6AFSJhbzh134L}*6KcnZ&!T26Fb@d`-b8{1Z3s}vdhYn^0h{hh* zYS8C^3OdrzCCF`uX&%yAjPjB08k2EeD8je*gX-arx?_T>`cTy8i{MiNO#IeU2<-ts z*WzNa0?vn@lzl6Bx1TtyFe6=1<&SN-Exiq`W#VR<*ZEwf6%a!a0!b0oQI2)^Vm@_` z0YHA&16$2Rp$>b1ZB>fDd$#Y%E)<5tr%%0TPsaPeGsVWCnPeysDngFDuR!0gx z_pMThd%{M&%fnT-O}6ndP#bdmR^dDA9#^Pdnw3Rmqv<)f4H9~yaLaTsIK64qXG;RZ zS)%#f&}mg$<8d&M@PFx&_Nzn`P}u#%>~z$xp9^-qiR#E#Y^X*ntphk&mcbHrT$e`F zXWT~ay2l5@zkBQ)PAv{OzkK=lFE87>2Po|xNqluQ(2i&9=3MT?f060y0%}w?>t!b_ z&E9Y0p@$J9fXT}!@=OR7zuji>LKAsu0H6JhLoT=UY)q%g%}b?cbmw zT=9t!idu`*;l!Hoe_-y+$=V}Rx4Ud zjaS6nt)1N?LLzU>+f$UD5P1^FP|Zs~5j|+)(W!0nQu+b6hOV>XhSR$@E4cknVVc8Wsk{?!5 zQy=Nf?M^}nGSWqSm6Na+vDgy|T%B1dL>S<-Ft{r5ZdOBCAe30jbC?d^CsK>Ps^%R*3}*%)`s$J zA?J}zvRJZbnZEfV-tRcslmFbYi|8qkNFg>b+6_9#OjaQsoE6brL}-nkQTEN>K6YL- zrk9But-p2VtzPnsVeW;S=YFbzCJFV}n+?wmQL^KU2Wfke*QVPkPHhgBiNpVHIX( z>|2-nqXo~axI--RN}?gg-th9>FEso_0>%z86deifj?0y{TffBE+fMvj)vnu5Vhqsz zPe_;TI=%)z)AGFyRvYagaWCxuq@T|SI(z)yBmH!A2*to`7edNLJlEO8%a)&)@_v75 zlQoX{y|JG0+3Uc|A5SN$Cy?(eO-^;Ap4{6e95hHeT2{j?*m+~9_l?g}ng?MJJala) zcJa>~`dHVqb4cCWR7YvKkF)Yb8M5UW5qt6?3RuLN=^|^+Z#gWm&*jNOS?hA#UvIlF z62rJcB7WJwA8TC2Va-E@V~`c3`(-ySqwjKP>8@U?i_yQ;Tmuua-{paW(uDe|Axv@D zmY?4LK&h-l|8&I_S=jw0iV-n*Cr=eCg=6FRn{TIuqd&UN(Begs*L29{`?R;YZi)_< zs6y{vlHbp1HjHPz-tZpD=OCu^%djBPn|*fDPHmOSxwe1*o%{c^0MD@J9~SBP^+_6{ z2&>QrJRY38=;%?hN$oW96OVqZXkrdMu=>%BeLhPB6+83D$C!xzNx*B#G8XAUV zWWk|`{IF5L6`;K;^mX1%NmiKLEYsgNED?V;hJi{Y(3nW;{qn~U=lubOF>fBKOo4g) zwupvcS*-D`H^%)MNG3wI*!dM@Wk7Q6?&A2%St`lRvIu2=x^*r?%jM(y+I46~;rFdX z%BuR*#%(QUU(D%e7IzF^rO)8`0)arc@#w1%!a;w8gxp#gadlGPPUl9|H{ay)UO;o3 zs@bRgI;H&rpbPrRl0imhm*VR7I3$>@AXQ(ATGEqOpY4@4)ko4pMa-`V|Dy6ExVphG zoIEd2Qh5*gu{77GPlG}vniDs^x6zS22OgT)F}m?LChtB!9uYFV+>H~2CxX_)*_1v$ zjeQU+Zwef)3mm85QG+~#e)O;0_kJiqpDotKX!N~vFwDg@Q$7L0j&|eo8b2^PFjDKV z)35iKal^3gOo?fQ^>9#7KtMoDY-xJhXw4+B9y&^VS8tMt04)So5dw?6MPqPEj%J0N zuIRjp(&DjpE!9Y@vUCQL&EH;}#=RV>oZ9JkRV*k!-K)qh`sLBq8xI*8hDe&vp8D+# zm<1pa^PHl%5-P{6s+eZh`XlM+ZL3Hr(pm;D{d%fB#HNVGMbd!*b34c`Z(0&S)o~yU z5yqBGa<~v^Q8^_fCiaYxD9jD|4k2nP_Hz=ahi@!TqFX;Jn8@Q>B!4H{i}q8fgv)00+&hc=vL##s6Y zlkMfVA$;}~*J9*LtnS9@qV1$NSW~nIECccGZeLRcnB%bx1gM4P9QdU<$>!?rMPc+g zw!VxZ2p|OdH4FjzGELH>y1702R#NuS?Bfv57_4%Pyv4e`=jFld_qUJ&5@3!t%a>@% z2i7PEMdC8dd*$5b`6q$rg>QOoC4e|7BYHi))$3qEu3p_8x95`mKIE0zG zWAW&hYJx+gVkhh#6o($?T%YCi*GOwu+Rg!84L~3~3ZxVZEvKkXd4IV z1ax|0gQOZCH_RZO5u8v4Q_zSJCx)YI0T~U(i5n)}JGg zJi{jL-3|sZe4!1eaQ}$#C?7=>WWs?9t9$~MyXfd>@dilY4&8@rLM<)m(zczGbB#~{ zAdiH_>|WomqFu}be1Lxgz+^7YA-gEV(z$i#1Z=Wh;&j+(ZP6BY-X{ciXn**MeuDHp zR*9K4oJW~>yBz?*pd@JTR$`{=19ExW+0)G}rPmuGMu8DC#fzHc83yhVbn%CyVu(EP z*%rXr&yV3S33g8fI0FtL0gNmKZ;}VKkne(>Dm1|zH@FA&8YH*rI=FNtXkS!yNo2kF zNXjfiavMYs;4gM|CV^N26%~r8UWma!nFze=eQ`=inBb8OFXtMx6%Bz`SBd!$?m6A_ zO@QfmGXORZTY|#fSWy&GNhM^-T6!cwG;9e1DB$fNkDW)Nz{lkdr63Tc5fjMilR{<{ zXa9EK9vPdu;&2LWhUirkR@LxgMBb-jp%{+l3`LKV`vK4vl&(`m(U%C09s&&q|J!Kg z$YtmhhUydGY06ryZWJl(gOYyVKyqyT8Tw*w(E({m$3n;pT+`dr-^e7U8y_TA;cTj^ z280!EX;pn69@_$hcf*-9Pik}zfDp<)t#XjOg$no6x`JSTg;>IovM|e2cECnpW=@G?$JZ&e+r~?H>xa z*VY#7oI|V4Y0w&f7%~f17c}OuL_lTIh6nW*vR#mSbvuyfvP9&wKvk#2Wu6jP!jY&q4`v^41q- z@>{w6UAbuC{suT<1z27U#a5Ffzs3*l^bib^4=Bg_#A2MHv*1c4G_*I5uKmeS2U7Py&E!UcP!qa&<)053=(_&26Q zwgAM3xd8Bru#+<=#FD1>|DnMXiuXBLV}g_>*laxvCwf=lXv(f0`z`iBD7r*$TI}|7 z$!NmC8YoGMzGpl4Tjk_7xS=2$ zh{gj937AeB-s&A=Zm8J;ozu-8&prr1$m?4In?|z&f`WfysXlOYa{>?jSx-jKRM_xE z3FpqmSJ_4*v9uJ9Ijxra*96;8;;L;1R+t7|b! zM8yuo7;Q956f7qKgYwN^2_YRZ4we{R?Yr2(e;?kWlVhFPa|P<<%d;>ect2SprN}7G zym%<+%njO?UfJg5K7kY~nE)BMGa3`87zdNl4rJBIXquDbNee)$AN6ov$Y@Wk?9 ze-d0deU7P}V_+;%N6|D{O+sX1MV6=2yu%evvof z4Rv+Q;xYBI&)}|!=7dGxRy2`ip~8^F1QSdVKf~8C@MeN5J!P zVD@fLzF|uWaNEhpdX6PnKDXB!4_Ya(C0kj^^H{PMf_g^90)#0!A2r!vW1nWj)JDON zB)OF-R-}GQ8>bNmz7hnANV}k2st1Z$Tz%e-k2-tB0qeA|r2a$9T!ZO#&8U!T`L2-te|_q`cDs%JhY< zt^i1;#l$uzLX1x@uWWeB6G7}l9vxHwpHfJL24ejyeOiQg#A9=J!H~9JBrA`QvYCi( z&AY+Z21Ja2+`$JESIWCKAK9-tW55h4u!pI{UXij8s+s>OEjX{xKe}}n9M>^@5N*v_ zrce|QC1Q(<8#|m%-yVx2NW;R7I7`|S8$ooiR&)w!oo+d!_3iEO3-)M}2Ht5yQ+)$! znv_9EL5IUVm+Dlvo_=-pamj>rl7jiGS8xoSUi!Rke~Tm%Q9mk;b)SU3$V1zK2i74V zZV=&n^mYV}{IqxiGWiF=-Nu>wX5K?nr4ap*Tb`tzY+j_!%K2SY;LPwXYUdy{dd@&gQ7K%%fsa9LmOFJ4gN=azB13Ue^HaqXE zpojThDPHiBPX@dV&c-&C)aJwq+zwQK&Xxq?1E(!eu=7Xv|4b&+)qCMqy?cCsP0i;D zIt$35**GM82rxB+b6)5#cHCKvl+R21_dgEQ5UUmz~kyw-B$}=Xkp;^k+jT z6}Gc`WO8^8jry6)qBNt7)GIXy;XxYVxx;4M@MFNf8!p|&qkQY@Ps7`fUIPTauyFaa zyz-R3m(p**YplTipVF5B$FLj-i@7>DKX3Y_En!SXZEdOhs)-}=?~qGvP*3%p-;Quq zRNkY0|74Yp#;u>!-2ZZR%b%^jrUr7KK@wc;qT#dfO$ zy1W&A?!s*i4WO+voA-Aa*#uHGXtHL8oo0P&t7OI|KBV^3r%$b|P=^cYejS{&Tmc}w z!_jMNSlWj?`Xcr1_c=b?IBS2~uSMpe(tP#WEIcfp=m*1PXkkWC5mU;bObFg%$rLzZ z!lfl&AsjmX3)l)z4&)10Xoj;&ntsK#8KkjYU6PPH&hs^!v6HA@zaCf9*3!!V4}IlS zc?O=@xjZk`SIkv^8b_l1XdOO;0&X`ghftOZTmmvu(rjT5CkKb+!?XV!cF=)o=zS0@ zt+UPo>NgNAwzc6&3q2(o9yPy@sh5nN@-#ez7E0YRG{)X(KuzVby58cNl}- zNti2Ad}d*3X|qT!(Y5*E1J;b7@mgoWs08u0V&P1Fpt5=-x*WG$o86iJoO#0|8-~)2 zZ566#G41|~6OFaZv3X(JIo+%X?4_=aU!*tVewe*i%j{0~jaOihq^lSLxwbM#23p*-XrCB3V6v)5u-8Y zx+HBNIR@cs97VRudmzHz^{o0{ZTdoXcz8tM#Jd9+hWqalXETk03Ig0D-_3A@dg8=- zKE7WH9)Ng&@SCp2?mxc^9h(YAFf@Cgm-nQ{9*4fb4;nS&7BDQrezUU0_E6d4zkp*n zO<8|`XEFh4-P4dW=<{5BlBVR421UtAxMgv2Ea(Ui-VCmIkP-4w;Of6fHW&wmJBtWp z9l>6xUK$%iEAV&1`tyPvRCeqd*Z~dznTJR(X$gLxc@rgo?0~4sKMR!6ordO5jCP3d z2p4Uq9kRY!tS6fvB>DTB)3kXUVsI%rl<0k2?J%=Z->pmp;Q=5L*Ye{}Q`vL;ycJy^ z86s*fDbd`N5kN%-O=gRD)d8!kQ1n~Sfu3JK@@rJWEqeBHdu{O0k;m)|}O-UR`MU=~m^XpS#Q6Np79@s5WpZDx#y zHX4LvpOhlxo@OgTWX-~iBTFWDLb{kvU1A+yC(pvuScE+rp#zUBhg5#64 zusx6=5QPU7>O=pM93wYH6o{^s5&C+1U5xgh^%PhYUYvfnL6}s7VjR}B&Oi{oyVnzj zLdl21aPGFV0#a|yp~f-^7M)75N39s6PU{y)75xGMVDHJ+cVO|eV5B1+Ygj>f{PE}o zk8~LoAY=+Xn}=%W3h!>%q+yGx$;qR2Xi8U9RD?GS7D|hf>ARHVY`?MiVDRv4RrwK~ zU$o)?iEb-B%N$S>PgsMno)N6AGLvjYFlgK%VSEG0gY8SO*%k?p)eH)jHi4H|y5MYe zI>pJU)R$JaxvHuPnpvBh=K$o1>V+%Fk<$IQ+Y|IxD4Fn6lRS=n@YD+ORAn)H&ZJ$< zTVHM(!;gTey`$!X4m(tLz|!_qbKkIoJWSg$ zncy+!U#L2u#-wxZm3QTz@nmtUGU{t%Gy$W8+*cg~16auY#Ual>NHXs%pAWtluBc3Q zU$AsPfm%#2#F|`|mY{##N!rnfrJ*dKb6MN)t<1-OzO|XTIR;#%g`p+WF}cxd_IEf# zI_$mElp|)h2-T2&IW?9T0&4Q$!Go$>%#coGT=N^?9|hh5b#U@u9>n-EHio!tjDpB5 z62FcA1oC|fN+xUANs*tktbsNgp6ywQc~6*DDDgnN3AxACpV>A_9TBMThq7k5R^dx` z!N8l7B<-M|+wUVxApPfMi+f%oy+G4;Ws=)G`0KQy`# zzpLg<`EY@O(#N|#OrCoFMR)8D-42+C+sDmcH8G%8+4Mg&BX4TQx>QN@Y-mX}shLx~ z1H(L|+^==?3lm}Ww2yif`8-FFb8@G=OT$f0NY0+c>E=Vy9&6?1wL&p~>*$SsS+XTc z^EHH#W}+dj{G(l+#nW}^j>CeIMH381{~Byb&Qxg)uHA~7Qv_DT7LB5z*`SImr`?Hj zigqpSk3QnJi4Qvigr@oK`BC7rK2_7>kyCQm_G4h(Wpq`1NX3^w8B1ZLOG+`|EXbtv znp_qb2oVw7U`b$U@Sf$G3Qu9)Uj;G}-PD+5X^+cMn!BPVC}=BY)Qn zI?{!chuPme75w^~jyVGJUQtBMh*;^3P`sh|OfXF9QUEHH`is1slk=aKugHK_Wi_qJNgTf6jDY~HKKvA>8N$XQ~JX8LOwluR_5L_c{SC*a5)VGRCaF5A&bUbx0t zkg+oO5=8KWlT}S9F^es@blAshb0f#M!K(gMp#4bYguy7!@I712heCN$pJ?vB&n|J9 zV(YwI>H4eBYN#l=9r=~ ziV0rTQ_u=?b{~T$9nV6A*r=EpR&wUwj#>YLQy%|@=wP@n7 z-`A>04W4^dhRUrDVD21+7zCXB4i6&?w0X|EMoMfL@3|Q^|$#Wf=<$4>iIBXNPGARbuIfl8>Mazn;svn+2@2`=+Cxj z9CWCA<~EBBwW}^CLk^y)+qa5eIJHM`GV@8i%`{YbX5?{dAeeE56pyT!g>+5QXkuv_F1mXsC%GzKY@mR# zb!T#-*;k%QVIp+Q)OuivCOn&|edv>{Vg0X=Gx64M?`OXzD<57TkFZpiG#z62n`XKh zu)fKOslZ-19`bM!K3qufMOrw#MZ4Cyjf8(+BowRZKBfc_vw>^6kVPm-==NQvkV$*J z4VRQ33G=uMV@DmO;RY-w8x{BVkV@i!S7m>4fyP<|hVSsro5iIOua49EDl~YQ}&}3U_*WK0Y%nYJ; zzErU&O9|T^{j`f?^V7?@gVsgkw?i5UVS2H=l$?N9z8v_5oo#PB7KVQ`oeaUfvYKIe zoNRD@X#9)e^5Id;`!A_^3L~i>VnoqhG~T#5m{p6J;ZhX`rj8ekg|*e6pYPyL>9%Y= zQ(z78grWkn>%)Hp@k8QoIqF{uJR16C_^fZAo3YgRTguMl=*LBK95uJfPk%Iu&=Y>8 zHLvk|@eqtP+nu?mZ_>FrCb_kpx1z;#H+XLz_dZRr5eu5Xv8MTskqLiC9r$joX}|Ce zD9GjI+y%Qp_r~^i`*aIVax^lGy!_^D{?TG(*j-?xSUc!6`rVB$24zg!?RdlQ>R-rP z^$h-Ds08mW%vR2VIV$gg^#tT`zsGL+i53jZMtNz)@`?{rD^uSC;bHw5hFnXYS?kY> zEgn)H+_bbunmjnBf{+Z84W4_@B-%j%5lHop^i@wv3oMVo1OBQXw2)A;z)lKw=q#1c zB811e1HZ6<0GN8mOMZU5So|w4NY4RSdp}56B@gA(b#>~Rduu*`nBe^Q^BP{QbkPSWsvwyH;=g#q;2xx?`sFgQ< z^ltgL1Y~1_6Hk6J<+$-3bxpIhcC*Wxh}kP{GEqZDBEo9VQ9knpS|@iYPJ z#jwENd)N_&@Jm`G2ulOEBnTKW9vPS{*ts2Evy%NdZ_H_0mEJtt6)vja{{aT8|-z7*U{ zK7#G%JyXAp>fJjg1W`|J)vUv9L6XvFVotN#A{* z+spAezZLX22F$k3pzH)nJ5WZZn)CVc#6*&M1YtZKiZtPCsK|hL@zKo(r0@Lh=p@V3 zcL4H=L@@e>gM$Oa=rHQp@S>`sc)5w@WZ%!s&JOs|5#v@}7P^D!S-kp>Th#lW0;`cv zlDhK=tTPIwpnjjfgXd{D<}_6z>Zg27l)2C}9N^ubL?fHO?K2(K&U14X+`gOszN`8W zcp#6M9uwz*H*-2!A;ti2$KJO;Hb*w0|40nKB_Ie;)FlmiV4ySwOP9)KL>#awZt5xg zHo!3tkG)^?_sad{Wh35Ny_$i>C*nv%uHT0$S@!z6qtj+&Sh|E9_{YC zHY1q=l;Fv-dFT<-SKHVKMkK4GaDpewK*>4Yq(l*WVj@m+SV2|miS`$Yv?|@eCuGS} zHen5<3JwPBJ_!R@hoGDXt3s{0Zl_T!Z{V5tVhy`$WBbJ)#qEE)r|S8#TeBsm$KuRz}J{-Sg>(WA~I-Ip(uEwqR5#OW1Z&lcL> zy3FwoIB?%R_S_yfpE=1d`3)_UYX>+Os6JmlfX4IF5hU+_YAxo1>aFi9>bTuTuW!vF zg|If82veh?AOC(R@(iD3^zXcoZ^2p`&0$1ye*>S5;{*AhlbA72UrFHK%`?@^aX1H( zq+YUDUp-j`nBA>RFqgHr^Sin;D5-pw?k*~SNbw=mos9g!QhfRXbKIH)?v%kN9X7n+vQwg=OtY(p9Rp0iMx5de-zUgA zXwigic!W-0g&6r>ctfaBI$A1|BINbwNWbX`sxTWT;kaC24G zV(9c7c-4gp$i~wAx60MK_~+l{r`JR|bRQ_6n7@6Sg>g1xlZ>H2q4g$+SFQAs1 znmk=x?%WQ}7JbV$oU=vwF%f)(w6!mi;Zoyt6R>9k{RVpwu*5UTT_6BftJaC^F@eM5 zd=WwAJoHJuwB_x;=d3c%R^Z-OVYOXfc|l!BJ)j%F=_aHXYp~D0BYSjyDXOv$Yol-z zN6HRx8lNJP$%+u$fcDB(>`nlsyK@X!i?HO82+w{A*}vmJk4x|4)U`A>gEKC(A*SON z#P=@w5vRAm-Kp(H!=vdHIsj9?Pp<*pMD=dc{Tt+yM zns6g+Ca3<{;)9*ePY?!F94C3z7?g7U`SV9a6r}XPn*8>NzXrJ8;RB(bhC)=IsW@LO z-*5hJtx&>Ia-^vM7i;&vp9)qxl#9VWk=ulfE3U^<1h*G90QN|gN}fmyK|gB0RcHo) zTf`L%Z4+=hW(B8FsIciJ0aK#M1~#PaA}H#lvP>grxp#msy|lWz`my}Sf%{g6qQ)6O zI*@E{$ul`M75;xTU3WZ}d;hkwvqRadB!rx@Lc<6lNp_T3l4Li`Lb6xL43$m8-XRI0 zl4S45mKncG&+FIg{PlFslfK{k{(Ro=Yu+-58C!58C@Zh1Kt4dYDxzI{1Ocbj3J$); z!G?tq<&aRL7>gEHRfTZlAOIXZ7-s(d%e3=J8Hr>M`*;MiAUzG(F{{4X;dl?caWHd6 zM@IvKsibKZdjeSK&I)oT5c2Si^hre_EUBiZ@S_$GBW?$2B*;Uk8=4;;P6J->12?cC z>J6ELI@-8&kJpOzYG$0rs7|C%uF z)tl5?VmsWhY39Lnt6DmKb%`#FU~kWxgt^;S_p}>U_YWnXwN{XlLT_HlxSpOK!tWJ_ zypmF8ju>4bR+fuwCRp85bw?)gpl+sJ-$gfp=R|3d%;PXlE-o^>CLY|Ep(}Y{Paowb z(EsFT&%Ca;8`Pa9I`9jI)D9cD`x~gHotvFyU?4u@e)Hzu1RQ^EmHlw;?jB-CKi?R0 zbO+zHzwiJA0I-}2?Xyrmo*rjQQ@ z7gbJ`)$2enAhWo?A{r0J7RsleT}(60$5)L9owE)|kLC-o-S*9Jno~%yQiGKPRB|c8 zE1YPBU|U3%pY@>5KTPT(D&#aipZ+cBK&+e*K` z66|z;KdDbdN@l(GU^9W?xT7>FEoqXafqXLU2{bVt`XimFxlYyJtuGACgg;|ME8O4+ z%@kxxQ>v5?M>o;&RGsT`oS;(7#K@TqUv$A*ziYg7<2t?6=*KR8Cog0^QLFq)rs18p zfz>2^BVtY^WGfO?s{(zMx2NkBUVAJE_CL5YDh5P#*-Z|V#&JRC-(XXWz6g=%@=&`fnhDm6yC{T zX~i1<=`^CnkK{5EhGjuR0Tk)hEv+A|wH8pe#ofY5gBTB3j9bRQ{ML(!?Oli>dUonF zxhw$9<<0x&C}RVDKx*dpK28IJC=MeRb8{Zf(8yaw<>f~fYu>P31XPBe$yuh29Css= z^3AAYDTrzB6AUbb6a4HSl0{>whHe_@!8tB14J{)EA)I$MN1C|< zxD3vC2Op%twQwks;$Y4wVbMWkwOnf}G5DGJrp!>Trqnn@xAR~BE#c& zDx&>WvgDu6GTdr3zL3)CaiaQYq4vfvY0hoi`=N8`Dhq#H6`^zcME`{6yUYdLJL((H zjGqKv`yrN;u4WkXwzBf(ZnM&CPfwiAbz55s#ZkLGhYF2O zUhr!|+B1G69ZClzrh8308$*D6Uw&X}3iTcQB1z|IQv`17kL?)_mc%*>}rEP6yo zA)!^G2eTG4*@Ith-SG8@#@^lhBcdIQt1%iUEm9WwXQi5q9jXA{Rv5Y7KaliDp`gd^ z*E@O>Asn|m$b{0pie~}8-oE{>df%3pIle&R)BJ;=eqd{_k{Lgmv}$yUgq?2AFxqU!_C0bJ$kF*&|>3a{A~ z3Z`5si~Kyj0;NUGEqoUxHg*fSK(wASw-2tDw{-rMazfYD3n97j#XuCECq4jT< z&E;j?!yqCGsv8F8Rd@K8@!CvVd{2Kf zxk307$>D#@CM&NylyFaj*rp~!QE`b?(O zov(Btu*S)QDBP$Y5b_WozR3|b0Q#PqK7#YyW@F!3FYQJJ&QN5W3ijHwox5bc05AA- zq&ec(aE$rteloT^6o0iiH$R`4A^476|L?}+4|l&|a4enq7Lcnqy|b0PgVe98^>Maz zG_eW`Yk!zGcwfqCa%kL-s=Hu1pinon%NAld|N1E>!#`)B!8R}m|3&o5iqVb!9m$xz z8`Zk<)zjk75fShPM27HT0oR{Fu2J;3@(^33mf^62bul+L7nB%;0~YUWSL1o(RY_Pv z3vJ`^P(*Nc7!vG65&ak1#7N^G8xv#ox~HMxp^gw8xmP@H$!9HNA1c09@8QZr?#7vO zq%ln+(z3F!VreDGfqy98h#5FX^W1uyVo9&#kcRrYpXA)}C{G!QN8|YxA+-CeExQh} zuQz?hK#?2iK)E=0?jcAv3`(Oxn0jC7)^KEI&43)Zc{7+qdUvEB!`~KiUn;rSmU9v2 z{0trU4L*uq!4JYOqZChS(D$)1L-Hm!?SB3|ks{R(M-NSQ+DHm=`x->1B3w_^3cd4) z1_b549tukn)FimbQRkP`fRy*I*L}Ph$Vt``9imcKEJ0ngTHOJQo!$1^CDJk_pQlv% z`}IfWXL;#6Z|WD41k1MF6iM9fBW zJjD1t+950|y7)ZlN1MXA*h8?yfzVLV!^R5bd35x?&QtrvSk!C1bh7k?u|C`@Ra>GG zzuzTFiYj(=O#;$g4Jf9r>vSIzr`_A2dFZw=K;1)tH5EI~u)QC%; z&TiBFuCq97tCgVxwdI6R(MQSimCQ+AdrO1rPQFU9ekITmwS#0Sc-e{e!7AB#F_~cz z7vFw@s@*!}ii#(+pPgR^AI&>f^sAF^n4qmzA>e^j!$_KXj{U`9k z4ULczd1KJGNWCW|!C%-XvRnUHX?hENV7i&eU<)_&!LYk3MC+Db?doxsdM!6nDjY#t zw#7%7E+tl*Y1T5~nW!FU^62tql-d{^mkV~htpARapu6^El!;J{$k*#%=T?!d!SPGG z$1ckI_Q~gI3gk(=&7Ay)PWNdn@)!E_ENE?r>NE%WwiRtEOIXGBedjD!**^E}nSBIr zhW<&pJPUENz^f}Ciy&+=kQrAZQsoW*9Q%PLXDwDJ>?$R$S zlIt&aucx_qTU0XHZfwn>EOu<@`G%j>?7tIjCCR;>!Ca@K~116Nxv=e9;E!y$eJwQrlmx16@NYmUr$uKjJ_;(uQg zhQ^J!*J_-hDt;O;9`RX_BtAhQUQ{n&qSqkS|bbC>}b8WIidR>(^!;`=g z|I<`>3ax|V76aK1%a{;Ah;Yuc-)=pZS=s#^`J`FtHrIxyo~gb&L5|VYEl(16Nq~R< zY)*A}Y?kv>#-AOY54j@0J+wsYl>TnITK%5>&e4&B7T(*EwHdl6?$Z}HZsp$HB&a+V zx{!3K?7w#HTkkcpD!1~hq@0FR%GCeKBg9Q}(y16-7ZU zo_s7m%B1?}SMCko?nqam>Gd|eJe01LU$h3oHl8mpx!vgW7tSfVI~C`0%&Am~EA@Gn zUlrxXQgAtAEBl0|g5}14a!awQVtQ(&J6jV9I}Gjj)5#*(X;%p5UXRdipB@fAd5G9I z6@Kd<`+m*g4Sdt)lkqWHCYBy~G$}3!=l(0bL6~qQrabeH=0fJIzJPkR6`w0ts*x}= zJT0%tV%1!l%#WXcZI}y13Nra$Y3zJ$V0-&1rJr>-;}ic5M!AQsP8}A{Ai{*WgIWie z4Me%ER`SU&lH<(J9Bp<@RpvC~dt8;<`zFT!NhHS}2QB&J%Lmel`AAxPOx|Zt(J@n# z_FU#}Qzs)fB+-vSn=O>_*pD{KKlhO}ilapbsYiUs^$U|CK)uY~7o{}mrDX}T;ZBIIR%uddxrD*w&W^$x3 zn+@AH^xW2p9HU0}dz%$o^5r{%i5KF zt+BSHcw`I(H|7jv6D`$sy#6@mh(A8rb9;R^2ZO+fAmaD^6GBD}MZ(ns19PjPAt7SJ zwt=VHZ)op1D}G(_fWr0~U*)A}9VoIP${e{p^{Y9$(sT~dpPwd{upV8igh3rs(ZNW_`KnCH z`}zk4cK*(JR=;_ZP78hz@ETlyj-^P?t6m>ZKnzGbgf0u6oe?9IXnhr9^{Y28`@V&r zb6zmMSVTkw*E;@S7Z>HtF0Kb_glktro3J$CN!~$B$U{Orbx0FF5Io36!0xl<&yJK@ zB#30t6s%B8bx*w5;pacl1M5CD>0}EGR6TZ(4F9l%Ty?n?Po3xnp(jS1Rp$5>>{Vrgqih1V1X5Wqd?ryv7KUGL@KBuOZqqx!ov6 z>B6xWv8KgcQ*ZRLYtM;C*yIDwtzCA^*>~XNuh)|8XaAn{b}luxN$F%^y%zN60{IU` zZio-hwmN;dAWY$8uAM8c46%4tw$9_-XidXhWtN=>)uE3cyG=UDV6gdqZfQf}U4UYn z8e=5(j9fj)^zZ3eHEqbsov!;B{aZW2H82A~!U*ku(S>@7HxpzO`OxY1{sxIdP9n#t zmETRQYhuR!bPh8S#=(ZGAUi@b4Ki&p=hH#FhDcG!W3Gj^%F=$4{W1NS`Um(v7TIVu zZ~a~yNxa*TXC-ehAL76Ax(7X+fF5UNFtVdGA&{d$^xh6>v@6&I&-8mGmXV~5h+;uLOmh6NFTPXCE_A?%e9vsq(JIJ!oraTIt>6WfV!sNs zizYpw!{$8=Sq#E02+n*MzAMLhT%VinrCv6YI`s_=PfOzum&0pGsHbt|WS1mDk|37; z`ICh72DT%;jOs9|QIP%zBPvaE^=`cVfy3gvIamyXKTtZrg=zBa)@INy z$r*|!Cehe4BaMlOKUJOcG~xZIzLM+fNiguiL{q`72MKn)ZADNpvrs;?>G7AtkF3~= zmw|(e3q%BzKRzyt*gUn*Ye?$;z(bwi4cR>bTtAfl=5OM|4N`#3dz47kmM1ZE^xW5 zXbUnCBz+^i9rZi(v}mVMr{sArzvUy&3y-{LNr)s@$o3-lxVtFlEN>w?s!w{8lT~aE z4`01;;|N1w!Cq~f8m6dl@_X+547;pJH2v!{_oBb`eI48SN9_&oI+;h)ipJiUvqGYW zq7Yh8P`4KlWdA7crwQ_Jc_q1xgK&UVCeESMKr19?Fh$A<{` zINv9c_&P+l7prc<>(g_LcH^_OY;h_DyT`-l7!*h$HTIR$QzOL_fWADIxtsV zQNiTzYt9N$K1mKNSY>4gf*J-7`zd4S@9bnat=jjbvg-okj?DjfR07|1O%xp+1Fej4 z{s42qGgcYnYpji6j!H_Nwpu|rsQ()~uTV5Iv)Bc`_5arbFvF)(TU-0<*RSDW-nrAR zz-y{E`%o&)BuYQHCu~5*C_VWvFbo(0hJ%j6jg0If8_Bw2ybV!0DUR2kBe09)5^uQJ z`6VC^5^*&_u!s^YTYjx>eweB?ONsjBQtoloig_;&ntcgH9m@L<)9j#I zX&QU@r^i_Yjt|P>GuJ0(6~JGmLi4b|0};!+Ls2?DpK1sSSj=_O-UkRdJa~W{Vie#LN=8@N2c~?e^i(rowp;BZ3~@T+cM4mioYxNNQt<6*oN6? z`GE;xi~x%bDLWE;R-qVxwjcg3gkWm8aH4Gw&m7_)QHcBXy4P!@qGo=*j<3{GV*eAz zs%;D@^2EvQmqlX{pS_K^d#|M0E~GdCz8shX>48C_{=Vb^gchClFeBlmCFi0Hh|s|Y zYHVx_7l*rNiYBVlV_Q(#2bVD*`K|ROywMdOagH)9UwYR^?Z#+IIoh+bvVyeI#>Pi8 zyt;+>H{qEGYQSp-$?e%XP*){N%1Mcdc;O*fm|j8hsucU5T@n4$D^er#8Hk0GqCa>o zL&uxvr*9(F2JEQKb#-+gKYl#g7rwZAU=G)SvPB>L^3=s`T(a=upt!xPtZp?eO5*%6 z&d}%e!?9-!kWsaE=vpCblB*6e1OU|oWHYR@M;qy^?L0lJyQlG(BX{D&3|uWFhsLEP zT&TV2o;*UrDLW1RBjZaywe4E*ilGN$brVWJmMizXLm>eEP?YoV;hq|i1$&XBN1-{d zqhMwHq_n{+)aydXW^Qe*5=IEyHL?%fOmJJxT>LVRQ_V*5W8^6!Xq`(^rSc@?POfQ@ zlW6mRESX***(1wDx!3=UqJzG`7osTId3ER2L`ws~BN2CBNrv)v3bNC{rZs33t@CmR zhL5f({g^#P(*4q`X}gFXjt@K{_$Uxko{$;1Nxr5{|MjX9(Kbxxex4CFuy`S~8|YlP zoxMHu`=5iJ%HQU$J)tIWk+RTUhoc?z$MWu{co{J;Uj&9i>I$&dD;)*NW= zi!Q?2!TEUy+(OPgkqu=E%(v zMz7%h=@p<|%q%QJ)gawisi8uN3@d`JA_`Bt*Nm~LDZ0EAxd|+e1W-!ktMcQwz=$jq z(2rnPP$U63Zl^7-^0*;Q=eo7^glqLC2NA<6u{MM2x}&B0()t!SOS@Ac7QjB*#x0y> zI2g3R<M2)w<_ZSO-9-fV$uxwsd^ zJFMwY&C-$6UEtFIfQ^#~z}dpW0)R85J!n5NKCYXe%0sFc(dg{pP*7UBJMxkg*kzSh zsBtO(&qhe3uBxKs{k?oEpQMp|Y4WnntOWHnobK1IU6W#GsC7&yWp$}r#@(E>3Rcmv zG$)GpJJsgrTSLKc2PbM$J*H%*)O&cXZ(=DoCkOdvJhTMEMZ#3!k^hPnMA&?yr6?7n z9_=E~ct_eNtY7bsdC95&Zbn#E_|VT)d$td3H8(PJU-wu%d&*r?PM6?)0Y{UuD8c{$ z1!}N)Ub3T#+fAoDo`6wViXC6;+O1qY_^Md*{r0PdaHjxT3@8^fSej+7zsGHNc5xMt z!~R}4Qk0+LJ_4}=z>D7trvQEg@Cfx{zg9LH8uo0TD_O@`M)FIkqEJ2{17KdWh3ftf zyEH7xz+o<*8ya0$-0a>VK3OdT|CfZSXP!;|qS}^xaB3fMF|F_09y^8ti|fKs=Jge+ zd%4*h?!fKuAO-a^GS-kMEXUa-%$*JfrSvWAi!~3Qo1CknFba2Ip;PL!zr2>&K7lJ3 z^dCU3=@rm0j<*DvkZCE^^z(QGhB43eMFGmaI%E&89ZzT&S-@p!p`bclcXMGF-!iC8 zRetR+Fp}j>b=ZuqZuW%HF$l6!;}TtLsCDa1ZM-EwaS>C?09^9iw0H+wiXg4RBf9HW zQ_CQ4M`(mZ>e@YUaPTgWPPY{;I7aK3=CkZS{!h#RoeG$^?rEC`#v?A}Fgu|37H9T} ze|{pcH!r%(b->!hwiJ2;Z+&Zsk}Q(2x!s!_RQY@+5;;L>C<<_Vlk+q%Hr_&=I*9Vn z&`{jx%lq`=pKnHfR(R`nr$qa~lq>z&+dc2+?2Bc?RnZNA0QrKq^DGyv-`_tZUAw3S z1+Q31)5G&s=iVCbvk|WluUljvdC!s+)p>+a@r+W>A^T_T6Zph&--Zg1+Ov`lQzlSY zH{b>9-!)eWV?0)5Cf+EKCool+r%3b?mMshuyZ|p3E=eSip7?@WJBQ5v1G)R$`g*CQ z0ov+E>8PkaSZ9Cb)n92nF#wvvk%Zg~D_S1J5G9wyW21HWZ@_x3;v&#*c9KF*i-L!| zcC{|+jK}RdB=xi4hPoh6K5CYWDZ@-Gpd-+ILjTQzn+AE7VXj04Vi``d_}bCrcLz-{ zpZF4cX^`45qx(d0yM)fxR8dR>*m zx(O0iBPX7f^o1qD?C8izd`~s92Q>gMmrn+Dz9!w^c=daktWV)aCK(t5o+Bj zoVpB#znV^RhN>nSm6GN|y40w?A7i$rCa^J?L}mh=;@(Q|1S7F_KDdo$M8>pPtu-u; zC*XQ#^lY$rVR2e7l(tF5;)tPP@}anR{-(?`s-Iwc<3?ELY1ZExpDEWA^Mhf;5Wj*2%ra( z&58$T2?Wh|*f_y*Zw;1(;^FD{YOK=ggrCLFebHgyF4e4vf6AT)9tt1>7XIsRzvDJV zis~ZW{n*DcL0|JcFUn6NMU-=Qdkf8H*_haO>sSAHYVxzQC#Xq?u-M+deW6D<@Fi=K zj0@|-IDRj)Du6GrX$eJ#X<_<6W0Yj8T85*u^TWe1+Xp(gsKdcjNMd|4{k+=AoIF=5 z9$ypD%ZRKX^<~J%mxjL(s#tE{?wDJxclyAD9x^l{X=SqNORWYrBk=;RSr@bzIz?UgzN8_}8Wo-E0B=UfLToWXqEmahZYBwzjr`w`XT&!n9$ehN?)Ms9?#- z-`gS2{eI!V92jjR<*KQW^YFdMh@rXSV}AL5m<4#bK#{3|Hpx~DKXa?NIuI(Hu!YAv8Z@9Q(?G_Y1}RPU_r$ji!Zf|e{QDS%;&VlE6bex@w3mY$DSlky}vXcYn@U8D`bXUu}FOKFN zSYORpWxlnD++JAo|>pEZsNTrlh$ z9Dlc#+5ualYh#32n84oYhD}^d-&duc{%SWl%^am2+`?3Dar{Oyu`Q)D>G#t2To6CU zLjCY`dJ^t0Y;Kea5UxQ#!!P^6DM|-m20qjeGZ<2#Imz2upGd}>g3NT~=Qv#Tdq=f| zG9cJ8(oB+0Xoc<#f}{PuQrFDC;dg=cA;mZ0()%sBLkttrEjqaS!#|*e6f(`Bp&@i= z{@qwXicnsb-SJKAZOR#Tt9;DuZgS{)D%t;?HRt&DC%x>q)zzUVuHL;Xo2Yq*buUT_{?cT|5;F1=Yhkq-TQ{#oQ>MpOF2lZN==`T1nTGU

\ No newline at end of file +| Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/atom.xml b/atom.xml index 648019b2..39d19df7 100644 --- a/atom.xml +++ b/atom.xml @@ -2165,6 +2165,30 @@ A\gamma(z)&amp;=By[-1]+Cy[-2]+D\chi(z)\\ 2024-05-26T16:00:00.000Z + + https://fuuzen.github.io/cs/os/ysosv2/lab5/ + YSOSv2: lab5 实验报告 + + <div class="hbe hbe-container" id="hexo-blog-encrypt" data-wpm="Oh, wrong answer. Check and try again, please~" data-whm="OOPS, these decrypted content may changed, but you can still have a look."> + <script id="hbeData" type="hbeData" data-hmacdigest="27e5fc65ab2cc7099ffb20e8d7788ddb6c04f28fca90259dc262b30be6d46c2b">c30d89177eecb6c2bd681297bac3c03f60f6add349b92ae09a93b0461a16f3ad8579c0e46ba30d7e45c4eaafe84e25352daeface85f774dd3b167eb75202e1ecac73371b260178eeb7b0705fc541d817cf7542782552d2e002a0645bea211b0a62fa00db1d5c5185646e810967510cd86567114798aa31c47727161b937bcf20c5636265b3fb3a761f8ae41e9888e34103f7426884f2a0f8b2d6ded5a415dde2931275f4bff011798754cece2fa7fff5184156078042a670116312b396969d0dc2155df655ca849d37fb87daa1cc6ebcfdaadcefa9dcc5047cefdf33c27f72ef527260cd32c23b30806b25dfca6d05f857ff228bb61ea511cad56978e35464f9aec4e450b5e30a8058cbf1a0a6606f845eb9bfd5b8ab0532132eb7e3f45c5fd7f90febafdd306e657a2d893fdaf45f2363cd2cce13f9d80670513f3b63d98694b113e5e029b1c0627aa9c6842d4bd3bbdba94341c0e373792340a525590c596b51b10f203619ef982d12f82921c73a52ad975d978967cbaf00488383024962889c057fc2acdd4d0b9eed466cb73739356cd2c21b5564a91363d0ea7435080a5c05c6f72d20202d80fb2c4ebc0dc822906ec2ee57526543d2863528dd05218a709e1171ccbc4f174c2ea716988bb72e6c540699720db1cafddca7ab148293c130530fa536ea5e9c47db70d25f198f659bee33e6efc331c5c1f1cae5dc61126cae4e65fb5727aae70f10cb80b77b035a9bcd6711461bef0fa5aff0e234e8710e98739fad56682ef557eb19626d2cffebc4c4fb3bf5074f5e8f3589df923f011913f9af411bdfa4f705623c9368965cab9ea130a3939291574b0b4dffeb2627b44149c2f38ae14dfe52ff810b8f929ae6108f13875fa1518c592106c88a6486c01f2280576a43bee3ac049a26d35ef3c1b8f22f2721632551f770fa4a8a42c0410982d83628cf926019f5a89d4ccda9376798402869810f195cff3c8f6a611d5d0843bb4d2216bd1d3b3032448956b691f45993a909cf8b078c22b34c038220443a7a82592be31bf361cf225363bc91d33ceb638fea44aecd541444e07f6cca11ab01751b213609eee6b08319ca21b78aad83a5b896e569cc0bfd3aa68ee50eabee389c4f4dcd40aeecd1ea4b3382ebff14d620368be67760eb06774418511c62e9256f32cadc1dc5444f0c5e1053b007a06e68873f0c9adb98af8dd84923d38400e475894ad80bc404459cc5b8e3886266c8827ce14feb7aac9de594175eb6b3d526b03cc5f0dbf1718bf1c2b8273e8575f7e935cf5e56489b534d650f2d75333d15b463247ad742b8ed95d3303a300b7a97c8d7e218276ab1005bf44be5d7fed59b55add0bf213fe7dfabedcaf05f88285b9196df13d5ea40aff6761ffc56e56bbd80a66e76c74ba2d1760364b53377828b7ef5fbf020991b1750f0a34360149f7890e37fc38cc742009d83c5a65752e3adbee1c134ce6889a86a6e5d7cf2906d558b4d93ee3c3d81300ab8bef487dcaa3b08e148c6f73427d1eb80d7123d260ea35680508216035d0ffaa60ccdfdaf54c01a93ab28da97928b554ba0bda988a048e2ac8f2f77e3c89ab519041e68723c8ea39bbd81236c3124b4a6ce094e6e2fecdf1f9245b2ec6a9e9f929004012e597e3bdf46ebdcd55ec60d498f6b5c53340d22259176614eeffc2a29f9b495b8c7e843bec21e90333d35b59a24dbe3cfab74a324994062993c2ae8e34b31b90780164e2a69a280bd02a158d16ad2ee832bfd6358fc5defe74acec55a4b4fce044c5d001214a2c3cd5314ff453c298062086032bbd58baf0503e349073ddcd86eb06bfab247cd46668322be4d563eb55ef6d76e211af745b0792c79b7938a31a429611f0f9634bebc11fb1a9ad55fbf821666394efe2eb4379b144668a4dff25d78ed9e3ee9027757b9e868ffbd59d688442b748ac828fb5424c69e9589fe21985e9a90ed6c287bf1391edc25eedccf2053dc8e033a83d76a6875380ab87ee7f96d4e09dd618e59b77bf14057d6fd175ecc3b855d7471f4ea721cff0ff589fa343844ca80ee9d52ba869080cd79f16f813f089706e05f1d6b876e165212a66afe40f5d802ed6006c2511339062bdf2fdd9b9a4e744babead284748806a7e6467e9c5b0fdb1971b02aaa33a998f0339e92ea3980764511917c3a4d2c4b948123b76745a12a7b3f11f0844e3521f3ff17a58e34d0724c9935c127eb15760b8c30be3bf63813417b9ad5224439fedf5fad493a335bbb1da54aa772bbc23dcf8253a57eb99687b34f5b25cea18181dfbf535afe3d4c77de8190cf44782b2217f078947eee55a5d954652dd0b01f5f47c92f699db42a10cf316ede97c7b2e7fb358c10022c3aa8b9e64ab7a1078e7c7b19a34846746e48ac7bcd28aab0376ec0d3784e43ad2a0c25742daae0931a26ecce809a4e62808b9fae573c0c344fd808a076ce3a070938272e15c749e2c45ee46c7316fa719d4fad8de2abfef9265804b3ae7ec0118c90997d28b21af33715de9930e6f428d085f25d020d4cf3725065452bbaa5123e68077b95bae5d92f66a8d6b3a82a21139833bf8e29383ad024be22c84550749cd5799c2bd2068f6d3f241d7e3085f5dbe98a24ccf29e759026c6a683ce76c10a4513a9240b7202373bc997d18843c7c929492f3dec3a0ecc05569931e34e642e36619c4bad80459e27c826070b46baaa6dd4a42d401536b354c1fcb619152ce9710e5ac6ba0ed0bd6d4fb66a4fb5884946db1bcc656fb8a6cef719cad6b82a922452f087b3f12723f1ccf2dcece86cd998189ffd50afbca73993ccd3dc14da7383c67ae05993519754006c25400e2145ddc90c526f7896a10fb867fbdbf6432338c0cbca9d5c4ba7b558a89ae4af0359a4f40b9592b2337af9736efd15d0b20913c683afdbc716deffdb600528f9f1cd639c440fab77c313b3b268cb0f16f9d8dc73d7a1c197abc5cce9537c9984dd5c52fdf3ed3dde1bd5089de0ee742179218cce4fe5e05a46bcf1a67a17c37c45aeb7aa172e30da7d5ad875628dba8ab88844a5f052d941a0248e3009d7ddcc7058d1a9916ac7045e1178d1a03100cd45dfa71a65776d35e0114954acc43514fb319c0922bbc6c737965c26bcbde658314f74fb2620aa0eedfccc74776682e9baa0e0fa4e57362fd29d413c7d849abe97397524f97a2463fe7ff84fc89fc90b273cada19bbca5bc9cffb32abcac3ab150c1d195aaa7626e88765ab5010dc587868fecdfd12b9b5c67b840791c419d71824ae2f65f4b84d95d27441e5cbf22aca7f0e04a0ab60319257c50f7adf7c6972f6673850e1a7b3a981727286bd017ec71bedadca15d72e734346c686a878598b2727fe967b6d05bb5c49eb9ac89f151eebaa266d822ffab5ca17c22362f6551901f992029e02259405b2cf9a6e4d721984d0a373392c729961a47eb7c79e2375501c6a1798d3c28b4b7503da32342f048ba000ffd77bb070da7f0b2f9e8cfe616c82f93c7740d29c451cc8cbb58b9b373e76fdbe0d49fb2ded433b8f67f043ee7011d5391a8d0b95c5b9b1fc6c22baea4b676b9b78e3db1d0291f8138852f1bedfa381b41432e1a8100723ae3e90a80809cf60888160d93a8e31553346f194a75a714a84d669b01fd25b59658cd193ea243eb4d75d947783d703a074645cba442ee675481bd7acc99bb16f08e79c2a45167a26c4976a94ef5b4e933b6bec74cc283ec7ca63b8efe758ac43fb7ad8247b0350b942aa5f8bdfce1791274baae6832f36b5185ea929a7ac74bda523282cce27319f3fe39d822ddad37cb3f83ec26c6b3a991b393558a3b205905a6bcd08e237a8de59d9e7572211005005d963b616bc121743cfdc1e8aae089ce62e5f25a11d33e25a7fd1697c847a5a2e4b5fef87aff7659b4e270f9251ddb9e9c2004df6d848b3cd7e9ad9aef54ff7f770227e79d71e27e668cd7041e90f7e6c03f18288f84fd6809994ed99d672683eb3fc88dab2e2173169972b65dc79f806652322a6d0ede27c797c40aba3e4d33deac8b2e478b79b49838ae7cb197522cc89b5f795b52d887ec9b0618ad54c279d37c8ba74b8530617b04a2b84553f478bdb94357b61e842244e15d34f4b1ee8c780067fa45300d74b06071aa1da87ccbdf698584e6a4893d25abb51528c9689a37aa19809b197040586c8d59dc10b730c603a501ae189a37fa44c78d9f8301d0805278e92d28d609ea9f8d78140166bf7ae1c946f0a4230bdf907b0eac4bd0a3331e2faa33805c3cdfbcedb2bd687bf8b6675d9486df5a1dc758737bacfefa301652e91f80b6f0419dcdfaed097e9c16b117ec5cccbe12e8f1c12f21b9eb218587921feee7efdc4720f76d219356869cd50bc8424a19fe0ce58ea75a06c8d9227ac260906509cbde466eccf210fdd42f595a2008ab286c8a9b30e0d1cc18be0df2b187db064245fe005b934e2145a3fa3e23a880646b5828fe3c10dabfca65b8298efaa413c4bd62fbc273417faedb29559b236bd74dddcdde929592a4bf7a5a0b44322d6cf792ad66a087d716b221b4d1699df3660744afb2c3c1d74d3c9d5ac079bdf2e2cace9335ac174b42d9b1e3e9ed2bb36c24455a49055b0b33597a4a8a8f2933b3ae9493a33688e54e0a0ae81dbb1d2f01b801dcee6a4ac28896a87333023348ffe16df1545559b9888f0574c1a8f78f7fede0db878a1aadfd0ac3f78a4e3330ba98a66fa0dbc9b6e24d463e7ff81fc0f987b84540060df60ebc526c30e6f0d874aaa96610bc52d2a5f045604ef63b581bb9c1c437886d808a6eb8a7f3fc7e26c77114ec8ae23b4af58e20ae49837d927e9801f8367ac5100e425ef7bea38bcd98d26fcc101506456054db599f6375868b593860a9f4f1d08ccebff9281ff59f571352f7bf054a3b6f67c75d4314b1d42104d31a18e77d7f139c8f72cf62368dc2b87d74730405fe5bddd45677935f73a9e957c87acfd2d39126b5e2f87f35c64c89a4128794d1a5d6a0c187becfeafc343c64efd5ecda3800dee6770ecb660e9dd7db4c4ee36cb31e5fc1aec55d8537089a3eefa033de2f0ddf7ff7ca77236d8ebf3a1cafac377b48fc88e8b0a55d3001c5b06597861fa1893d66b0f63b3d1f0ea79fc5dc7a54700b4c4ab052ec2383c2dc12399b6309ee039f0531841f84c5fde45501ff12cf9625a35b3c2c870d56c88eea81604c5e223908523d722331471dc13ca45085ef7a949b817d4cdbf606b73655e276c11bbdc8ed73679080ef8ab11dd699ceea5d76430ae35d83741e5479e975cacfe55f9afe9ed4a3faf5d80e3ddec57ff5bd979e38efbb9899b418a6a9c26915ff26f1d004a67d562fffc799fcd31bcc1a67434afda4aa7523c9acb1769f58ab09506d3731f47c80499fdbc3afdcbcfcfeecf7362e0c1c11f6ac894995e547555f26621aa788a71b5a424376cac60a69fb234c268b348acf06a509534526f64e8dee8df55ba1306b043de4f52cefd2e24782f72271fc5739662efc3e44b98b2cdb2b0b755925cf8ae4b2864c511607270a78783dd7f7565bae804d8bd5938e6ffb585c236a2f7ab38dd39e40386af3620ab60659b732397f7a466948ea3a757e1c44c73c6819462d79e622dda4cd7302ea8a00d4bbd645a86bdfc87062b8649d949bcd475a7b03ebdea86dcbb5848d87da73158d5bdcb085fffa67de084598fe9eb8508959c5acd9e2bd4b04c9754c3b0ea4cfcb4f75c132b154b828abfc7c454f5258f9ec5ecc474a765257da9688d78e694ab0104f9d69f848d0c6201e2d1a27125b99cb7ee92f53369767b4e743c3bd0f2f66de3a0cb2fbe1750ee90c0e80a95c441140f4a14cb38cc261e1c347e6655cb3063462eea914f18649fc20d256806a997c0efc40025a52b61a22971655cfb69c3a94d0813a07e94bf0933a4f990573acfce899b569f54f1e7868ba680d14cb0b7a1d075466f60918d2c9f73a8a5b73fb2bab04284a4ccb7e3c2ab7916b4f95605abbe9959de6a3710640feb20670ac5389fab7497fc516ee7af47828c9af31f59eb8324fe49cccae9abd08773494ddea84c5d173c1912b88f0f9722e9b11a4e93bc4621c5b35d5d2685948562444c5938debb5e6f3955d3588acb30ebf0e301490fd8d3f7b655cbae5ad5374f440f9a4d3adc4c85098699a671f2b86bf312bf6b3711b890989317d60a924251a1ac21635122e0c2a9acb4742741cc31750dd76e1423e1575c1284d16e973ab4cd0d86b3d9a8f846be58e40b1e590abe6f57fadc0e16c35edee610aecd2c27cccc96935970afbbe4ebbf4fcbdf234c7b1152ecd69d7eda0a62e7a0da197139fd1f01e2b36de6b5adbf69e299ce9b119de6d535a0c2bdc0c11f70489975f61c210fbb10bbab3966acf3cba321ae8b0ff5e53a5e87ca00e29df0814157ffdd07106fb2aa83c74c91a9f26ca59d688ad07b650892ede18e112ffe68e5457a7f0be9b6b5058a704821b77b8da717b3d2c0376511f373db6ccff3fc18be0cbfa698c113661b40670f99b54cf2f4903919e3313275f56b270787319bf2dc9d01ce95650c999a91895b7dcb5d2fe600d86d8bd99dddda7a77a4f25272cf21cc8530ee8ccc98dca6eacca291d9e976d5f552f903b780bfc6bdea24f8e419ef1865458d833071212adb6da557dd623e29c6cc756224303d14e448637e0be148d5b2d47d7b39c7b9410a43733c5c478e350469eba25c87995e19376cc7266cff278cc8f1715f8d988fb14b236e78fa46896447d43a57899afc86a5df23f1d19224b23b7ec6650267e0d53daba1bba3433ad6ce57fffc76a4265cfd6264bb2cdf53e3a562ee2364ed894c7e53ba75c28662fda66d5f1d36c58743d9952c65a4015670305370534073cfbc98c4ca78e0dda4f90213cf4634bc3c6ababd9716fe1069732a54e643f6d51a7d43f0891cae0d9be37457a4d7f7e290b51a36b0266a87efb4590bb36cb4b6dd3a2fb6418d0b3f063f5572a98748b7e17000610bdc25f2d7d26020a84ea7f37a575b73bd340c7ac2ad4617110f82396bafef669a1b6f5af6d483ec6464ef7f16bd0d9a67413a74a8ee685a5f79914aa86679e8579bd8f1d662f330b413123051ed104db09fb39155f3c5eba156c781a4cce73f69de7c44854925f79d9599aa75d94093742b45dfe3998e2baaaaf07d0093da415f078e737f76d1cdf63db71fe9adab6c7780e82933d375085e29df554f16e2928358b031450c1a968bfe8f6fccad8e096232a8170e6fdc31a9e81bee8e83d9135bfbbb6927fb8c08df39f020a38159cd881bcb992c97ac5fd8473cc4eb41f31361667f368a65d2736301416385e1b8162e69a312cdeb9c59575382f6bd59cf6579466917a385913880b5900ebe224082f640c805caad033fa7a9c540baf0f9b7c6c8138828d830f2eff29f785b0686b1ccc9f64ad6a4cbcd5e8eda297dc3f4e852c3e7a08dab85084ea31f346e89507a98bbaf2eefac354fd23c68b7b4da35af3527698b50624cbf7bbc6643f7a1a949db2bc83d7a8b9e94d5bdb97366b2889d5d89b0d7845126664fc94bdb73bbfc59a84c232d6bee5520e124b31707e840953935641c2e8dd7235ab761d62ecedea164ec5aeeaca960bffbef87d2624dfc443ed9cf8eeafd5f62dcd9ecceb8a8c2d9de668eb8fa26f59929f4528ece92273987fd3f40432594831695212f412aa32a7e8ca281f4b0dd487e07780977921d56e74d8e007089ac62f035bc22fe5c10e27d4f5c1d60bbf925ca3c71c39fb7ae3f11d513673a0f6a56e28035e2916f0f211c7bff04a94e6da52b6609e95a5cdd5c11bd64940a9203890dce219b188e00eb2e3cbcececc25417653f500ce628ae4b96436330eecd8809062e785fb1c0b11b646817a0c95cd83ff37a9d84eeece87315519f5b799672d851e7ed0da0e28d1125a23ce9735655ebcfff2126911af0127620dbf29ed7d92ad87ecaabbe34091406c2079a6ffbd78228658dfef5d681d2b9e7854bfff32e628ec3659976e69288ba3b412293e6176ea26912b540f34a766c75611a7f0bbf7e1481fabeb7c7158870036d23bf00e2cbc4573ca6ef8fe87e0c234c9fb642030499f2ed377432b8ad4867d071c50a58621bedeb6ca8d4ff192a6db6cd3ed45f55c53a97f2e215b287d08b3e54ca4b229c2aa6ad3964401d785e803b59197dec987c3b0cabaa96d0218c1a7e2d2d358d9ae6eae2b6a39f4c5fdf283dd990c44d574a01574e6220eb48f1fe3e60fc0f6e77ad6899c92ee3c5afc828b822b59c4ba1c1aba9c588b34694ca38887717138d4a67496f83718e9450004f610ceb970db7af49275d740d1d1b76f7b80b0dbf681e640ee9bf1fff48c04545e85f54ce077b69f63eb830bc73ddba0f5206c4a813248e870dc3e11cf1d8044fbe841401c2c39297c350431c6ac88f9f1d97934bb2d6f2a47cac865d21f9176e4d13e86d4f7d37e5d6b6dc599464685a50ece017bc1f52c795d796f443a7d565b159b114eb8b1929177a8dfde940115ddcd5c51c486e0a15395f3a585dccb14dd8332c99d152537c04766c2d8b199935f76fb3b4fa0aeeee2d962acda8afd0a12e68ae584d6bdafc05a8f3477d9ecfb4136511b0784fb3ac029a1166479f0da21778cea69faeffeaacae478864db709a1d7a65c5e2124272c47af69d7e039853528d555ab31ec4d6224fe9b3d74d733a2599f06f94542359d3e82087a3fd3d728394e79e4606815e8c37d2752b1840e1d78729954ea88cff10b577865a251d88a8b75499d28afbc1353052c0562887fd89c6574c55a70a5b1986b9b949d3c10a8d05d82fef4c8c597934d6ac06c0d8e6331021b7578e4303a123384e62b0267539b951a54b370532d2e0a9aa6b5f0ec1e1c0933a6ff8e34fdd8949d395854fc1d8e3e4967847c784ec5ce3aeaf5b381bfdda7c9e6a220c16104ff5d3f3fd79bc23b41baf6714acd03a9453a21be9e2c5c05dcbccf79ba752e3bc6c06b419a8d754e992bde244289b550a54552dc04477f6ebe6ce6eb175e96c7bf83e7b35dd6ea365ed9a86eb939cb2fb7dbfa8ad73617f1e5b7adcfc0e78b7aca240082b6bc11033fe9e08b763d9511e935899cf4abf79394b16f76a2502b865296d6ae0017433be67083e736f5593893d6b2672343002826ac04f57c13f79a2735aff16a8ba4eaeefa435dacea0efdf2384ba4a37ad4ae4af0452c7be8678107c9d45f2529ae6d35863d58f5eaca86f60b74a5b5f05b9530c1d4a07c9bdaa40a155cd216d2b25465e26daf8f1af3f9239386862328b52407c6d5ce08d41a99ef5f08948a7ebf282fbaf3a002ca7052fb30beeab961834f9ad994cc44383c82a6eb7d20ff83478ab292578197b2778d75a0f03f82f8ef4424c58c6d15968ae94b6d34b6cdc8affe04358b7621380a63b81e073b5b09b376354f8b0db80075902ceceebb715a7bdd6bdb98f4c037b19966f310e065c53a8dc43167ee8f75f37bcc232e902720cbfdf868d7bb78a82f72b8ca70cc395d9eef310d4f672aad8d996c2a548714e5f74d497cc6af9e764fffc5235c51f6e7baee2ce05fc3d4ba7f70ac90bc9858294b6362edf36a54f26966604ab37cee697bfe09666c687a62ab329ab4040a2ccc5a1ea7336ffd60cbae01b2e14482e7d127479786d4a7866825454160649e9673b8304cd0d58cb7f02cc9be67b80220a3ec0c37b46bbe4f21436a4c480605c23315cbb790f05e46e2cd2f498b269cc0377f7e57b8e34bb3cffcf341dbaec9c6c9c25698e3add17b4bf0692e1cf221d6affc74fc02aef529e6514150a6a405c82d4b4178d94824a61b07378cb9b83e1570e2e82de7b3d28415dd29725f7f5479d5629d5125047d33818bac98e52fdbf625b85b4ab849e719ba18d3659467d66a222551b650571dd1e33b58121b8fb6e7f532cbd15ddd3a8126c64cdc3e2cb97cc8cac626ad8781671f708ce2d8a19e02ae3050dc63755832bb375483d2c04f1d546c7bdc34f28c15ee42f6e8158344835aa4091fbb482ef1d0dfc4cf8533ae8ebb8973475b9366089a7c7531e4d21e389fb96eb6ab51fa91258864b0893caa42876398a97060e8de295cc0c2750a0ca4726cdfa555ca4102cce0a899055c3da6bfa5efbb8171f1098b597a0bd13b97f395e9d58bb7b612a72f6977a57fa48bee5cf05d459f5e9bdabe33029f4dbf091e8ce27184286fd92644a42b9e9889bba94550cc5b2096176198c5d955a4bca259f9b5877f997c83fa7537d055639d4fa0e2e5ed3ec1ee6c6b49d9dad4fc5e72e612606e64352d7e3a78257b1caaa5380069a1b602ba4c853457cb65f9528d2dd2b40bad2314201c22b72d49718fd5adb8c82cb3b4d51c3490fdde9b332218644ba0fd8854eb626984dc044cd5e82c56bc3c6de11e1e42a64afabcc79e68a60a4de779f5f70888cbdacfc8d4f512e9e6ee3119e241dedcff5b66864af8007e8de5763ce6fe05c313c2b7e629983122a31ef6f59c875096e024af8200d18568707af1b8ecf7ce15f0f827a9581f4519c53a9dbb065c357a687b1987c12b7da86d3f430447d17c2987836db160b32cbae1f34d68b8a7cb1f14b8ce56c7b12fdc0da24046bbcbe44806915408e514e4268bb5cd1b07a4b41f9888de63204b1e712a3e081e3c2f43ec0ea09bf1af14622ddbdcdf869aa2428e200262e2a4c17d73e670f4e3b0128b601587f3aed3c4a1601a89aaebbe13f66450927ee754032589d922ca63822353958a7605a09c5033690f98bfd335764ca14eac571efa8a864c903c0ffa98453ffa3494a1a43579a12129ce81b14b84522faf97f4b444ce67a4ed210e060684325fc6d4f104d0c65cf1fc9450fd021ef83cf641685a464a620e4060fd7494c55b97d7edd501622fd40fc1986eacadd975de24a89ae509419e18569dbc899a2f168d0835e4701fdd15e95fc59afc448a91a3d0f8c07c68bb2b6868a3b204bbae051a372d04feeed76c155e8929afe4d1c445fc61f2aece6dc6b0ca54ebb8c0c31415c7337c0ed247eb292fc2b1ab499c9d76db22224623b0704e4867dbd5ff67fde324f1ff2c5d089c3ab59bbeba120c1d3225520d5b83e09f18f36dacdcc973ed2dc063b7bce1c9095843857fd5913cd8d90675f0bc356f1dd5b112f89c340f7faca28b74bfecf311231671f068c5a62e542ee9572cc2929756d99b4e1aa345d8332ae946ffa4b6d2e8b9e5d930e3371e3143c8b4c709772c60e36c31ec6006225bcb8b4482d54f2a573b9066b2624fea8909245f4eab0a345deb52f54249e9a0eff6fb28cd9ef244d6653fbed93e5253e3f480077676e715de68a0017e4989b9ccc045b7129580a2b65ef2d1e9be65dc2ce94c2a5d88e188358907c113dd0aa3ba1bab54dc0546b1fde07dbd85051c2a9749561d8d2ed8d1252a7fe53f0d6b8e0806c7ad64690119179fdaf78125bb641a578d6d5bafad0864bd8dbd15c9766b0adf9f53e03cf239fa6fa00b583016edc9a2f23532cbcb094b21b487b3d034c878b5d58b2ad1377ec8fca8656f2ef05aa138e26e8cd5ada5d646befb75d341e7498e605dbf3f5961d71684947db4484a0e4dbe0618bc8d63966816041c9abe74a3c284bcd4c2a2192a10c5d6c72774e34c1dab9bb7c5ffb226c443c2274acc0b048ca15d4fcfe042cc874d03e91a5b8f6c5f8eb9235cb739fcc961b5f8bb3832c16b760e993bbc2877af1b6fff45bd52f50dce1ab7f09d6883eafb3777224b8b7e32654cdb14ade89c62736aef56b5013b0a41a8d9df8f1204ed8e2f9db62f115015bfedd8f5af0b515983f525b3b9c283659eaacb1ab8cd1ea19f46c1dabf9e6a2de9d2f56777c8ea433d0a22d90ce119e2f5721eae193b4b569f552866c6805984a948cd7427f98f365dc8877df5d5f521adc82e7a0f1623c0c7f44453b17a6d64fae0ad489703a7d2608f559b82d57a1aa42504b9fc80c55e9753d81548788c4c00c8abd64f201d41b558ba4bcbea30c63f63eecf2d3b409e392c42b8c395a30d478bb2f750e012e4c814847136492104dc30c6b5159548bb3e3fc18642e4e77945a39c5d5e47c98353b282ef436d1fac346a47b3126543e617bc886026090ba173421a310d12c6ba60af1d1fbcbfe17e11ab9875f133ecc7aebb81d5f5e0fea6b4e0eb10a499d59311353c734fdf811dbe822661cc0e2fb2040be19b0b4261441ed27a0015d9bca9bf850f5de971761c375059ce707894c19d4db61d0f131460790772c767c737def217688ef61eeb339515590d3105f3ef1b34c5469a3ddb1a0abb3c99e5adb32e9c89670e8739f119876d68a205f0c8ef9e20b56b8f0e3b191989db0e9ad24aa72e9787d8377fd2f25ed6f8bc66b1139c798121adf9f1b92e1e6b83833b46e74256eb9002f78355ae3348055f7c77d1e13b15af45f96d771695ac8098d9783ad79b3eb73f0e021a052770931bbca9b071508156259b80e9f5653ea6df41b72b67c03d4ee15bd70c263b3c91a3ae2399e6212ee2de84ca12799aaacb2303c9cefd8fff16c79cc5df448b1ef101efb193d179569c940c27dc57216cf2a0793991758ce5c82063151fca42513a1e75df82b84547363e5c1bf69eaddd3316e5eaf50afd801b7467e431830166d53095659e9af9f8c97cbba3605c23c47dc3eeb0c8f7a1b47a864c56f2a83a0b47b1cdcb6303b5755e4f7601e2c22a76432a199083211dec2877ac816222d43e4f27d1742dd0690a837d9a45df7819757fd933b9451b5af89e833ec53672d3db2810a63e7df76df6c12bfe140aa5909c36e6e6cac841452e60fafd64ae7b4bbb44ac3026d29460a36faf87279f7ff3ff019a5cb615f57aab22fa5233526875a6fc550f3696a36da0408734052d8a1b474e101a8184ad3c59a8da62f94bf5f3c31a4bd788d10be659dab94c0d34f1cc2b15e2f2238fc9220dcf3ec844b65ac2ea4637e1b94bc7f0b18dc8f32122d6a42043a1a28b357edbf00c22d72f79ef03d678e4c94ef06a352cb6ae39ae77663f47e7cf9604fd64e1aee3d63f7e3813538142d60e76b2392acc28e9b8000b02737b2148d5eaeb317148361341f36594fa47841cfc9d3aeb9c8ea110260bd876ef90d383c5f40d762273d554cca12f9c09084a3e96551c6705e573a3a577b4896190c672340b37eb4de216b95d12cadcc1c565cd3c86503c246fc561ca6c8345bc991d75257e9b89331d953a5c6d38cb6e6a7461cbacf13d46945db19de8b56c0edd3a68ea9ecf89cb201288e8fb8cdb0903578fa53f9415fbbee3687ec6edf3264bfcf749d8ce7f3365330de601e2b59c790aa9c0b5953cc46b8ea08fad1a1ceb66f9eb2c3e4edb43a21e2b6c6cf5b33508ce37f880685ef883367a4bf3a6eb609c2bca34d6969d5119ef7f345df235689f8f0301bd091ba7b19e94e3ca1b0ddef7e15566b51b06e1f4fab76e1bd91eb2cbdb9034d952ca8244e5e16ee66f48e2d6886db91be608f91a2a8234561240217de398ac1145a12d3c534c52abdb4a601f1f215887ca7eb0c0f732bb5fcfff5c630d3652dca1283f5a6093155b851a3c2e2113d3042421543cb7a73b1d295fea8fb3d7f7ff4de2384b396424189fb1835c4f6e0bb8dadda8a5df514b5d34263844cf358c74b66fe2920c9a40112e2b2fda8cab9a9e58251f7246df4a1b0266cef7ac4807b6711ebf89efb3752037676be96b3e22afcdd1cb8b47b93f0b77c3562ea24c73cebb4bcd286d062a33ae55c488734d81266fd5b8e802e50db2fe40e1dba6bfba836769457ed206202068def5fb673f2e382c774c37620a7d757dc6016a1253d76868d4cc60a68b7e4c31a30af3a1f30b0eb20566c2759852c579652f58fcb68b73b583950454600db57357931e704f3d958741955f1e018aa01c44c519f616791bf0531b54f75e5c955d79e73194e58ead21913585a4c6f454fe21be7492a79302076389b3a6ad229e11a1945af0512cb6f923def6f8c4c5c2eeb92ae79a8763020ddd4db62725cd91b20d21bd6d26ca2c0e4b00b72c62895518d4f271051ab6dcd15ec68aab069dc406d39b47bc6493e68806485154c0280ae8cb5f63b501f20886b177d112f56924644d6b6240ce6ea685a57c2e8ecfccbde27f0c36e4713b7c1d84df1b4dc0e9398e6a272ca57577009e2704ec0b55d0005242b2b4fac1d34f39d5062126f45c263ba733b2146600083002017ebe3f08a55595e54cac9dd62d163c1e32be0d34d77baf272f411f835cf9d973de4c4a10f846d2aca66c4054873acd825e3cd9ef8a59f67bf2f66f861cd48d58498598a7cdf81a7b79f3b1506ef994120652a2844a33d31570e9c2c16fa4c72f201c78a3e7ec1692b3346aefdb2f3ac6098e7acd3ad1e92cb4e51d16caae5187a00a392bc0735ccb2ca631c63d46c3c6f807055be416bd278cfc353cb466fc3a79840f0918c899eb8220fd1d0bd183d0da6aea2b403b2462a76a009be18a0e07a1044788a30558be3236da1849e94ce1dd831db80c8740fad2da491f23c204a6ec998bedfb1160a8eea61dea17edf1c3617604cace0d3ff314a4aa6dfa338158dfc106546c353e58a508961c179bf19a38a298510854d3580daf40427f53616bbbecf07954ee521ad37fca1643db9d48bcfe6a9c1fd1de0411ce8618d73245e0f9526ed702a3652898e34a92a7dc08272148964c7679f7b8233ae983aedf429a207b82ad67c8c585444602b1865fb1f58578b7b20485c0784dff94181d41c94a371f263f5e7c837eb2b17972d74c2b4597318e07f37f1b4b5bfb1594e63974048981613b80e97927518359e79faba85281548788c2a19532a6d89873863553699053d43c2a315ff3b5e1e7c1674d98de2c7a946a71109b9659eabce48911b4e17972f13ac754acbd9c8fb6630d2d56f6f97df8edc91bf89a6b2b5b5d95a4d8828a4d8bcaef6b6180934dd7c3c2fc93cbbd6217b6f5a3b2b19aedd8d269f42040ac9bd89c740ac144aed0841142b1ab5522b541559c58b7ccbd140a4246d951d51442e4ea0cadafa5f9c9acd4bc76a8bb56e190439b2fe8bcc61b574a740cfb309a6836a7d71e400db24eb9109afa077f9721935a580b939fc652fe9e0c84bc5cf00e9c145993443b72db5f3aeeecee2cd709f0de97b472cea80c236aa109dd95763ef7fe9b69a5474c0a45a89b87f39cad9f37d17102c7125d3c40d0f1491c7abe3427841efb38d40b358e227776d4f90bf2d3eab95e2c6fbce7e89a1ac037aadc8ece592632c39fc845c327eb55a410aadf761f5d77e2ff2e8c090672f3cb42f60e8dae2cca5d6f98b79e9085d225e468da4168341872669af79ab76e0f322590c1edcbcb5593122ad7ac4147d78417a935c1d090252bce9b08a3f127bc2a65a39498c9aa49b0ad99ea7074dd039cead3761d9a33dd758e47f3bcb41d38914373d382f9ab2af1018b4cae5ae2e8f9f82c681f614e32c84647893976dc4f920fb4b1e062c47eb84c3efa029f9096bb4ec7c4788c7bf3ae1d6279cc1d19dab8a85cf486103fe7e91b3ff081eb87148869586cfc3bcfca0d2f004179e784120e6238007f7e33619fdc3a3e422155b2419c994e5305817ef2136e5b8ed4cda3718d5044c428d4b5dd6e222f50b980541b6637a693e87b28481f2a26deca43c6061ae8792dc0f10299f2f84d1a1e5c07f4b0d82a3758bcee37144833cd6d13565d61b54ee36234386c092b8ca70e68348dd6672e0cb579c14ae4c61f06ea5c73733a8b8b4ec91211e98c53ab4c6cc93b0ce34909f7a056ce8d584486344ab759d7845240b9b70a4f5cbd01f04d90344aa2826a39067049d460b719e9252609eeebd59744755aa2bdba28fdc577e770ae6eaac50eac277d16b8721355065cde0574087b8f2f25a1d0983262304832ad46330a4bc1a992d4509217ce6c5787a02efebb59d82f1df9ea87b31742b66f38753b692fca314254f6faa1078455d0b56befdd3cec998af6e3f85f13d4771e9f49197a98b4f8e1dfe022e0a9c9e27e5ce8511a8667dfff18b0b9ac7a4b1b24f6dc04b54d6ab5273e62ccc160326210dfa600c82ccc1d13cecf609c02d80fb31e36f86110187d484a044e13bf7311189eaa7ae5dba143bf251b197d595f1e25fc864d9e0ed8ab8872de3619ed0078ac883a6c546a4cd5ea175c014430f78218b7342381de996f56ca9abe2091195c41bef709a477ebad4b54e73fd59994d197a60a1fbc6803a6608764d318751aa15306aea6abb762fbb6a4a1ccc586d06afff9a2f1e8f193a4f95925af9b39b57bb857c9671f4f7a94d139d0df45b5e40110dc49bc4679ab84bcd4c279bcf3ebbe67de2bb72a1276b50095a7e74f4c07bb466418fae281efa94032aa7acb9f7fed932f87dd05daf3972557a392d2d22280ceb7e5f69100d67af2e8a6a7fc7f70e7a9fc5b12a7cd52d99ffe3db9aae557bd6bfbe828fcc6b46cbafcc19f2bca1e7671c8e5d4378939d585bbf0940edd0343ec6ce11c89eba05301103c9cf1560b31abecdb7c99771190fb4d1407dbe852977b4ea1921db40cbb643fabaa0bbc6302f1ea7da98ef39bf1dbdd61aa69f1f7040aa6be74dc46d11914d3b6c25eb3e967859dbcc0eeb1b8a0aef124d3206fc19d68ae3ec683a6296416ece8c3ef78f74478277da839e2db6d9d603a034b9921ab9334523c70c84b46059af09358e5ff3c20c212c4abd2507088db05c78e62b48ec534b49bd7e95ac29048bc45755cc3b9207392b13559253eaa68c068eab043f8d101e4d270c7240c50a8269decd5bda44c7957b3dfab210059f3db0a6eff5abe7bcc3bad47ede528d465af79ee44b5e6e06be67c8034004e9aa4cc0916fe600825faabfc740c6c1eee28f7718a2ede58a877d3bc5f32c63856901f2790efa69fec0fea99de6c64ba09bd38494a88859c77cb80699c60524b9517009e54be44fa10d861c275ec88dc376d9e3c10dc3713a5e9c09e57888548370ddd91bdaa66104e08a58ff3256286c914091fb016fbcad89c2bb6fe7c9fede03dcb7289a10e7196fff056aff30f875bde60fffcb5d7b4330aa2c4aadb536101d68b6a7bbfb22c82f13f23c92bc286c27d5063cdef5b488a1bac0f5a12e6858e4d2e2cd3b1f0bf8410576ce86ea12253cbbb86a771104e876f44f0f1a0389716d55df907e93d5775898fb623821ec9bab2e1b30f9dfb4f9990c8be466d80cd40e16157767bcc438096c35da8c93e90f31749be2d6e5d85c82c75f1f389cbcd3a217296912faa32d45dff00fc507d7f17f48636e9e92845693c70449ca2956d6c62b8f7766ec08b048af642b94fbc14d550c984ca7279e9f0cc40717eab97256b8bd8e4d36f029d27082b6d9f0a8a426c79e996af0a57de30265e5314b058a12146441e89ea285090d78c57e8302d0a88801b7dfff182275cf70aefa7e8f1f79fa9b66dc765d507ac3af690107944d572075387bfbe7b850d1d5a8d7d780d8194a762385fcc971509ad76986adf0b432425784be1f44443a13f293d46603ef6793b1082b54fc985245516af2e6533d24d10a4c3b21a2ef97abf1db0ab91756932f503bc24951b2279fd971edea3be00da2ffb5bc5cc83a6b65435dfca81c431aff6dc4fb5759ac6b6ec597b8a082b50402c32af27518531681f5be8e33a51d11f5596d521c1f01c0c4ca6039f2b009ce958a5193528ba41815108c5b6980c6507c62e91c4938f6ba1cd84db252322c6ef39df2ec7470b2df71544b677b008b75f3b9cea300c864fe146b40786e2be2941b7b60d47595baf9c605c0094bdde69a5f9d4eeadad726abec23c60d3b4721a96180d4a9a7fbe414c4c7898f7c0ecee7182ede7052c80438d816eb4561da06776f074fb430560fcec0a97c6c296306935007f324ab04b438ea493de43f474ae2617980ba3b4dfece01dbf8c98fdf0ab00bf917bb6326847b9aaad1d2a8985c3aaee659addae62b10132adadb2f63acc59c276a3b3375a60f272f1f0851fbca4b0e091a8968b050267c2771a62eadfe9bf690436bf0a3526690c801a73fdd53435e154bf9897977219427a658fba18a5cfc2b8c5fce31fb10cb2782e5952ed4319c7b9ff0a34a4d30664800b668afcb1f02bf42688b842893ad499a1261377e1e51e56970b68b867acfb4fe044d35d8c0ee553a1e641c57e24d6d4b420e7ceb2129332a774baba0a8a8d47aa04eefe03cb70030b50e59ada6bc0c3f2e0a92b2d19528a0d57bbb784c0e445413c35d31613c9d41128f397bae335af6e9c6fe2e9534fbc66b36508e7a664e69197fb60215f6a6278a1fdc6bf395af70fcacc60f7133677b57d130fa8666e1fdb884af8f9c815ab0dbbeeac8cc36ed49f6854b8b4c14ad0d0e8acdafa9be6dc09d4394754482d93e73ce701011f6e644885af8d35d2fdac7ac508be405fc0b8ae57984dbd69617b237c2885e1be45629d5c02bc6ab8cb587e9c85e660ae2d2afee0e3bc4c5f28e4857896644dabc84d9df7e8285fc23450f99be6f553dc65a2498163834e60b7fb0c5909f8e39f5bba1b4d8fb541b021eb1f8fe2da51e4171fff8f84e38824ec390d1c9534f2f45fe7c7b3b9e12e517b25c9202c0b10aed4e0a0db265e5c4c2d9157461c0d69ec153172e94bfacefb5904f0f8fffa679918a7286bcbc9122ed47e528be774b3e13e54b985879e4224283906464bf330075fc56293edebc91b94c6c7e5321db4a3a788f97d4172c685cb4b37438fce1170e95d0f1c3bab89a6662a80c214715ad2151699c3b39e47eed63a73b8a245c7dc12f8dad4ab4744d9e87d0d1f9a8645e62ba475342cc9766f134d840c12907a45ef8e490f0a20db62928fb64c600e174b4f305d41330fcc4cb7b370091d1ba454cc505f85c80b33991b931c6ad88eded8f905bc9efbdfa1545f6408825024860ebd5bb218cf35bfc36fac19203133192aad1cb99ba2ff3f2cfe1e921504c73fceb04aeec9252d96a026fdcb5d29a5b7eaff8863bbf8ebbb10d39ebdf82d8b41aa029bf75333854c1d1db46e38fb3db93921e7b56f18bb007d468b3377b468f6a9566b44228044f41ee90831f3214b9c72d49a87afd901064bf8ad00d2f9284ef0cfaeb3bd7a0d7904b8d02e553f70d5df4bfe3043146679a978be3a09df3e36fe2451aa855028f45a53cd49b644f2d2f74f2a709524c36754385e946118b57e7512e80542b165517a032c323a2d6c58cc592d1b1f38820295e35cbd28c3b5c3c1168c5d2c803210563c23513cc97170db9a3254ba445e32a245e25b1db41541f7314daaf46ae0f55c3746aa782bb87274bfb5e2687a919b013ee10c246507db49169da3783217fead0dacd35c1edd1db72c852bf74067da675c73ec2fb7875404a222170bf20a80ae63dc6bea93e18424abc6c46c4577d7e5d5dabededa641129d076ea48dcb9604346945fd856b69189310adbfaf0ee39aba96865d19629fe703d752b6a9f04fac6887cd10273ee6987f02ee84829ebf52f0f7df490abb82fa56e9417cd180d0e83bad13eede4bbbe42fd03d96985e9092f90657778783c56d21c3012bf50327bcc9db3420c4653250983e91bb617e1c0a961bd7f67dd7b88cffcdd0875dcd51a1727ac7c5787046f25e82b3f8494ad3bac0ee32c33cf3d9655367da2ebe151f83d820064f8453229c9718ead49b1709409c14a56cfb55917a2e787e2787ff383ae256c86b9c318861ecd1636491c8bdad4b78ae8f1cc3faa47ff092b66d59c24b448893df52fc409bf9fb82eef536eee0908d5b5652dcd7f29ca3e0082fb4523932abbbfc0445a7dbdcf83d9ef37c0df5d0b339387a958f5db64e74217f6808257325377db0320dcbd786649f0da2d2643a691215784cf058c0fc9cd5b0032912ce29f7c0abf00dc676adad8b611f52b8bcc2a68d48f5bfadaf667cfc335841c5eedf3fb4c9b2f47113c78779087b8010f736f2c619d3b2761f198ff6da867e09be17acd872473612e476bef16b78728910dea14c52bca704ca2dd18a737931eda29b5c3f92e86a355a33231f5aaaad60a9d9adcf4058e57509d1ad494d3f2cd6a87d331f6b39ccd50fd17f6dba95f5f156a5e409f9f666f998a89b1cf74ff31fb22baac21cb74b6fdd374063b0dfbfa8b9b99e154ff211f487b14db137482177c89f24f217b22733e68052b6859aad471ccaadf0058f19b947b8edaec3f4d0dfd4188445a7b06ee396545d9f1973ba5ed0ec78b345c9b682b0df16169d9e8f7410e633724574714c8de47c55e379f1cf8a7b4904908c5a31b1b158cedbac2d17a6c16d126730c35bdf2ed122d2c164c8fba8bf469f003d2d335d4831507db80c2af1aa8290d26f5507fb4b6e12a76bfdff848daca132d0dd6a55d2442e88d4b3057d885f6d113168c5912aa8d1c522ea7323a6719129b90f85b39ae45555666623e7f266a1f8deb4347e231e7a972b961b95352fade14ff4120b6f580f5c7362e1881558aa73eaa918ba1c0d585c815d8f6ff5088d1fe4a75582040e2a3571d664b9f9df18bae4d4ae4408f2827c399031da80cf88404d9f499ea11404b3963259d60ae441b003018391f9bfa2d4f9ec67e08608f2abd934de3371ce80638f429bc4b4d0678f52bbd4d541d2454f667941c23d3cff4c92738976ed5e0d22913e6a8fecb65281e19978a752b0e1ae5c927de41f1b73069f79820a3f83d2dc7ffa13520213bdc66cc7884bb9118906f0efe87e22a878d1ce2c1fdecd68e25a5cd424ddc79cff69b552d1102d14ae25130be53ce243c6b663790ee83ceee91c2157cd2911c032879356f7cb40d8ff67851f6dace62b11a000b8c3279c7c7eeb7fe3766bc2ad180b571ec3da364ff87d4bab9ae34a1cb3172d788824eeeaf711e6f9199ce3518193984a81fc069d678e47a6dcb04697f4563069022fbac2d021bbfd93803a2547777adff9cb591778e9f933e8ca000143bff6cd35bd1b4790a3489585755f645bcad91faf38768b005439156823a3fad038b85f882fbe19d27fae9e17060b1e1b724168db2cc774f340c2507ad40fc1722f74833a98f4c0664c4c8a7c65f057af54e68a36e20cfe085cb5771439e9ca09240e5890090fa6566fac17ff55bbe4f732928d8493b04c9175e6be84201510d1362a2d1d3a4ce82c6e8b4aae86615ffd30aef6d869eb724de0926a3ea9c5b2445f879496f1d13f0026922bca9e19b2fda74b7ed4f553f8a1f52096617ad7daf57cca7cf20b76231b3e6540d0597d7564b6344d615cf5aa43af3bd993faec621d864a20c90bfa3a3d88cb44c14560f1e577da0ec78f99f1ff245371f2da4c6f2b419c3efb683788f9fc7851aa1948d495c45b9c2c0ce86943789856975181d2b08ccbc4526c9fc125def3aa4493024dcfa2606968c40df3857e6b4c97b9842fc2104a1cb6552710d292df443a23ce5ff78fa1c51b81cea4624e973a54a977904e90351eaee33058ca5bd09c74ddfc6f1b3e68aa48787e60d410aaf2ef4833918bd1a5d1fb3fb755c4b7221f94332789ceaa9577fb6c5c189fd07d111e16a9e99563f3bb1fb7e9cb0120b474027899da220cc2cd84e710101238290407ce25f737901a219edd916f6cd7a1d4d068340aeac8647f958c8ae54d43b4c00e10877af8cdcad39381e630d431c21982bdd18c09e5a3b5f2d94091f160667fc61e056c113ac77730d8d46ee0169911d60b0be95ce94502e92dcf19848ce50808a6c264be2ea08e200eb0aa064709d501af47696ff283c4d15a65393673bf7e000ebe45e7ffd76f8b375f70cef4c4bf04ed36d1a7ac3980d79bbf762ad2e96790895fa2264b10afba3e72c1aefa6597e3295494bd86f9df4ab772e7fb74919617ca0a769b798081b3a1bc529311f4fbff55a275f23e30607f43c3e647dfe1bd3a0069a107431d48412c6e533d4b369f6e231679904beea780e99edad1350cfad986c27a6cbbf6565653a8b00c2e793ae6a1e2c14712f410b60099f816996af9f7e89ad7273202c9f34b879cf052c9dfd79f1ced00fd6b7ef67c20e6e1a67eec5b7f5ee31dc6b8cd20c05059335d6e9d095d4adf8fc10c7ccf4040e1492d3ce89896b05f6c178d19f9f387df2b214b9dd67a722e4c6403d657b3c9f45cea7e49ed382147341b78a7961936d9a152ac7578aa03ced2f34edb87db4ad266e28bde1ecd3e66f331674ca840d66ebf13bcc6887a03f662f4fb908232b8964ae5029febed548cb2b683b624bf2d5cc85db43f23afc9c6a2d1dffdcb2efcad4fe1eb2c500ea7d0c18d2ceac74ebc1a384ce2ed309557e62993b192475f17e7d75e153fd9b5e352b802c019c896c657f6e343a5148746db1257e18f133a7ff561f06c6cb1e2cdc2a52eb0e12af51b65908a18161672c4d536f8d79090e5db08e37cdc536df48077d46a13aca38720a5a408ce1ff06db48103b69e84538135651cfa89cf29d258a7446926b9acc7a4521feffb4800727af21ea266a9e8446b3f3919b75066103d80079581651f93e3000b53d38321651837a31f22649d75210cf5f378e845c099b694a9397143aa822a66c39e23ad9198b43a866a7e992b12964169c3724ed4e14659c60bfbc99fec941dec4ca4ad4ffab0788a15bd6913c7ebef8f0d900035c6c0d3edc3e12e655c79b8a9ba5b7b2bd66720e7f60f9bf26e42620cf6adebe14aadda8652c869d824bf2ea848719340521b6060e400524efc5bc4e7672ddc6c3396d012324f9de1103985fb2d1f33577047b1e27d6f70644e460f4f37c59fa42a368e96812bb32b495427afc7aa205548454c9222b9d71bc373a59c6a428355821b082a8dc89d7781ee1779f26817413c079b5760327dea32a47d2ed6733484d1dac46ba9faeeb231eeb553e0be1b5c224182f30a0c7bd6625bfd34659914e537838ba49c080d706acfe5c316a91689849f402fec0ca586382ac7d818b75eb3ac26a90d4d4ecc3645b0b820472ddab832c09a2190bf7c6df3d7a4b5b3b3049658d84095a95125dcf07b27fbe665f06d8b65804b4151eb7ab7402f43b47d5a6e69b10c68de70f0ba43c4c9f96ee9419976289ba0b8efd4adae2c7268009817b98d780859c5dc942cad5d4d22155da7c28f50b43e4b01f9aa1b306ef4cc73bd3cd4abf9f40e6c0c5dd2121dfbef1775b129a765a63d4ba6b9a7d2ed2ce80fe772830abbf36b176834c3cc0a8c8a786d41dd50f45d535e5615c05ae72bd46a8a648d4599dd69ad9eb785a3c4837da1f286dfbe4566587bdfa431a28e665fa8cda4dc0812fbedaa49dab87e8e1bf1f1507c4092fa6a95cf92f4121d4eb4f280eee4497ba627f9a53186d1c1e7dee70537e63bd4144abe09aee9d8cf95fe6ac45ddab8c95889fe6511b9a5ebbc0eb495b8829261ec460bd6284d09099aa96ea0e62ac53c96058461c7deb6dbe6015e517f3b1cf11b0c9ccc509e806c2969fb12653e36cb81b3f93a94b72beb7b20bc00a1d4f0e8e2c32538c8c31233d1d02d4159b21393a80b3d3176a1084b6a83f452ec7c3557b38cfc405e8ba1d3c0f1be1266c1935ce9660a593e57ac5f8528a2666d8e5ee8ca060918882de62b1ccfd41de85a6d380b707bd0943cc0fab146ce6a4e5e5dabddf4cddca39a70366c776ae65b02c3f8221bdd24a77b875330468c92cd3a67819551a5967cf10e2b96f8c1ac38ed3b8e9092dc419d5f09107f28f80d40bced57e1b7941ba4b6dee79be7531b74a56046e16586104b5b7702647bba0ca322232a41dc33830c31f4b8188f25cd0f48e37b58aa93e02d96b6bc98af35d7ef46887926cddcd642112b4721cb78ddc36c2542ba558e1b3f3a7a6fcef3e0d0d760e5f1d09df54620997483fd71e3e3daf32a5196f66afbb95c9d139cf88486e14c0a1045664c9a30eee09a5b0d010636a545a8e8d5225eca76a11e139ec5eddc7413741501da702a89fa91ee9a0437e72a599256831105524dc9fdaed501f9825c346ec1533af97f833dcc70e3eb97202f856c54f098bf02d44715e5590a3292089f7199ecc9a44598bf63444b74363a917cc0f02d934140f17e927ffabdf4ce945d58a22b5b1dc72bd0b546671984251a0ee368d5c324f88245e92897c3b1c0c6de4e22a2c4292d838e8b9e329d12c339d478a818bc72a9f1f90acabcd238c1dadf870b6a470bb95214fc6dd5961d19677f0f6a31d2e2cce65953fe069954498102520e8f622cdb9bc5eecbac903fef1cf85fe05e4c173d8810aadac97bfbcdebf48383c9044f3dd93b95800f39974706f46da6f0fbd2f52cacc6a668ae47b0a07e9dab1133457e30519cfa4164596a6c83937c3ee9c4954401f628e19068985074fca6d89c8e9ec3d10d8ea4be2ac66e2ba5d3c3078c163e61234c3a82fe3834e5e4f691532b2b056f88cd12794e40883b1fccdb075d9de1604d0417da2c3949035131d00c14a5df67951a9c221bfb3b58ef361170163d3012a5c928318ae06f3838f10bce966cc88c28c21ee951f8b41fa8ac6222d3ff95487f4bbc4c4b0116f2965e5d64f78b97c33f23905b1266cd1e6e1859b7ddabeb439c3a1d5d06fbe69f1df52d8744098bf45b5eeb8f844b6f1b5cc4442551d7fc8ba94461d752af50c7e119842c198d634635f16e72c7c0eb76a2296e89a37e93613ab1eb691d77bd4d73cc3fd0cd27ff906c9a86fbecb4ad72c23584284b819fefe1fb4c17209df66f5b9f9d6779c3ee312864a9738e4137691433e742e8e10fcdcc5345a4c56b96f25a71553cbe7dd9ffbdfc6b6210daccb2876488f2a1f22ace3091a3e529cabfadf6316c67a8d73ce7d836b5f6f39db51f70f95639bda81527e37c06b467ade33dd2dce18a392f305e71aa92deb1d5abd47d060144c4d3feaf30a2cfc6255a274742c2a31a93bde4b32709f0e6135003599654c620f51878702e5a7320f0cc269f09bbf216436cad0c9d95201b12b1dda5a1de3495b8a7ca35d12e310777b12eb5de38156260d311b0b93280666f38acd97adf5e152ea0943fb74cff762e1ed3dbc69d8a99e3b5c452eee1f35c1bd9e157dbae0638081feb03f0577d9f8cfed9fa43c97d1145fa0cef8406cc5dcc0e6b1bd3963a4cdf4222cbc371cb530a183f637dc2b223e04aa05569573fe072854afdf6f751fd3d10bca99bbd17ab81e46c2bd580af597c28a053794a4f07a9274c1621524c6f016123143bfe545878ff35db7984eb683147dde84c2747ea0fe9dfcdf2a03a4adb24d5f08759fbf094a625c03d0975c43409853f7d26dcc3874216524492e2562189bf25ba551979242293b918db78ec64c974f83543284e10894af676543eee04fd3211bcd61e23a36bff5cef8d012c06e9a9e007f1a9c8054c4030f3915049dcab9ac7712f73e990d7c863493443cea881e67a562b033c9e5b9386df25aca04fe264ae5a1dfa72af3bd25e656292c2c9fa3e61bba87301185da378ba8da5eb50c4edf74195fea5864456bde4d26250377337874a9810b0dece8758359ed7313dc5a121d6b6cbff1f2453f73a27fb8b6543fb474819d396e487df2884e3b14e83fa7be1b43fc8ac262a478a6f840e34b9ca5164e191c1d09b33cf808a68a027bb50c5754c4c7b153ae7ed1103d2a5eb03fcd99b305f08c012a30ddaebd2b2645d4823fc82c898e848ff440b356617546235ab81cdcee7b0231db7122b2540cdc17ea2db9984c958e95c3ec29839c2f80e5e5fc0c821f9d56c7499cc15d510d792c704479d1f496811409a60b5f0ff7204f84a35d9746a3359949665b3d7f7e3f5c242a3d7ccad3386e2421225a54ce92e2aa8173f58437bdad13544710a4d576bf660ecf777a85774a10c9a31bf105b26ac007e8504ea872a58fc1da8f056a907dca70f344824b19bef0d185859389fd3cb4b3f6e0a57d8821f2571b4d2b17d432c798b92fee3fcfa78811355b421b560d9942c4cdda65e871544fb8ae98ac9c420889e08ee128b1d3aef12a0646360b02b686bccbeaebf2801c3de8853251a966487ace9aeb6b6a0a4f98dfe740f1a8476a4dcdfd4a51a98634431fdd9ece1338aae95c2f599a5290034a736c1ba836bd44bb14b8e0c65d3104a7117ede2b446bb2d885daceae82850c8edfcb5522bae79d1a20d9ee1088d2d26064b306cbda03420172def2dbf00055861a4af075b9c0f864dc293d73498ee4e959fd9841c41fde6c0d40dcf61f16f67dfe007407c1a56c4230ffdb6a313cca507a9195a77a0504e11448772e42b2d33caf9d440fd4c9975d5d3a9adaa1beb5f17619fc3026b6107c72bbc27d9bd1af1620eed0e7d9a3339c25491bc6c58c5bfb598e5ad7784bc1e40833998345e4a68578c445ef85ad7fbefe7680e224971ea211b6439f45c31e93bfa46efc28b0520477a13bde98a59d90138b16f212b2d444ade3ffaed234542380e6e957ffbd4bb1c9bd5b09237f4cd37651030746da28e2ac88643e9352a6fae275ba12f8a4a94b92f454418888913db84f63f108b8ddc07af57526733ae026bfc86ad7399fb202e7bf164305744e8997b4e98d2fb63c1487192517c4caa86fc619d9e84a66422abe3e6f445093c52399a6959ad1d9902bfb9b69458e41f0d20fe7f0abe4a5ba958c3208bf4d60318b3c134af2dab2954e1a704aec536133a22825f70ce2011be082e623c5d1daf086baba128139ae7de1e9dbe4d5b30190b09afba601046d39e7cdf9c00230eb78de1bf75e6ca5b9a3b7a8d1b11d8ee7bab3d7ea69a97757618099e8b6d8662cb3f91fe4164e8262ebbda5846c1736f96bc6aa36a9e18c253026b7d11b16c16cba1a5f72f6d026937e3dac1f5f3b55061cc9e864ce0f5e20749ac9a1ad0bd83a55b22c4ed33f2b0db47af21a5a7d53df4b0681a52cb6a8ecb0424794344392507cc81416e1df4434e1bf6a21d782f61ead1cd1ccecd7d221bcc7c2cff738236e40b799bfd9b39a9ac7368a6def370aff8f92f86440538a47101927e322ef088e31f8a2e02355d679a406ee970571822c4a2799a485407b0bad4c6735e646b7e3fbf3792554ac87e86940664f062752d920959e16c01ee053443f5e05d29cc186d5d1b0b84fdcbf7870aa7f5a5045bd0311104f5ea958105bf8582757ca56a65366fd1942e2718a58dc1f513915b38fa153cd43843f9bd0e9c8f2ec0a1f631060cbd32a39f4694f1fcc5ccca41c06beff8444ca2c08747e2be96776446ced89770431b30876b91fdc59ebf71dfc50e48eef32e72818bd1ceb7a6e36de989a624eca6de8ba0e20f0d3a9e1ce3bce220b71ebbfcb08668dcec0e7f2b4fd7f641dbb06d7d66d50a7dca52ec308759a54440291c8b1a10376ea373c24589cdd224dd935ffea304cf85eb8cad21ba7a3e25c58edbd05c4e0f554e5ed5eca142d71da1e6ab86f4e8c476b2d26c72e6ed1e0ea988caff787b9e0a1fc0a02ede53c013aeb90dd6975f52e45ac705a134156aa69b4765689a32b8f287114da66db687c0d65a4a5e5d668c231ff48a79b3074b3d405635e73aceea97cf0a2d5b130b41f9d4f12e14b527718d445dc724a5203f21d4f965282520803701009aa9f8ad871774a33f2a4ee627905b21e4b8e253b960e722f56527d29a801994be7d3a6aa078cb20b615e1fd65df2559f768079c263b54f86974177089644d11bcfcb918cd808da64e7a4dd2b597bff34d3580de1d7f8d6792056e204ca30531b5fefaf82b15419546f9a48237dc7508635f8f7c8bc8dc238299325629902b1cadf23bf9ae9ba7543993babdb9e1dadf2137c3e10229e895184d06712393d4d344b3a188446876060536bd49d754e34ac41f78a240b869c392c8fa13d84f202576ff13d062133717c3b8b6f71410966e69781a0ede8ea761e4cf21b21d47bf1f4a8899a5607b229d79f3dd4d296714f004a03946242970e4c4c475bdb52da2f5cef3938a5c393823a0d6a9fe7d5507c4fe07f68294b98cfb63bc056da6382a2c6f2689c16a071c5af6ac36c442e91d49c41c2fd44beeec7aa7faec310f980026fa0dd67a55734c8d7e87674d673aee0231add4e9caa5ea8516a19aec9ddb90d00c8b4512e7ae55ef22c05d17486ceae3e021db6c44465790315d765afbe1fe685992f7e7edd3ea23cb600dfe9b86b1d3f5e63a82eaa0dcbad73e35f8400ab6540eb2391a8e3d49e64dc3b4bd1be18c78655a89a183f24c8609acb7e25dc8ad3668daebf9c48c920992c3e1cda2d98d4873d46a0038710a1a16b7cf9050ed6ba4ece79b35eb9bfe077cdcf2f2e24a9bb64410d55b2397af30f6cda318bae3566ff3d4e45cef965d2e4137353e3289942fa5273f9654dc62ebaa298ec2e6aecdbbc20833caa3696e87d5676105f54d07c4ac12068d00d886a1adb6e73f8bd1413e990c68bf24db5a99cbdd6d84756cc2bb3efad7e72513fa32a7d904fec503ce17557a30b09fbaa558dd8e7ac44bf05e6d0c6daa30f4406b4819422e29bcc572131aff1aeade714a4f5c229f75c9ce251f890921479644247184619124b077bffdfb06af0829bd7740b6693a2d211a45f39833cf7007365db0bad2b62025ebd0d3d43ab967a3c1f9ecaf4da6b2d1b31f3f044337aaaf150c9c8a281fc71b92b59d4b860cdf06b3aad13d563480b479f7458319b9341689edd19215cd29627e1968a6112a0ac7ade7b333ce60488325ddad8d347e1af1c1251a32e12beb361f705dfec378cf8d454cdbbc05767ace7a39e7f6fe8f1ddcc513255c32412ccebc790a4ea73b6c289aa9a6beae19c463d7b199ad956277f59603b216a96924778f26a38cf3c965410e2d7c33f07bc4fd37b5c84b1ee118eaa724f3b1ced8a79a83d787da77703b9795c714d6047111b1bcba36a90e0c0fe8c6cdfeb151d5377567b333a27b75458d9b2f9e6d33f3600abdbd5cfca730883982493ec7f91dfe67f2dda83df9d4c46695f2f50da3058fdf72e052148808d56e2f1e7446b808f08d861ef5c470a328b0cc473b939966b8f2f514fde481eeeeebecf1f67b84c2b06a197aabb0ce7d9049b1fae1043c9354015d46fc0f7ccb64c1d8955d5280727d1110436b633aca1854ac614c30fbfd7798daaed86b2242415b554917b51ba98e6b67c1e6a09908095e6ccaa960007aca46b7c6a37f9231b6d8a4a715325e87afeef6ec8e8ce8330944ce2df64dd1cf7fdfb994e78ecab7e6e5f05b6a5bf899dea59a273b91ff358e359e3735db535223a779780a6021ae37ce68f05c4f48af1279acadb6d9d3563c76f07a1b62a13f148124209708633bc38175604463dff32fa6fc89d5cf31957756c7a6fb00eb08113a1a941c0909d634902527d396b88749b4bcb120185183efc23b2bd7092588ba55062de1cd7d295501a3605cc96613f9bef2730aa61c21c2ae869c8288af86ac3b7a4d0def2e259e82dd8a8268266d73b4afa7fb89b4162025a8e2926ccfa4143d6da5abb77f8129573162ec92e836ec027d67e1be6bcd2aaf81060ddb934388dbb9de2a5e298f822525d41024e3b43fbb86b2498b206508d6dbf163631fdb145122f375c71ee678c4c2a37337392fe46fec8bd483b879ccdb6f8537d36654001a52d9b34758798919ca2e285e8354d55734e1b050c89c09c3945da195c6272310cb1d51e5cee97c9a94a61df8569b69702fd19d31aa13e95122a760c55b3f64fe11ed7b98005704bbcda940edf4eab09063d04bfbfb4fac60738f0ddf721f8ed29009d090b8c57dbb530d9997efaf2731983260bbe055ce57e374c88ca7803705e1580459218fde5001ac8d64fe0dad67f4dcf59d422d357ddd45d8d24eb7a092d6300112e6b75c1b25331a108a6d27d5ce23bb0b2fe0737f2407e16bd20aeb13b82e7569244816584cae6cb853094f22a997f8f95660306ed385bbb2424bcce9540af37ad76bf9740a058d398177327f6cc075b5728d72b6266dba485e73c4b1824bc86ea3208fa3541ce60d4b815b8ed64bf4f0fa9609f8a4f610f7abbcb64be56852de9ae9174c5661bc0e1fef70953b5179115a7ab768dc2e802759e348b8f91b4829793990d25f14e0ccbf07ef23c490a4adae6dcbab24922cfc8b080a8fee6f9ac96ece2b0a302a981ce25566482f82959afdf0469cc798a24aa558ca657e2b677e05e0606fa3620a8d0782bcc72b1c973bcdaa3e594bb8a6f4c0b77662fc9dc98f4b410a937ab081d1ef5d46601ba897c77d401c2ac8b0d9546bb57a36be56323c60c7ff1da6e1b73e3276f9dc02f8b6641b999f58f311093f547348d0f367f95039be5d6b13f534b65dd2a170f173030579d572f8639359db9607144f03ce0d93afa1e6726eab117887dd40a3ae6efda3e20d4123bc7785cb6f5df6fccc5bbe21c64d1ebbe80b1bcca78e87ac19d22694567fe01dd94585c47505899d3de9226f3c44cbb899ccb71b086626d4671eda9e1dc4942e619061c455eabc424405235ed2725b56e3a52d3fb5662ec351262b7301c8dd800243422d112d003f89a01b2180adb5addf2935e723d792ee224d6c1a55c84d716124a0d81feefdf79db6bec8dd41f262d3c07229228a5aa65aa830455c91d52eb8f334ae4afddddf9220a6367d31f4b65006939c80f81a87cddb75c388a485654f90d203587971fdd1cbeaeb0e6d8a7798fac434398d32650b3152568ac70c34e430e77a2107ca8d637f2da72a2f0afe156fea5105be895bcf2b0c3e725b860929d83c849227ec385c92360da9c32242dc7934e8587be3bcd96913892b9277602722eb147cc4c585824bb341927a7bb991667807cb5e1d8fbe2036a99084ec5c6077168233c245a3c193ab22f8fe0184e18b5cc8c44566267b0b69ef6821c545b0893cfc327b65124c41df8ad656235f5b1985b5db7ec7ccf47f3dbacfaa9886d048f190d8bc5f513f0c47dbfe85d6ceab11100eb54b3c2a310b2d69366770888c1666e8ab0622ba3d618cb5554f9f6c3ce1ff05ed1e979cfd23d8cf0b32e710f9279053669358dc442f0016773a4ff9268ca031ccf2b708039c988d75dbce9d37fc02fdb4410e1e813c86fb1cfbf394c7b2bf74d3dc0b2ba6ab44a7e5655e49ca999aa2d0be8fb097b934371b1d223942b004f87a2f95410e615ac45d7ca64b641982eeeaf6c5c81625281cae69d777a8a59cdb982a37ae0ac53f8719609a377b880c115d9ab947997b5fd6ac17f80173ae10ef1ca749f7b24195a6d37d499ea51252ab5c55910c0c0bcdf6051e69867981fed959006570d29bcb16b9dcbea78c47eaa9a4347808852e4ab9187379c099100fa68bb5cbbf999f5f4250fe770d3e2bf5e6a86726b0b13fdd0ab84ed23acea66caf7d98dedf66ddd08cdea2103391b5b86da502c470a0a481199fc1d448867c8d48c6f5c5660b4d8fefed288f5bd8a07c81206da7b3138b26221921e3e594a65d99daa1e796bb4c0d757a4327d1d71f5fa3bf50740ad25248bc5d3441d7d067fbd5600a7586786d332882837ba17d4ef70099f514fbf6278af75f508a42c50a82f46cdc1e35caf5f47bc0bbe191de8f0096ef5857c106a3fc90a86f71fce07912d358747faa16e6d31d2ae34f0f6a8291374f59a9d977411af46a1fb8c12f7c2be9f42608d5f8c3f7fb8e53280c293bdeba21070c8b74c52d16b5b8988dd85044363ae465bf878fb86cae58633f2dfc8724856fd1c4d710467ad3ba8e5c5c0a744dc2dfcfd513e3d74a330df8b203faf36a683d9910fb240e207a5b531dc9eac5c67317bb1cd1fdb044a69374160dcfc6ce55f74a91195fecf3caa437c7f49ba6256670657c74522a35949786d74d648a0d58f9c435c5574e42eb1299d731ee6c77e43ad86860abddb77064669af7795af3183685f2b4ea32b1df4fb83ec5d2c4baf6d6d875bf216d0e55b7590e3dbce0df5a1d7fe539b9b10be371fbba7ceeb59de54a04c7463956621a6bd89acab66fc8d4d04bcf3de083faacb204154770149d3903d168dbff662f48139e85e1c4ad04287d1835067cddba18b693b3c35218df1bf01ec1004d5b8456578f412c6a064714a9f6916de61bf9c4e5e963065093a4ca6cc96a13d499903442e9355fcb313d054712d8770dd2a27dc803f4cee8a0f26e60376170a8ca146a9636cd7b9c098990fe22e87863b3ab8dcba234daa7aa7134e6e54427e903ec78cd987a79c65d4357a5a62d5f27578f06b3cf5eb8003d515126a0f4c4bd01ed4f77b200793b0bfb537c720a00bff150979a42b3640880b8629651ee2437e55fa9d33b84a770d58139ffe5fd7b4a1a0b09ee7402f0baeacb537808e0a48661c7cd3b492efad5d7756d9dac9cc8093f4d0d9934ce4ef8df570f7c96ed12d2437837f7a24401d6f7b3c2c159c68472d7347d95431cc51cc651a5a1705d722be721bb4e74ae54d6ec905b1d9f1ba2a7be112f2c85239540639dbb124f436c8a00bb41d17dd5f565ca339047757e2e6ae0ccc251b9ce1669ab55033f73d16bfc5ca4465017706d3f5516745493d9fc5758cbbb1484569572a7d7349b5fa54ee2d7b9403b371c1686f784fe36dbbcf6389a4d3f65beb3d9be0a83bd23ac19f819a05e038e91032c587b2e4d36e4dd07f99917e8ac681f0c6d92c70bc8af412a68801f22c1e4dafa43ff630b4763f683a28a147383bf3bf9c441b7dc4d65566bab97c6b7f4c8a83775af14bc695b8f6c6bebabfa0896b267f6008e972f9a91f30beaead5550ad56b3ff08ce4852d56dcfcf4fa522b51790b0ec81f36d26d7fb4e3293fc293432b47103d638de8b95f5c86c15115d200dcd079de6e2abda89d062c83f3f31f43105d1499c1b938352a890a2cff660db12459959816bd95b6b5768939cf9098c350b1e67f671e0bba4e382b579b6234494f16b87d54859a239e245e57813a5adc0302b5569f825e4bb9fd7256efe72ac178632072f2715e41f8fb67a40ec068cd74a7ec1a73f6ee91be8033a16119bb663b9af7af3bc09ef9570065715872baf79c0519788e6391ba9464c5c68d5301194f1abe87e09cd5fe0f9a1c2ed5daf3328db0636da95c042933bf6d6625368c2913d39efdf45a1bb1ebeea3238f60e4b42b8fa6087d4108e9f08db5c8caf404d996fa850ed86719e3b9261ccf3a25eb9e324eb7a563f87c169e9f7616aa3413e7f3afeb45a973964a406da6a4b98070aa1402d7a92942826f01344aa90f3ca3b0e60c4d7fe56a1bc95d9f49c17d86de0f1d130c71673c2ef17a17fd052840c49318b12c6d5bc6b719fc958feaecf32113d19817ec064c7f5e3004f563f72f19dc3d4b53fcff22d126bf64b788d212b5866286ca202f5feb806ff6b98b6ac59ff8a96c8ce59ca2508edfb1a832da74b3cc5fd1f8297162347085bb97068f8f97cbefda6b6347066ecf1cc9a329f2ad0a535176a240063d7fd0d52965165ff6c865d14d7a7b93f473a7a890e2947a1458d04bef693c3ec7457650f5b1302a65bd568b36e1b2cc0f43a5f620e4eeeb5649f6db1b13e28dc7a04c3ae344446599afd77155bc9c155b7a771ea5b8976f68a60a842813231c1dc21b72de49d1334a1969fa81d1155c7786b8a5d04036c8697caf0f4fb5a5c0c9c960bb8ea07ec4551d9622ee3ed12291b64a8cfaf9963e27256cc59d0117728012d38b8ced7c09304434afc2d5d931598286ebf85063c3632ec05924e29a69b1e6204c48afa783a99318be799cd341daa246c335dae01e2a266f01231e3b9deab7d7e45c7699c887851b4253e95135666c1bb43bc4b2e95d2ad7f6a1daf6edb127255f8ebeed21f17b4f24e65c5c413a3dfbc6691ea7784a000ae85bd45ea46f65772bd85705340a91dd8d6789177d2fb767b61ced31a7326ae054e9fcf8b166c863b494f58a63c89be1961c93ec559dc9760fa87eab778091aaae4390d00626e90e002634485349958f08dac2781a337aa75a9e7fd17224140fdf49bb19e3efce118bf002934a3ab75f7ff67e9956d91f55cfaa5b4b08caebf9f589599b37b1d0afb3f22679d311deb99fd7dc934307d97f325a6e44f9964cf8efefb9290a3c2cc4e0af8090396baeae48dfe58037a75308e93cae499cd41c9745b6ad032c04936816f4973fc8adbd3d913c3a7b3538dde0733442d059011e5fc93fd22b9003fbcbda7781f6dbb2cc402ab76ebf0ac0bf9bed216df18b028ac129fbb4fba212b4a101c3456245adacf3febef19b38483717cc5e813005960c78eb49bd37961d8f74cda4f7bb5bf5a0189408466e06d18ff34745e1d1501df6b99aed59ff1e21513c83f7d0c82796c34aca01115d75b45fbd699bba68de59f4d90b0bf95178b9296cceb7ec127d5f6d84f24abecfd8adf8905c7e1f7a3074d1e81b4fb7f0f35ca00f1017cee34a0d87212b3f09b8f5c08eab9bbd2c34f43b6998bab5232b9a18fa2fcfd9fc1c7fcac6ca7a5f7a5ebfeeb7829e2ebd7b0fc20879fd10699136ddf567d38d075d0797afca5c19149d98d2d248d53a491f4224cfcbec20a15dd4509bf36d4ee98b31c1baa6c2e8e6f9246c9edd315f540ad16234e368524a4a47a906744760623bd99bbb36255ee3880f39b42cc80d4a427d99d31b6dc2e94e9a11c52d513308a5f958096872cc6d5a7e3b1d81f8d013c1ce370bdf754ec82fe6c04761f329a6e4934595fe4ad8cb3eccdefed7f4a6dd207d088729a798ad0a3be7240fe492a3656fb477e9e11c6225923e18e6fc5b8bdce002fdf84e96ad7f8e14417c2efe46306678ab6221435283a4ca3cf2ba010002089d034feb6d69cf7affe2f94b4f54b5c5e0a56a7bb4e239f39e8dba3f540930d5194f6cfc21c661aa067d0b9696ab1375d468b3cd2ee97a200ced5047cf10f25ef48800adb118a057bec722f48b40f069ee4d874fa0eb2d7708f33c2f358b70df31982178756a10403dba7c73922bca3fdb98ffc3242dcbdbba7f5e2af71c5122763e773209bfc167285dc12134c909762d2db92442d18a552aa225c9b160d9d008d86c8083a9d2ed899c4d93e594c3882cce3b20bc7f8e4d2030e11e697dd9c3bbf6990c5b622c5bb6b5935be67ff8954718409395a46ab18120edb1f69107a26d6026daa9263f5ea95d0f038ce6937315a964267e06f28009694c65383c7ddc10abdab9a68f057046a4cc42b1d65662e126eabe49b55e3a4b59ea4f3264e2980f7d0108d1d58b3fdf4ce5a75747289a5054ba644864eca57b1537c9cc9f246b866f3e3c7b98eae7374410a2a908160288bdec5b8b3027374aa4effa93bddc8ed47df03a0c231d809c66b2947346f54bcc5caa12d37d235b1a86d2315b894e210cbda07405d288decde1ca78d7e59c16b8a548caa31d575352c5d773c6ed7fc59fba5280249f83e63c4d4e18dc48492f5c1288a5c2d1401ed2d8543559dbffb4a56c6003073173052d665474cf03ec0e14a5be7d948acf96ec25dc85103b29ff0f60b6a46266a443ff6d8ea8fa377e344081025efb8bd8e0e9b18c7bdd84216d1bd5c12bf00165e33b3949e384a1725457399606fd74c2c11eb9aaf110e7ab80509a1b10dad17169216bd5fbe8db2ffe42985411ea0dc98159ead1326a75f03f9b69c588f72a48527480540dd79217fb67417fcfc440b2f16d46b1d53d5c1f7fe789c600ea655e851bc8b3b344ceaaec4eb70fa90613b638e8b759349b3321526e6532c96dc9ff4c7d0fd69b4036af06e1d18f80a512124881544ebfedc82a1df6fab31047e675c6692bfe32685ef20f0196e0fac454e12df2ae5e93cae111303d896c05d90003495392cf3a529d1318b2cf8633c11bdf5b1d26706b1643cb7e3e869fbaf25c8cb4fb430ca3c228ba47babf65d87e779c392d101dbebe334920203b13840017ca661140e163d968f2546568e72725423d30e748b41d99f9b5fa83647cf61dd9559c24d7e86c3f3d81b60238cd7071b38908f3a4a172f3aae2c8d25908ce70d7d2222faa2ddc393568a020ed0d5342d3230adcae0e6e2b1313bd3c08c98daca039f7eb7977e751f3e669598f4ede34fa0184891030c729f16e667792a1d65c871afd294bba0138df0d2db04d93e46fa67a14053aed2a0a4b54a9735077d0a4b63512e2ca0fc6ac5594b1ee4a7007c4b4a97f1f10aee7bcf1e016ac8d3d1f2c4a8a63894087dd825c1ab8c6f9dd5a88aee3058d538693bd5aba58f374d42212c1dbc482b6a9c588bfe758b12630b4e928f34b2bfb96fea4a5b095763542c5cf8d9f56fc699ef4400e0dbc5ee3a9e5ac15b27e1daba245f546c72560fc9c55bf9b895466635673f5c3057009a0b7fd96b29007c1e7dfe3ed2bc21a19a81d1946c14f0bbcbf08fa014d16e8838033344d1bf5ff3f14311348b07d6579786d0be9f11b7186db3f993c4a7c29b3bb2ad8f5283f79084901d6555b2875a705afcbbc77918562e3bc686f04359b84962a509cd058f8604c909188d81e17748f1c6d90b9569d593640fae61e1e0acb71610f5b888513687e589c1e42f0f41e4cba1afdb3a127d562756d5d1630ca66fcc093603cea34bb140264014fae2d62e689a5bf2704f30d48d434aa1a56407d3da0c2b3e4b158b9007495f1106d16a4aae5dede6c45348a1195d62f9f35350b30a24464a21a6d6b0da051ecf3c50beffc79d1429f6d9638a0702a33a58a1b5b296ba28059007e15e9ca5bee4883df232abe5f952037561d3a674e3f13d9887c2a9c2057f01490410fb01d42b75d6dc74274efd63164c445e01636485011f7dd84db0d03e2e8e75fa61c7b61501fc8d1f1de9df139baf2fa1f98802963933add4abed6047331f4638bb97d0a899afe9e2c8f136c979068aa79e64a978396cce417a695d97c021452e98a1167f052d89821ef135bd994c9da496e171d2db324f1f0082ead9533b8f070ac43f6bc0b35c5336c0d9cd97d2bb99b98d985da4715f2864764926a92106009b6d97e57d83b06619496e6f127110696b4e646ca28206f5e8c66e31867272de36a688e415d42a48b11e1b71ce491183140bf2fd44a0008261abc257b2fbba62495b2aa205d86969446a4dcd150c168d1d7f12de97c670e373005275d75f18a0821a574022986711231829f62f2be21631e2a751a0f5b9a7cb7a132126ca7837ee96c5850885cdc769484ed9e3c2080ed4a6f883ae01c3e1d61aaeddfc326ee582b59ba146d165ca6b0bd9028ef01c648691f931f1120c9698af2b08294e1979d656db108a19f7ddb437d8ff04be0e42f4fefecf27c149b29782696ac26b6e2a53c26d22689ba5963b5681decb735c1f32eb5432eea620963a03daea9bc29565dfca114bbf29c5450631e212743a8a973402e82af9941497a09763625176c64a345c65b1aa128eb775d83fd23f5948ddc7d5c8523ea954c775be11cf63a0f1e7d3b7db751d709929d6302205983b85d9c04b50a51dc27a5367d8a84208d12f21680c1828511ef275230e744ece16f83291b1618bff0d06a566d264bc9eeed065f38ab34ea3aba65c514f3d0f243e1d170dcf35c4cf3c0d1bf47acd253169ee0142ed39c8639a1c85f463a7f2a1be86cfb3a16b44cefd23038b5a52fe4ec05b6647f2b953661afb0178bb38e657c3f70bb758310a6e1a3b55c539aa2eaa7aaa9569a195580d8b308142b23928c9fb271d6fdef685ed777b6783c7ba467f5d3a3c72db3a91a524952808ccc8eda52460e19192a5c722f9ba4fd3cf106bc6b7d3690dbf7285615575370d0d8e92690b3a1af9c12e34f247ef72eab0c6caca4bfcdce40627191cc7f113fe1432c018e6f016504a5e30c2cf623e89a06e34789a10db91944e7d76cfdb799c125ca9d7d2f05c1e8329002aa5e33a0b9081af2e12392327bacafeeec09b6fe5afe7ed3d567207fbe3a25fdfebf9a8c29aea7c12f01ed2e1fb219291b4adbd9f529ab78871738a6e2197dfe6eb825bc5f52218d7e5d93aad5e54b7c876b32baf0a322640d0fb5cf042feea8c1fd216082fa500639dbbd1b033d3d586ae09efa16aa69631c921404a6adb97e3892a25425538ab7ea036be11531c2543f80cefc0305807582e73fd0d604141f433c524fca46269deaa919a6e94ad08f2b199e7d0fd01fbe2d53e6ce5cecee68fc8e570c838fabea476edb9cb310567c2f9f0dde9559dbe86cb115f14446072ce901f6fb5cec4d4e72a7c302ee4310592316719289d722718553c23d19c2639807fe1315d7ccdce350514fd28f63ab7bc541b4685698808b989d73572e20741ba0590070b3bfb4ce5e5f1a0804532d923d06d073b5d0838d5420c9f7ee577c8a6d7744843ce69f82ac2821ce542ba67e24a8963488d4d48dc2bf1386285d9f13b87eda6d80709d125ac86f1cdcc182f3721617600890e0f6bbe6483f4c90372705ec3b1b0d251adc2ba4c0a222d2fc703791d0a9a3284bf1b45a641abf44486d190cbe3cd29ef63488d7ff26bc7026d52b79e459e71de373d0b70532c4f20683d8677898a835d0859951a561dee9633e757e08b86ed4532681d982dcb1ba1b337bad7b23b1c27eca45ca29464a574539e9f966d493b99a4e4d4d291dd8ee27d15f3b6df0b80b5b521b5b6b2a3f325f05f8904d6a6c4b896fd01940de108d2579ec6cac9a57a466c6e515fbbd86bd6543c71498c22ab582d0355cd9b8c0a23babe66a0f76c70c0d201be1c47a97bd082c1d28aa9fd5b08f90a8ee067630c1925da0f2b7a4fa42d33b2e787c20ac2ade35cfbb8530e5e0154650a24000f8bb5c74e693b34d81e4e0d6ca2884d07844e8aebe92fa44e1a336ca2864653c14eec4f1393718529951b4a8cff69829ba63c8fea426cd511c3513419e362628af15617fd3ffc43a6109c123bea5d196f1cb5ac18b6fe673f5f098beafa5a59c9cea8bb89427cafed1ce7ad13c3fdfa89bc2343da260b0dedcfa6750e117171ebf9f6931efe0f3e2a65a5a27c6315240a83ead59384658bdefdaea2b756df32d196a2aa858a1c4449e9b87fb3b8d100f0af98787154f2ed9b4305930899e9e6ff9c2d09630946ca5466dcae376343b77312af4db586e696079553f0c39c74facc2b7ac319f97d6912cf7299321d186471892b23791ebb13ffc1b43a94aa5705a6fdd290a65209a49298fd6565c5f181b18a6b81f5b98b572eae164d57e589b4942f9cd0b4299df1bfebc47d7ad94d10826c4e46800a595cf588ca50ef2ed70e617e6924965fae9522e8f1c541b96c234361efb58841b2c68b0cbd1eb04e7b1cff75d71b098ba6e1b6f11b3309c8864fd44cb639f4fb9f3fc7c8f4753fe31b245b866b61ad8a4d5a390674fe1b807e30ddb33c0c4da7fb74dc2face874f5a08ccfc6c688da157cc31ec0dc353e4e4e4a4b3b19744997d00bda7cd959cec07470247bd799642b8ebdc3fb9d5e65d807bf0eef9ca338284e7af6256b8f0cfacf793956f3972873e4ad40021f8d9281c902e6a7eb14b3f491b716e7f011415a0b261b705073ea7f7b192cd71c5170f6164cf0b3677e75b9bd7c71dd186fb7420ca13a39e8646aa91f42f88ae2d86b5fcc48bb6ae45c1b94a5cadf1a002c2bd7aba257ac2458f36909e7e1e95c190e8a86f63ed57399ab18eacf91350559c33916d2abc596f5860cc79f5bfc4d65ebead08143d3bc8faede6368bf0f7f57670b12158642afc77b0e2cd0d88604adbe97e7fbcc2216bdcf2b6e4be62ce7ad15a551ad307d49e050a8b8de634b7cacd525e5ce20fc4bd0f7861039ba55525e3c6df9ba0b07046cd5f8430f60130460958b0769c2625d7bbda34b1c4ac3195d6a892055b724b562b4820d8acc1768e08325051c50f8a6e0bcfcd3933b72af67bfad703db775b33966de581e892885d566328714560e3d8dfee47ac8dc9a6ef0a9547c961fc3e0424f9f0a53af280837b3fdecd7efef4a9ea74568c416d7949ba6c0d2be7574d280a4e193bb70ea7758613274a77e67dff316ef72a46b8b2805dc42702fd344dfa2b09495b64ae029a2758ee9b695759e9866a8117392ee29991f6cf83c7b951a1cfda7721c5988fdfa2425e7e4ebf0b1ba702a587d29830e43ac39b12bd1e8a5932dc0c7981af2e36d7a55076b67e0a9f64bf98fd05e5e8eef4dcc2dc0176c6b97a800124d7bc7d411e91919a836ee081844c66318723f4fc64debe8798cd66be42894c5ff022ea5977ad9249cf9e556f3b857a5508f41d6e3a2bfdeeb564b5bc76ce1e1025ceb8a0bd62c6fc94c1ad281e2f38b59ffcce5b60263e5e613adad51c4b6c528a5b5a9d4448f0f4837be73428922eb0077f661618fd33c5692641730a94304952da52c2419c9cd8e1ff8ea25bc476fb1ce9145c8b71358ed447668e13d02dcec8c8cc86dfa38a27c87cc643dce43800915108ebbfb78f17a2363f1a67b28a0100ef49bc9d57337f6b90a24465160bbbd2fc91ec8097585d2efd99ac0f7b5454ab014ab272742a372120e63c0062ed2f9fb1745c61cc702fa08cc1b428dd2f4d3081b042b075897f66e0275453a8accf927c405fa35728ac75e42ea4841721d7219e8739b69c351a6ae6c1a77131ac957854e56d6b513dd1068f0cea7282eac274a8fb5ec4fb6108c7fa50242005bb0ed03721d8687e664c93d9a37aa64b64a01e8710626fda681b3f3ca1fedb172fe54ca12ba44a2602a527eaaff3255d2fd826a5c6e93b4a82fd8c6ff95f9028119c82252518b8c23aa77deeb5be877084ab887b1a118a1082a225acea52cf56384616043bd953b4ff4edc213218388a9a570f5521e6e3ddf5b155c58879cb64e9bce1a96b825d614b0986c821c9cc51995325ff9793f418e69d8ff0084d300a52fa57b524c497486eee3ad9517f1ef845300aadadc4f9f865e1a9f1a9c848547673d44318f4fc85ce34b9fb1b4d3e662a8f8206f8434a6da9b231dabdc4dd8e57a9b929aeaefcdaf3683272f1102f514d39f58dc218958f19f3e2b905a8d48b3096ebb0c496527c02517f58d6d5c99b2140975925ed9fd1d64a3290ce44876e2bcdeae4aee32a34740c68479e019cbb67383a929df6d17af9aa0a74458f3658fec1fac749050582657f798a27fc7e09442023664472f575f3270e378a682bc7f11c8baaa552a5c8b6e795f37e6a45ec0b2617f5b6f1ab3b2db3a9b2a209c401c5e8c14b133fbd5d87542aa949bf3e2b17938249a5805e64436b73a37315f95a90ad9154de74979ac2a417d4fbb7683a98a02543961c900eaeb8d924e80b431d93f81e93fb210eb67796c553f1002a3df31613728677235e4d3f259b71b0adaa38e44853272d87f2de8a2522a5c0b957fbc819957530a70a9c61d0cd0304b8c4e1f0a22a80ce8443440e7dce257721ecdba06372e53fc6b4ec0db39b4729d572f47143d87a42b95e85876aba8f6d98931d12c08400aa8d692a76e16153b0c65018c38e5d906c46551b55e6a236b94808ad23b82c908a850622830e4c429c268c5b4e3fd107a9ba95ccc5aaa5ba3788dc343ea22017fbf4f4ac04e28887d6a4aa1fce40c49787b5ddcc099b64d831520a9c1d733893c34a19e0cc0766a01c6c7f81e0ea506983a6d342980f6d60999968fa8ca14fa63b43139f2b44027a82031cb8440f2f69bfd0d461853257a624f6bb184429ba760f4e70b5d681f99d856a270f784888ba4bc9fc218809c72e37ca38f50145e5c0d0a059862f34ca03fec98aeeea797896cb8f6631af069d42be2bb197c8126ec82aa3cd596b72f59b1692319d2112d9cb618f6434031a9630479283a7cdcbe896c5259ec9abc359d46caf50cf9353a24b47afbb7e34a78ea93c697d3fff0f416332a21ec64f2f0775fd75cd1212f6a325b19fa275d364a83d9a9521737c194d3183ad3e55ad63f90510c1c1edded7bc0cad6432c3daab2d61d91c72e7d8dfe84449f5864f5526e5b30f62bafca6f661fa99b309cc67ca33f7c93e7759f8a7cee04689f0b883d01b1bb5025553dcddfef5967f21621c136870fb11e32fe1cff78990f73a3e5dfcaeb73ede71656ff120d4244ee47588a600c9382c32feea6520deeb5ef4a206b535496577098b2103ddbd81921e67570b857bf4c53aa7cc309aa48a289b3d14cbb3928b90633f71cc7842826a9f62f6a995baf9423c8e90d986fe30208b99ab516692bd9c6992ce4fdf73b1d9084929ce4948f28304ef6d76d40993952956c726ce914f9bc3689b1c519a6351bbb914e2033736a323a8aff13006d73ad8ee217f8cec24e0c802b39d6ff67479486fd2a7a33925a277a309f1b1af944fa8c5d8ecaf8408d53416b32d8b65be8d84ef16b27266ad56d8190ba25db44ddefda6b0b00e84bee916691ce2f290ba7dd0e32c25d72c7bac795395197d8b7338791572b75114c2be9b2128df2f172e05efe52b16dd6af6ec73335654f2836fd0e5a7679682bc0e60dfa6c2dc3b835c990c5ef945839649ea1f32cc55fea8543ba61e0d0de6c14c729f5c96b7ddbc82d74185efb86f770fd25b3efe9100cdbd7ece1eb4eb322156ef334c156f49e058700ccb985ecb072eeefe5adf4fd22b2e93716337d5316b1f21165a88ffa23a6312a4d2b45cacf460308787ffee349548f3821315e1ff5042ff832c57cae5cda12ed1659dce02387d5133818ab73a6198f4fb62252eb1430e39fe002191d15f5ba9e8110ff0d70d5baefca08c994f13b8177df49b13849173b854574335da7ca8c23a41a8bb1303ef83943deecfd12deb6c968370d575ead2d5aba8402f030f5a099312217f9c88f69dc3d4e667504f5be0bfbcacd54daa9686efc2cca01bf3aa53307ec3dd52d55990a6a618898c43ee4f4f57749b91a548bd82eecfe98e087c9173eb601afc5f0e5ee28183f0e08d4d750163635bbb0d3d83fa53363b9760f0dd84d70e05214df90565b93bb56856cd71f3f261ef7bfaa1d3c07648d73b068ca72281bb194809feac2817a6128102761d1055c69cbcc98c1e680c39ffbd706453a9e801e3e0408f491889e7032b55a6425e182b7c4036e2444c9c6a897b2a06a415cb30077e818af5b70767bd6bf171fdd7b7ac1a2ba5fb4b94750f9185939c59e935c3acb46537f56d40b35d23705b718c938e8297478bdb93a79242a93a2e6b3b84af11a9981080194b662107c687f14aae0462ff05f88ff4c9745ea82df4ccb7340ecb32853c423afa192700d7ec7004730052c259ec5e6f5c5861fc625ef9168384cf0d59e4d027ea1305c3d1251aacf7b8582958ba38a3631d9be668639b80a0437b364e3fd2014e39760b29cebcae3bdac129f0b8a0d18dc81018cc0c7b0ebf79a9f9dcab1db09337946ef38ed3a497c9a9eae9398f1754c9b00e41641e47170ec05cb174abcae7a23cfb6cd1c899f028874f0b71a4e34bdb4dde795acdc8c817ddddf45cd8e9d5f71f2575089eed1c17e2f21f7ec039d35387fd7ffc493f88e7b094562d5014a83d8b432efd8b1974aa7592be5506dbc521b67e9afca4a01bd06047710725e54877ce147c5c0bc86d9d0d01a85498a72ee82d5c3a80d5be7c8ddb462930993998384b0a04bde2596e6e87937a5237ea7a945f56fda5220367030fe6810956e8dfecaede81a6c1b48606efde6ff8392ac5228fd1e60af0ea8c549aa73359cae9a34b6f771a95e260f9f90b31cee8c2b0ccdbef11c1490f8f8573f6c1c4b49ec6c6b5b36c92ed933f3dd53c6a3fa0332377a0ceb800eeb1c14972a68da129918871b9dd7731a7ca094429c27f2a461372028df5710773c125568f2c35f179cce3b789b1229bab4c129ee4ec9cc7fc9ae1caf55ec1fe8a3c2c85fb91a73067e9193880a1e39eb4bb18a59ac1b44b90d8adb71f1afcc39cbac2b0017c0fd7558a5ffc8329477a5a27c13c337ab402a3f759fdf8b97317e1704c4459a4c273076566ebec7988b341f963024fe1d0fe4d1a9f5adcda466ac9a242ed8d2923f161d46100b3867ff1b10528bc07f4e47eea845d36e479aa446599ff701fd3a129bf24f32470067efcc5a1ecae53415eb9fcc5c06c948c99acc7f296c7ade4d96a841e484c186ddcd8662cd78147451bfc9e005e51d7c3e4a4b671f9c5f4baa0ff42fc5cc82a035fa596dd61144779a99a12310a25edd4980ecfdbf55f942be5a12ddd0506ddd9fd15ca2b92ff632efada851df71f62f6136b322a98c6c0f7911b74f47c6ee9c7b52d4e77cb68e17e118ce0c8148649f055beb1860aad243356a3be952e9b8a678671c92a8e66975f13736607a8c6b26395d6f20bb65e34d8f9cd58a05deb64d0d3aab0a3c0bfc9d10285c9767767381f714ce4ebc8f688143f90759ed5bd9cf4941c2e373a53e07a1f50ed89f0d14f74fe5eebf52187906d4479bdb6e8f288cbe6ba1c2a566e1fc1b42283044170e46c9dabd204e2a0581eb528ab9fb2cb05f7bc9e3f56beb05d8929ca7a482571fe5f4eb50915fb1fd3717771284b9979c2f917b8342937ec40606285385292b125007bf9eb5df381eeea84dcae913a0037dc4256a7f8a71d6bbd623e793b8c3ba9f2db73330591fc9ffb3d57b1e69f02510adddfd52af675a846097e7f22ba9edb10cc5e809203ce1414ca583013520ad7d966ab22c908b31024cfbed3b83d6d2f32dadd778fe86b74d466ba6542aad1202c2031fd61021c62774502744b47516bac5a7e26ff6f450c2d1be3c14e25379063c5e7be14f33c9d039eb4c014c2164cf57d55b6162da32a29b72d0af394c7dcd7716ecc2cf71283658a0337d731b88f6e5a09b624cdca48532e2bd820c50f23cf886d3ae32af80f229120097091b0f8de5b9c0174a87ede3537a6882faabb3e5781cd3d3b02998ebe3d399e11779872d8c4be9e07c3b6112dcc0122ffc9d2a004ead5d21a5e23acbc82ea14afea545be57f70abacb9a9151a2bd088fb1a21604767f80d5dc286a776709e354e739e2c7637b98dabb804a786eae11249659bc6182b7b0a67ae8f5ee6d8eae9e7c95c67b77a800476c9bc326710bf2b1fd747824107bbf566d20b8b02ddb3de4f03679548a50841e39885c27a15cfa6010ff53a4fd2e7eac038b06ad03ed32dafd5271b94c83c02bca4d723deab2418eb70785bbf5c62cba51367d8d0581da64e8326c618d7d6593bcb77270a4efe30e66568cd6ddce09da1ea71f89003d4b6aeec0f3d657041ca85da60d8f17019aeb44a8a8dc6d024accff0a95cd70328aecfb61023b7967c20a30e3b3f620762b58174e742506e0e1b1240bcbabac0ae7e93f73a7c6f1f969fc25abd22afbe055436828b56ed709327332801de6a9d4311cd28452ea417dd2dc298305abf853b6dd4c854c1a690fd407b45fad5eb4b421b0922aae69984dbb0488622463b03c40ac605fdfb914a998f8e4c64fe7b4e981b2c5a8e6144d8e3e962854aedb9595152c530a95b97ace2b340a7c9c636f614a8db066cd0d710f2aff86f2c203e7bb83d1c57de8020d4936ef10b9d53e89835a0886102a70b5f6dc7a539e69acabb027c681521f823cff642e2c8c1dde208a8572c1a48e03ca93758b937cd750a484c253b810cb44b849eb99ba2e5e1699dc0b07f6086bd0a8a56bf82b1074ecc05964fb67ecfd2223e2cbbc90f6ac39dcc9580b0d15bb732477358405d39a0a73aed10a160f9da41fda76b48353b674b8c42aec94d3bacbffe305677ffc5e8d8e9c6b6470b32c32eca33f5201b5373444f3a1b2d2ca7444552568f0a63027235941e4927ad864efc7d33f753cebf4184f41924454987ed7c84fa03c26487a7a4f1c7d4043f960c41074540fdf83206cff1fdac0984770880bc7bb76985ec3fb36a79c75b579e55bac76f16053d22f87d1f029b11970cef76a2f8256780ab6f32e289b24fa1568a4cbbbd611204c656bb56a7fba05d183d9b3d6123524a000b1e2c3159c8854c75946876253a72a864005c97dcfb2ca7218e04c3985ee4d95cdc8a36b14e65510aef5f6419c5744fee5d3f6110f241f1df649c650fb43b01b1c8f440598add81e9758549c76fb7e7d2ea0d9f89f01d2b3cf5ff702681b13428f43ae46eafcda71580bcb36593a4c7b7cc9e6a1a9a8116f8c81493b2007a26205d92b3a0388b4d015e8e1c07aeb1a8daaad3da4e7aecc6a445ba7d60b0b25a3b8005f2b221a4aee638bce9b71d7d9056194bc57a6b82dca913bce1c6a67076c2ca9ea9f603365092ecc1a272cc96d7debd7603b00f8b690028caa4cb6d6e79149341b4bd316a450e148b9746780e261b1478338bafd55b775a1521daf5430c19d7ddb54f1644989e05947eecefc8b920242f029250cc8257e57eaf8dca7fd7c5f5ad0d3d27b5938e0da17a01c8e8b5e8ee6859b57f288ee89e29a9f5c5b58011b6132cd9142d2ab1f40311cfccca2b895d0901e8d3a5bca9e5b0e900787b08958a4896d1ca2e44db987db3f414c8c2d87c3ca693f9e7de53a467398805d33ea9cd621900262972c536dcf4e6bf66f56fee83c8809e7e0a61420e1ebbd9d4a64d907c31da07e6d6eb6b19222901c96e60e408bc574396c1630869df8c01f4165cc60fa8d4d9dd5f649b3fb3318b0bdf437870c24a72c03bf8f74ae3066fb866ac8aa2071fd3a301487962ce192fedaf90187be8e4c9fe0fd5cd5164543f757fd28c94737ccaeda5b21b1698dbb5d77cbf594ce4fb5d3b88748ff6161a30c1084c68ab4f14c25b83b0b444ea77f094f123dfb518b4e0b67f4bdb7803825e44c8cbbcb66d013d9ab3d8b56ea5de4661f1794b7718d54933129b1e1e49be1572faf5b545b5e16c3212e2f82854327c0a25050dafef3cd55e1939b0ba4c5c59808d153e5e3ae75f0b6a1f3a65864099f22c72a3a664bc66abb1c5b825bab0abda76830521fc65aa54f887b5b585e1b95aa702ec9d5549cd6a64d6cd7e27a2ca8cbe76dd1fb5b5d8a45ed31b7cd76d3ad9daabba56e4e0d513b749ef2c00a42c4d0257794fff7b39c46c567c4be97673ecd37f466a570b37ab5745ef27d4f5a918f32b321460a51f1060da399df00e2a09c4e0264c4070dbdad1aaf84217149114c7c65c7bc115b3eb7d3144341807afd8a5407edb65ef0639fec96ac4187799c14e960111b7c046d84e6c79c543434420a8e3e9f15a7c672d8da655690ec281554f3eb2fb2cded00ea0a5c5f3431505fcfdfcc0281d34487b32590a1676a35fe0ca112e69d190c152f4d9838ec7f5ec6e4493ad1e348f7784a17ea8c1ba5a7897cf51eed143b58c84546b71b71916a52528bbeaeac381661b1cdd7f9abaf221085d43b915951780698c5fd2fcb40e2323fd06fcc9115c6f319082ec214f3c09055fb3af916677b4b232227661f2806d95709166b836a30a6cf23c6ae08b27020e08abcc0e2fbd9fb57b66a2f74659aa110330f67e486433fb08fdfaeb5dc32e4515adbe5e91f429a504e77483ad1b7a3ffd82ce3ca89678de0e34fdacf1523716633f1414af0332fe1e350d7434239410e3fcc59ef55c4ee07d65644437666e299f7c9b8de8d31bee09fafba19a1b3e5e499f0423d5ce585280ce745f0f4fd97177f5955f508983bc9ed310f4c1bc81da8a0ad8533c58f5c7b1694340706cd2ee5ef40c3f39b9ccc4adcde5f84a7f34d1db10be6e686ce33f502020e153831ef5af685b6af485255c5f6fb883d1e5f5d0a784d88d981c94c6e814c18dddb912173865226ffdf0ba5b5246da784af13651a62a894035b5fbf5515b3329ddef70729c14cf43842004916b599d43ed1c64b79908a2aede2d94ac4e101c8824033f62e92fb64bf0e1489f890f0900a5bc59c6f07a7c427f9fac94cdea070a5a144f4a8284b4858566db98e540d0113826a824765c5ff15f8b153d81dd0804b8cf0602203a870c0db766ce8c7b9a0b01567ce4da1b307d20e50d16c9f3718ef6d44637111f944b15e25aa1c4006fcf96826e7c44af1e88f14b67879285f6762eb4f70d0086640cd88a6c285f2c5db3f32ca8fdbe9c1edbec70fca4fdf50f3c6d3aeece196390f856b6c17f9453823120382ec5d47b2b343c45c5e153bf78482db975921120f3803902d9680aaa3af4a007cca997c53a718522e017b8970a2c3e4df0aedcaaccbb1d2995f4ee4c917b63e8160537a1decd1255e0b4682c1cc9a421478fba3e95f1d61878b52435ffb7031d31d292c0f53d108ff0bd83c7069f49d69b2bebbdc0644bbf85de41c5365615ae6f144cedbc20f2aac1e35c03408322b2492aade903c34c48b28984bf43ff72ccd0c8ebe8b641b9c8c7e277e4efe180fdd4336d2d8f48245d9ae8d161bdc03c406e7f9ec05c841aae12bd144f41cee97e30d8a2ec2049c6aa043a575f48a7f4385ff79001be940ca2a48683cfc6251bec924b3992fe2f4fba793918a66c1521a5a8633d569467bed022a83585ab1695afa1da505c90832f05729619770278f65fce26209c212625341b20a94f6530a42d86f4de8ce4b5dfecf94eb56197fa1875c31df49ff7d120395afb973d4af68db0edbbe6fdac279d5a33fd3788de07f7e8e42159943a7d9dfe8b3df438088f2de1b45238e09dd2bc7b57d51d604ad7621291b4004992b3b721bbb9dcd861b87141668a475d7cab6f1cabb915a8ab01c48b79592e99413891aecaa59b8dbce78cfe3e0b6c23e46ba1a005094d8e852b1bc697546d785b2cf777c93d1cfd8f4d85c0d0e047a077d336ba7ab43f94a5985165c9b5e4c8f2fed8071ed1831d38a4483f1dddde0d0e11a1dc1e14616f47ae1690d193e967d0679e4fb347f88af6919942e97af6b589c6767f2b90cb41564e0528a3babce8540d92f0cbb7a9bc5c917695a3fff758e112d68b0e24f03f5b66eeba9bace8489f3cc0b8bc4a970f637b8729325cee20498b1460139162d03c64f12b7521eecdcccda0965ce3d9ffd8107530033b04c66719aa416f2b6199fae7502508d90aa7132c65256a53d29e94ef29b1ba27dbbaec20401e4e728e0330d1c8a578bccbf6d40731c4b2ab71dd2e66b8d06b0b546e98e50cfb7a7dd473bf0b488f2e7accdce01dfd12974d47706b8718f87c9f371254b5e1ff076e2b7eab773722198820633a8a73865f7f2e7f85978ac2e08abbbe8637f80525fc5e0c72235e007721cbcb37d24171ed5646e59fc0b5f25ef4aa0f244d0f01267cc3a7cbd2b0e25a89cab981fc2fa283bc98a78b87539afd4a96f209808a253b3ec2b19ac89e049db56837820dbbb6ccc8a37744535d9f5eb41ba508519b8fd0eb8187b406c40e1a5300edfb5967214d337d351e9da00a38dfe075e58e20a4d0ced140f72de900c763fbde4ecc0b7f3cef43317df725771c607a4e947ade789d454861fb08785efa581ba7af4737480d83f294e445130ecb7fd345d8d941faaaafeab88fc68ac76e2fa25c50b1ff999a8b5184bd3bf8215e5c6b426a4b7a12332214f80242c10a28f219743fe2e87c31d6dfcb2548bd4a0d345acafd2916e1a84ee2cfed4e8538e3a1070c74678324c3ed86d35d04e77cf42d51bb44d477e0da1b04412676f27a711f8c210a4690bb92864b8a1f47d8d843c0fd8c1610dc286c5603f435895af68f8e3ccbb10a05447e064cb5e64153510189115512d6556eca8cfedface16684122a56b3e897368360b65fe41e58aed9bc625046f739b816219446c338c27246a6f62aa24c393678c39ff53f45fa6a609dfb221b46c5ac667768f263c2ff883ad7c4b051418ce4a64700eff32416f882aa71b58123368a703d1698385f41496e17858b336cbc3669447db8e072f870368819a83ca990e246e58dfea891c141917bede55ca3c460490ab54afe6f38ee3aa5a6e2229c63f77fb7336826ac7bd84bb7bc587887d2ff35e8db2710b52570cc368bbdfc2764008cafdba9f1025213611cebec0c3004c6524d79e2c54fca12540e9722ea0f2a1301397dc50418580322676ffbd0d479ca75cdeb28f105f62a86c1c973bfe4cdbd79dcae8abb76f61510ba07e5afc87959f51342c8f2cbecd31b6a2c51e4a23015d94139b9c17720eaa7f90313c3a3719f8d06736cd1a3cecedd4b9e866c6b93184db1bab5ec1d6a8dd680f0025b6318896dfb632c0885bcd82d34b7d1232ca93e0b08a558daef08c65ccd3a8f306eb78f8743fa14ccbe1b1c6c4c81c1d3acb40a20b7ee7a3cfa09459024c839a78dedd0a45e76ac508327eb02261a5b49999ffba09ba7b9db72c6503176c1748867ce3b38eb28296e43220b03892722ace3b965fde13c8bc9c04ea038ed253716e078366f6b3664f2ac11e6ab92bb39b65ecf6a885fe7c77bce6f0d9015d72056933b64456dd14d338b15d10b62650075cb4690a32da89ade6f48035eb12ab8df6db7346c123673190887aee56f5a2d7859d172ddb956eb244cc5468cec9324817498d7a25f86719c226cfd429a361f0d93006e716228e7086286a42e8b8ff2c429cabc7043b0a0f292868bacc54b25e756b968be10bc01671fe5b54dc477f2f3f97595b3895028dbb4fd04f02e24db5b98e1c844a0901ee0a752272b79b1098d20957370dd531c9d5838eb7902e080075d3f0dbfe47f7b2862ec8f807197ba07f36e31096469f689524944b68c7756270664d4387abde7b3e32ad943c8c5c4e9a84300a4300fccd83df9d58455f8da6ae54478f2ddf4be91fa743f0f562b6281a02b3f8b2092280886534ff727659d30bb2bf2b5451141a4cb0ed8607f38a468bef806d39a9a91e91b21eb67cb3695553aeea07ca155078a602d0bb9393ce4f633e2e2526353feb43fbc91570e241c359046fad23627f64d54f324df50f56abb6eaf3aa6e1cc7c1886d11d75cb75f3b84ab3f898c6a068cb01d3911f19741442a2b02c01044ead1bb294b5f56022954de7e0fa995162df721200402f50a51a76c6e71a7269cec907b2bd66e664825a0858c20a295dee464aa996865033cb020a395facf140aaa2f96c90fdb71776db7754019bbd36fb024784e4e240ffe787350d75e6b0f06c05828a2ed14bb2588dce025a6fab913bcccef863f3b4add29966cef082859da60f03cc9e468b863d51cb7ea774c8c15a261230f9e9bf852263879e76787f82280a9739014cdf6d1c53e35c7b6bd998ac2647538a6f4bd372884256a5cc6eb6ee43641bced83214a96ebbf31a4080232d3dc31caf8035cdd81481e05a1b4af10495efa9374b1aceb3e0a9076ab1141b5954342bb0fcc88ee25395c03d413c169757773f3b43b0ad49ddfe021827fcafb2ecc8b3d45649f2c97fb5d107f4f3332ec1547b0a63075a8c9e76cf3af7348a69f5e0eee8ecccf5336bc77aebd06800c801e9e9da957393103db04a7ed3d6317aacfd222e8442b7b2a37254cfd1e1c5cb3cc37d601c58c6811d6c16263118182ff5170b2b29a028370be08d612ac5ad7bf221a408e29c7ba5b2dbf82a2a027fcc37f5faea142af383ea5190881dab915640a2a24d90e0f6a8d6e485d3b7e27d68c31434adf88484631dac936013da6562a3260b11fd8e2ec69448bbd6e4ae84834370717a4d77abaf0cc71a07f034049f68fc297f3c750069d5e7b231d3c4355cf0dedfe422dd9f8793acf0644d6d6786e85e5c64e431d7c56c5d36cef21dda288a08d871215b71d7bc91563610ef5b79a1320cabf3c56eb4c49acc30f9c254a1be93e7ecbabff3fa09d3ab7758c672bd381e807f11d1007a2f7d15d3c7adae330c25f0b88252cb1fc64e8d2d69aa6e9553b4923aa5399d1436e9bdcb1fba1184c8ac991b693933a798e272d9b65bdc38124e71e47677b7f522a10c8c7812c4f0b8090500539e9cc72c6e915453f8a209d846751311d9159fcce60edf7f2b9cc5a62ee47676117c96323ea3c8047067a86c96d823276ed5c868f41198206373f049e871a7d3847feca5ac58b5560a94f423a7a61547697b5bd4ff5bba3c85f2500920267ed2c631ff46fb9df26e9c4e75c1652fa98f240f9d37c801c2ad242e0cd5a801defae3fd3c8b3604c751cf1e1aeb917ba13873cd501d77bc8ee037ab9a8555699712dde9c137d1cde6c939e0a1e04312ece31b316da0149c04d45106305eb12da5453c18cba0a6bb9886c02f363a2fa9cb41c70a267b21cf013b03dee9656651a04d52697be2b4759f12cc7b4c976bf0ac98a67f213cc973596b1c80181f652a22a1ce5163b74854dd5167878a8d2a068c53c3156128bf1477c43750e9d40c3c4e7fd6e7d4b2c3a1cdac6d9b67d8e50946fc435b5457bb2d446e6711ef7760d44e5ae7ada14ad636b0c966a19032b5d5e8ab44a9a4d451577b9b1ab596590f53b8a17be6f4b679aca19666da99431b2c3664eb2b12c06b6ad7913cfb11de184275a55fba881e1c8595fbcfa8b5bac5eff67bb237f2908527ffdda2ca61536fb2dc1f9da62213761622628958ee74ddd656aec3c0e14e1af3d36d45e608db5b77b67cbef803be1c57b369560f1df0fea03eb96d5c0809619b06328f32b8a35b798a1c9009366aaba8f97384f6fadc0538549e493aef702f30d8ccc27a77ec1482604db09dfbb7d48e367661f1598d3401047be91c69d4b11f14c20667b0bb2988ba231a3bbc2cfcbe7c985b2a8a52172ef2bf5f2e91e71cba8af14ca42986d22a9359afea2edf2b25a7b731a5cfede62310fd09de8ea154667bcb903d965be26c37bccdaff8d520b281daa2c44322615a0ebc29774a99f6db60ce6a3741965e42317b2d1efff03104cd94bbca15a05f6a83a9cc63e50a0b092e8b43105a74bac1a11af1629a9b4e13479bf6600671aa4c3c0d88dc453e8155fee5b2c49958b8f7fa2814494e28400f2bda5b40eda7f38bb88bc673fc79e4daa41e2a249f974e8a14f9a44823c8f688f237ab544b0cf7b546268f31d41f3a21faf99f5724142eac71c42ff30173e01fccc11e48632d3b7936fca017d61e68e0f70eebdb576e8e70538cad4d5bf0afa74223acddaa51189ccd0d84a313c4f313ecbd689e8866ac542746298b3499a96ffe452b7b8935949e12cd48bdf654e868f6e24bc142571c15adda0b48b0ceb88ffa443e7cb7b933245702ef45200fb3665b05101a4157835fba3d9ded2f4ba8c2ac145da7d56180edfec7c45f56515f0446f879177e8241b5b699bb9d4087fdd9d470a7273c104b297a70929c6c2a4990b9a44041c2a3f18a4b05e0cb01f3d8c2fef4a98566b8c327c2b6fb854619bc26baf354a905f989f764fba0be06771dd7bf22509e00e7993c74308016f5ff9adf229ad21b114b45c1ad4db35b616f85f19ea294819c9fe391a6dd02969bc71597c7b761d90c3f1a3dc5ba8714d619a38e1f9029c929f974aa9e835943d62c972e3c636b635d2d972977564ece3f7fb1c713d2ea17cfe755fb478c4a11458ccf11d09caf3f76567b98bd7ef54df84da9db34318dc8991a129e94ff5dd2637a14ac8dafa922cadfbf2b8b624ba11f026305e180f6414e0b07c3a475fc7026dadc8d71d89db473eeb94e6e92e96cd6ee689d04076817a4d2c44336bd33be9d59215b9b6d4c24fa0fd5fa60c36d26618c66342d9307577881c357690e79d2f5085b2ce675b77dd0875f0f2abca95f14b906c58fa3a1fbb0e6cd001b148cecf48d8d46ffe833e71bca59ea35d5bfc2eb4c3f7f6774aa8984edd42178aed46c94c47ef10a38ce30951c442f8f70c69244908cabc3f8c3146faed2cf27481ca55a2523986feb6a2a798153b64aa2bde846db81ecb0d927809200e9d1c7e0ce93195906a3df5bd02c5f3591e570fa9f9a26cdccf81868eb5957c611e4c43ee099926a3342b199ac1092611051c6e807dca39489ddab2d041fb3ab0038842d159d0c90242c29a629c51a4bea51b2ac478ac420524459615b6e7509ded4c571607a780053e2db6020c34522788a38d5c79b881b9f1a563ebd68916c9e0842eba64fc1851f98aa7b37ddb6071da81fe443bbd3c05cfe90b18c4539d57c79001600637b9ecfa59f9df5593042c872eae10c0764cc7a59abe1e94d5652eaf26be5393b596fa1122717bdc71cb0a437f061d43cd0902243d48620a3ff6f5fdad9536086198b56731fa919c9f215c393866dc6225ba163894676491e2b807cc2818022c3bbbd1922e2993beb38aab8391ee9a90b690ad892f3d936f01891705fe6162692ad7775402f9f308ad3f7314ab86142468c24a101b3742e2dd220ec8175bee8023853f8cc1f7fb1c1d978e6cb06fea5d2690fe95d1396a512e871fbf19299e33ce6c62de8b0274743b0e60a0779828a127ae7d920035cd866095a9f8bb70f5b6aa8ab603ede0ba80beabc08da2142031e21a66d2991dff6917fbdc3bf86068e2b070470be8c3ed3002a33c34e11b6f4ff17a454e72d41a148777c962e1e74dd124ba410b7cba395c709680a3fbe5e9eb2dff02142d277e25cec2242085925481fe185c6ca80a34ed2b56cd5a4a989a62bb3a5fafdd33405a7b4ba7d69362fdeac06bc5adc64aea2278715c8acfd2b45cdc6a433fe4359ea10b6e7b445767bfb0115ea4121a78513613822383d03049db73eeeda00d12a43a5dadf152f3630f0e944eed88bd1c44d256e53ec66c65039fbd0c555f74fd77370c89752eeea6595cd0ca58c171097889f4572fdf9918954182a3349cec71459c231ead91df2bff5437a96e11b156197eae301bfbd7fadd5103596bcd882d6ff795fc01687b34bd0fb4ab2d5787c7823163adc56f87f84e8dd2bca1b43b8bc6956fd596782950b407c905665abf93651ee803cca2eee9f2ee1063b32e7d0af217ed9e19216fe19e4de911cc2c7827ce6e58773a172081e4266d0f4b25f745d1ee7f36e4dd3b753b6257900e3c4dc90f516024422af5153a5f9925dfa9c1dfe8f63a6dd58fcc748a5ba09c24a120a433209c239d2507ad59bbd3633dfdc22210921719dc61d586075347c3ff6da30ffdc2fe785b6ddc97c94e30a32b1272cc54435b81a0468dc83d783f4900470d55855e5063596eecf5f85d51b32f929332b6e816f02aa716221d1080301a6d0005aa2b072a2d8ffa3097f1d2c41036af17852c6b4d2d00facc74ec638670b2ba02cb404411235493e16f3d9b1c3b7360426abaec3c7e01292192635f3e41bef4bd57600167dc9ec7ce7850bfbbf0ee37dd407ea9cbd5994bfb72ea8ca4add472ca5bcf00cfab1e1451bb73fecf8ce612ca289851c4f0dc72a108ce26be44148c09340f6cfd5ed714724ba56837583b721933aab543513434de0d94ed02a48da499c09864a39c4456b9d931f4a7b1c8211af4edf5514f2223f7565ef67f4e32b4f9de3ab3e1a9c1e75b90f6c3f2851db1b8782da291187f48db09e010aa28ffba06dbdc1ad0bf45d579264632520c9339d65da681891ccae4ebb43bf525a928e666e99af3327df65db876951a19a9c87b65694c9d0f2b1cc39b6e55c4accbbe9d381f5f9ccade15692aa63956a403a8634eb7037acccbeb8465bfd269c5cb0c63d676183b0b3f141c94597a13b061e1a924981078301302a514a1e1104a53df6086cb629a91fd66d6252133427a8bcbd686ecbf0adfdc209bb6109ffcd7bcbb11f34f08cf7059844fbd7aec31e14232de664ea2215996ef389a7d8d480cf23b9593366abc05b961aa77292a9268a25a05124d323b7bfbc09670d6fed29a4eedb858a8652bb76fde4e1f43060a912c933398447c274a1e84bfe939c20ba3657ea54a39e2065792cbf17427949cce0cf9a5887ca9584ab78162b911c98b961c5b1102ab016f07407e2c00264534584f65c688b04f50e062095234f0fb41688a9fd05fcdcaf027543a2fca07760eda4ebd0325959461bc29a04b22c4f334ea313d8d423cb2d9e310df899b5b6ce36cb9665e7f8d93bda3377ae35cf1da84c665a1691593aab4c6c7aa669fb876166cd1130037b0fbf0a3f9d7810ea7092c1dfe109f5ede766f04fa40308824e953b40b490d779f22e909f5cc2ecbc80f7649021cbadb1a0fa7622219f2ad14f80c827227ea7da268a654a13a7f213d34f8bfbb644e70fa3151783b74a069128ea4721677c23e376b170297e9b1a130347225d09788d47d7d10f45d6d5588a70b9e32ccfe9e498918f0c1e71771309b1085a25917f5ebea1fbc052f56d4beec0dfeb74fc47286a3dcadf2e41da719b96f4e5bb0268166c50ead08ffd7e5120e633f3a97a2fa30f8244a4a66fde98e0d0950a0992fa23896b4e77f06244ea89449cfeb977b718f2686a2e0f563b9471ec09794f7860366feb7ece035a0acff5a2b051d9ba38b6f2e7373fc5b544d73ac9810fe1dd5bc61a79749b29a5119433d094e02cddcef2a30e527f12307c02ea07cf4ac107047231f18562c0f7834cc71eeb12027cc5ae7e34c2dcc9088d04dfbe840771421b7a5bc3ac91250faf8412b28d395e3e3dbb7ac959d9924afb4d319ac5281ab6c6fa5ea7c2f8644953f0583e3669c283468076d69edd5e6c28fc9ba1cb58c90000ea52e7ea85c39624e0f1793b329d1d1ed72471ac81f2731516251f32fcdaf58237fa43023ca2df1b4b1deef91b8961c0b186564a9bdc344fb1a96d517d629df647ecb5993f2186cb8e32859fb3ce3a218a7259243dc9e9361668b647b0f0afde6aef8e51da7724ae2eca2046f8e5169977972571d2063c979b6bbfa5695092cbd3b43bb9755c253c970a0316a6186eb8e60d53f1d7bf6a6d43bea5df6868170813ae6bf1896b84decbecff60a8bd4231d7de16f63f6f63a7f3fa1faea7572d54b5af67b62c3fa21d1d9850ec7a2e59a8c398e720c639ca4fce73b43c049466c7d0454ac663a8b62d2f24b26e6205ee55a591223914c58efebbf83ce52f07e156e3f798f100a6eadb61b7199adace968976fe1903bd010cb7a1ccc371a5036b441a3527693df298e92f761956c92515cf958739e9de30b0757fbfc02d60f17bea0aaa077b48812f2de51846e167a6bc322a3baa9131b966c1ddd00ae7a44d0503530daf0dd397c1663c141d82567dbdf038774bd4a34818daca962a159b480f30ccc033e336d4eeb19728ecf225c0278d6bc01e2185bef20677c4661264aca401e25689c429475fa3c645004e7129cb0c013cce201636cc2b10a8843620e5e8d24e498ef4d8b7b1080da844c60f7363fe31f4179d2c4d18f14a963292ecc6eac076ebbfa819a894a2d1cc312f320fe36275a955933d17c4e5f4a6c2ddf2bfe904901753ecc097468f9ba015351362a68558ecf88f762c8a00084d6b45be66f4c86daa71a29d0f6aa1d4adf6e04b36694febefc0a5321524369af4128a5179a00500479bbf07eacc59add65e72284e0623acf058a90fd360a07e9d79a4f6fb52ce234651977bf16dabdd2ecf1202af14a246f10a62474bbbe9937ba45a5ef384ba263d50ee6675ba8197f618f20940e53f1714dbe0c9833e28647853a23e3c5e4e4343113391bf9f6e45ee7bf9c46942fd19db7fbacfb005d90d5da626f0ce770b4e5c84a6920e17accda85fec07d496e34db4c5cfaab70dd2b6084080167ff074e783d248f74ae927ce1e75d22b5326624d3ddf43ae1900dee171bff3312503bfcb465a1c55fd95831b20c95684726ee6cadde8a9b7bc7d3d70b59b8f23fb7a1d9498550d4f2fddf3ad7009097cacfbc2276744b6c80d1e6b00b1b5c6e251b2df13e320779bfea4872e22d71b6b11c18ede80c3f78a97f9e73884b9d27faa21f020e42e1e049cb8da5621cce79db6d9281e9651e7ae93bd5495f6a61beb1d90e2386306876d6cdbf5670d39e226460ee8553ea9affced257ed4328a5b797d150def2805cafe633652011b902037bc4c775192adf2cbd94183337c81f5c87d972dfca6d07c4675598f6aeb1396ec92ab516772c35203edb5c5b39e326b8af0cd78abbe8dfbd7e2876a457cd0978b017febfa3fe96f3f22a0f11e9a300d4d12260a7080d9a325c70a3439240072c08dfcd63e4b6c8ef730a1a3b7bba9b8e87818b04d4b079faee1f7fdc8db075e29b39c4e498d0f640952a617bc328030905bbbd6db5f722335eea0d65d3adaf0ac576b33fc99b80be328d082c35fea1eab72a62aa4009bd2e4ed58ffa0435ce2aa21c4baaed2a359fd645b6e247073dfe8919e1b9db4f8255afb7270a1117c48ba55e2e9338ddd4a07e1083982e49c4dd0b62879c99b047c03157ea5b8060597736635d4ead00c7c4b594c8f091187879d7858bf369549d2f18e9fe88ed6a2642bec54f7bada529fb931c4190ca0563a376326cfc9b32376106a24ccee160ec6aeab7e33757ed603534455b71f89e7414d4525c6c0cc4b3a1a185b5304bd35314487a138a7a45eed8f9378864b6598d8d88bcde6fc90f27937cbd6156672e344855994b1a1809139f81e3b7d825851b67ee39722104d6d84416c1f3b07eb55c82e349ceecbeab0fb67c136ffb9fad75ea29fbdc0eeb70e37e308192e6c235d0e38304ca4b8718680f581ab18d5eb462ecb98b564d0baa3f9681aa7c243e7fd0d53d6ffea20715afddc77c60ee5145d948a6ed7c0838133074f4f9a54b04bb6d9ae55b8cb2f35df593592cb371efa7fa348384bcfa5b3d9add16b9142cc45a4434d84190fccc3fc84c5c2fc8dfa8a9d06de3a5c782fb5c50eeb37ed1fb4e606ad0bdb72ca7b5641d530237053ec7dd1a1f000c32611ee731aef627d7c8cb37726aa8cbbc8f4b5ae425df91a4b29e8302daae93d2062a54715301ce90612cc9dd05eb7057fa216ec957940b26e6548b6710abf49eb958d14573449bafb5c8288d3b76a3423f3bf459eacb3e9e8e3f0c5549651abbdcc8aebdc8122a0422991b30f4bb4a6f11dc372eff0c80ac3e93954a1a90303799fb9498dbb06ceacb113e153c2afefa2655c1145b6a58827d47faee5f020b00ed354f0c08ec24924df65af0376a7eeda358546c185c8335771c7b66540da4b78041a02a323dd6a3298249f11a484c9c78dc1eb55621a2a7c7487e3dfb535145fbcfa69f6eba02ec50ed1ce2347dbfb9538e66ee3d9b2e3530d5777a3c8b84e74a6913a479cf48cbed8d4c573f1e48eb919d63b7fc0e52e1d0e0dec4d22e7264a0abc4e164de88b436d4acd354cc6f0212b09ddecc3a8df61a81b3ffeb4753417954020c3e986f3d7490ef7b27ecba81e0045ab999ce11de93a3bcb1e8e41562c4bdca69a7d9575caca8f58570e51498277b3d1b0703f77811f133b0aa70e28181bf32f277764a471263470c723aa9c079804aff448fd1005e7f210ffd84a1602f63371550c6a2b298fd1b17ffe23887a02ac29107780e23a2bee13dbfb4f52e9635356c61daf11d94c3fa9c308bfb72096524259dbfe62d35b45513ec0861b500efa05b04e6095a434e28818dd73a7db3ea6101d8b877a4c0a8279170dbfcf860b6662f2ddb93006ad591e408d1bffa88f34e2c2b24a83bc69f19fb5c0161d62192db2839316068e682e6254d4a5006703dcd9589c4b42538685e033c17c8d48765d452ad03cc0b7b816cbabd725f7faa7d753fefbd3347c045984580c09a09de681e6b0bba83ef9c59a0563a91535d73aa2c6665f6d597706abc1e9bcaadaa8d8385a9da93c2cc3e0bf51b471d92733d829cf59ce88657b8b824ec6ed063858bcf2b11eb7e0673c42622090fbc71487f482b9bb3850a3f14f8da49a7bfa515325ce93d0bcfddbfcce66eb24afe539ea2f4e954050d3081467e30f1357252292eeaaf39682b62b2af33ce46f107795bbc3f2bee4594ce3b6b9eeccb544ab524c55184e0d7ebb604966e5260e6efb1a4a6a6f5b08a47b883deb2073137e1652bffdc63dd501a8d6eb467dba61902643982663634ddf166aa74ae0f4f33c87db4bb519f2fc5a089bf67cc9c12d1ecac036056d6790bbb196fc5c0ea74e7258c7f0931f4c132da6dae9f4475cbded3a1d384ff43d4656a434b0c39c48cb4c7b554de5ae879883b9225525a40aeb6eec57df05276cb18469e7fb9ad678788c11b3b6a6c55f30e2005b5386bbb188fe4283c3486c23c3554c64b7fdbe49c55b737c638bac4cb027d325415ea08888c46cf22560e0f162e96320cc9df14b25df1334b1ec887e6e39b313671bf2a101b978db07e2d5b932213835f946cb5acf26545915ffa811917fc31e1414f3491e7529443f6703287ab3317bfd1f1857e4e80339aca955eb7b52c5e0aff7b84c2d9790bac99caa0c730dc554251ec91d7ef238f2e82c138c4662cd52331f8b5ed1f79f540917db1997505b54854f55705414529e6c0f748b0d8904e5ab412aa90cb2ade9b33c65d9ca0a2346a3f757202b858d85dcc81e48649d445ec111e20db5a4a273d2f4d707427334764720f2abfeebecb84ed15c9ebcb0e6b6ac77e375355e7af24460d493c8bb0c307df8f05f7c3dd6ff69a845b8296596d4d9f31936006905ac9ebbaaae7fa250e212e245f2b7980c2e9431afe5eaa98d720391b535821976b926badb58074dad60718edcff3e7ea0c3399735765f9ef011a4434b20e69b4391b3197ff7365f27dbaf4b87e19d480757e8a396068bf19656b4cc611ba4ee1cf8eb483fef7697c1bceb0ea87ce03c4227a392843d9c69c2dada310a55238ca78a9baa7376e213e17a254e1ccb00957bc4605e0321a1f1b181be8532fe0244e9e90989ff1cfed12b33c3b3ee964e9479c67229b5b229a3adcc17633137cf27021ffbb993aa6b82e4f12e30abc44f48093c85dabe524dd3804cf8316b90b5f236bb5699b7bc9b17583ca79229d071decf7ea938b92172124ec76633eed4be1c57868cb540112ea14ef0370ec693980eb3f91849593e272cf6c367a1868dc1edc3b6371e191bc343b3b7edd4a30743a94a012c0bcdd18b30aabd9eca3311886b500f413c5dff33518e86537ff46672cd8a861252a55af3a21c584eeef2103f337469c3a4ae8294585846b66c0b2197c91165d1e902c8eba8cc5c57d88a6fa053ff64b92328a8d235aefaf25c38591de67fbce81e56b3a813b164da5908906c448aa47b9946fa7de2e820eb0071c15fd8ad12b2385ae1f472eb880b0ec4be60d6ad34572a25bd4e877fe42ed4010d4aa456ccdfc9cc2834123f255d049cf0a47bd0e0cc5be60a02dc38ac2f9f11d69947adeefa22b7c419db634d8b7655c2581f2a6f697d1680cd9900f6c939ff11669b5f174bcfb50e1e1c6c57e6085ce8a1ce441a2d65ff61c9e85b245181f9bd412185835c17be50c3cb37f27d8ce330f5feeb4f785cef464a26c862848bf074e8e15b1f0b5c3778a4531353affb6e617caadbecd97c4d880297dd27ab0a3e2cc51761f853dc0fe1ae2798c60ac71b4bae067fc445736fd9da189df5f8a86c56d0a349d3c12deeb3d334c58a5654ef79520c518cfd0cc5466b02592beb37c5b484a4dcf385f62891ab0872334327f914c31fe66b50e451b39080da1f12fd92b66f4cc0f621995f11f204a200f3a9308b9d7eba509890483b3d22e6d847e6027769dc4e2b46df028c4e4b66ca7ca65c4fc73b9875a6f0f360cf6581aa872b3367f7d29039166a6728f7d6b17e5d31467c1f30e523cf6d7e7e92465a5b8bc8ecdd855ab8b34613f1b31ab3092949338a16f9fce985044530ea522944999117c58d97b4a25bbd7239941cb392dcaddc6d86da632c34e6c1e26a370412c1a07b6267b02f132a91bc8ae93648be0c9b952f4db8265b8c7a7d12d4d5a0a71b5681414b7edd308bf2130ef33fc59444254ebc1123d2dd5b29cfd85ad76221d7bfc7a887f81a53d7abade79b654aa55cea1df415edef932b4910502e47fde5af90530b0e2de0f708b8f13b97fa65f9d1443fa820a5dfded0f90d899cac4712ef0b03c3ab23e312478433e8888e87bc344d21bfcf058d2ebfc6174164da39904df2ea4714c3b14e5c8fa17d4bf094415bce5c206512e740b4e13d49a42a0a77af90cf456c0b49ed4cfd4adccba8dcabdec2235a6f036ee3f843f4d21ef229a28e89ee47b340910629bfa5af576a708f51c9c1dfef64300bdf3d4b078ebb3db06c5cc1904186215fc6273637642b5257e92cc35ec29074135780bd292954700507e70a3ebc055d890481d20d9f8743caa4a1306411f8ac87b2e48314ff1fe7f4d80586a1e95c8152bc24a41c89635535e1426187bc8c0e35258f4f119b88872e7302bf39e174acd5bf74a4c30b04f85bae6f511ab18a0a730abbcb08b41f2797f2b94eca86107858ae91d3075ac01e53f7ebeb6dea51e9f397fc3536aceec5b7eb884f5ae229fcaac8b69042963e0760449ecff2ff2c4d8108295ab9ea9a0b90aaebbc464d28dc87077bf18d144b6cd680d5c9d7f017a27e76e883c06a2cdbe17c849bbe4eecc65c78a7453e56eae50b2ada2d40c8087dba16c5b6a7ab4882c26a81e41931037d49dcb6f841c50636597d23c498bd04220901641fcd1bdc9a456be2b53c2b6593c0ef61c83c8fd5c18e22acb338c6633847592e31d9964884d2960ec5d1253736a6858132f9327e8b6a6ca65238a3b1cca33e7c4cf7aa3d87f8f5f4161e469aceb83cc6b3ee53fcad84cdc7913df84ce39a08591217a5c0d6fa0091eec1e18c56ffeb2ef08d4e1694bbd9c25375895214930f2198252372a5598b051223c1dd6dc288c60d0389b7c9a331b01ac354c1210ca1081645aeabe6d50bef7e3a5de781b4915f145ea7ceb37584a3299d99ab2f6e44173936bd15b304c75a38cf62e1ad52e0afa26c1c406da985171e00233702be1dfbc57ec1183244fab920cc60f66e4789554790f866542425e76970942d5e9d9c29c3fe1f1087a9cc9db87b2e361da67b7c929caeaaac793ff3f7bb603d95f8b08efd8630d7885620926b11a30dae336938efd277f41af5884fb480ae81bb036bfa9d182fab2489a108e058ff6a10ef78ed688694579065e6c245fa02c632f936507a8ae303d1c4ae6bcfe62d4c9aeb55d9ca97efa847d6177885773a430ec5884c5ee68781d9b7ab7f1de50f6ff340894e4fa401a63c5cffecd1d43a71022432efeb3ad4874a86fd5ac5312c783e07e67842dc9423290f93f02e4c5f3cc6ffd51467907cefce47617b7cc3790ba2b34044e1aac47cc6d7a03a968dc65cc4c17c9c77b5dc86d1fae4eebd61c6601c59bab6f00005f0b2b6564459c878a3d1d7399da3db4b2ee2936efd50d9c6399db8199306ce86ac8ba36296ac540d4241b9eba823923aa2e08733f6b80de8cf11fd7c6e8c1a20d9ba1180873cba2fb9d500b857c81a09758cb87da629cafffb14a2a0c53e54e91e6fa8f573cee33a1abd1dfadbf6626050b2ffdacc9c91ec88157159dbb7ab3dbf13103b7f3f7f680b2e8212d956984fc31e3a47c239905775166b9a79892452f5b2bea8c6f31b6cce0a211b25b5378565652146e39eeebe4c95366dd25c9c4566b36f9ee5a27c50067f038d7ff4ee0d161d962c3bcb6b3c5f670882009873110861a28e6fea0b64939b66bed18cb23fb2f2cd62ff4d94f08cda920eaf1ad3e44ba2d41d4c27b79a3bbf5ced128e6622b965bd62de8fc235643aaec35b72921bf26928501588b80067f4cc943b07fcd353c0c63fb2cb1589fefb7008c8402dadbd47937406260d56ff31ec5631993be87ccd14ba76f085b1186184ce733b1a4fb14ccdfdf929bc9341a669a927757e73fcdfa02651691f75eb5325973f65026025913550a74aaef9035f1a47803d1eb629464f224097d73c1785584acc6da2e2e9f703c8db5c64142e95acc2e0c719410f7d1e9a12d566762908aefee4237640589252583b422126ec076680b73a25d68a7d28d9a3ac303b0ec1d1167d3cd15cc12c0c93e52e843d8360044be3b20a9a552662cc2af18d03a5b2ccd6d417804172cd1b5fc5ed13e9815703a0f09541fe468cb2cf247e265d05c465a3cb4bec1797b45784a323f146e9f3f6892790c1b753592a5e6ed06f1184081bd377d59e4d48f7f299a09e9e58f10e856cd36b0869184531153626c235acc555963a47c37f9df9bd9d91c3ad3218f65d43a490507c318caf09490033797c6a0c3bdf5552571d0340771cda6b061d5620f457e81908e075ad703578d69d4450fe6a23f122e8b383c3d2b436b80592d3cd9a0766ccbd28862f941143578c54551a928b438f22f80e1b4b020b20b8f5fb925319273e335c76d69dd272761e6cfdd3a3b29f863dd0c7ee84fff5741539e235ac19105cbd529b283cfd669b6d8eec34fb4b24c4bc7ffc2eb6d9e5b4e5f2f8c91605671700bb8e3d01e6fbd14cd56febfcfcdccf52aaed4ca09884bda1247bd024af5eef167bc4a63dbc9f678973e6e41e313c5845355b3a8a1c42713c8b8f1bf81bff91297e5cd7171d3fc45f4f2a1b15309ea21c917066073240608c8e63506d6e9f1933a74753c9db4080a30d73929a745c7d7d751b9f63b28eb4ba3c3e4ff2ace0c5ed00fdfbffbf1a3d88e7e60414245a730ab2b9e3385c9d62f55bb38e26a72190e481704b353b61b761164f63b3d7502c93b9f1a53e2fcbb739892f5ed26f0fef09ea0d3d5484952175c4b415140cbf05ecce74031892d6b9f4346401aaf5b4bb01c12fea7624d82689ffbd92563e685bdb16a35729cb81038d03087f73d2e7fedfe8573298bfdacdf9c244094aeecf20e554c44600c3796cc46f535c9f7c69d8dc340c8883163d95b2e2ca29ff417b4fcc9bf762e0dea789d9f507da53c47cc6d8daa047fb2662930cf19e3299b99c666ed959b29d258d334635aefb4b20076b59194d2138d0e207a6a0f4d05ba310ba824fa19efd5799d6bf7d0da13fdf34a5ad1cd96b86ecf767d30828997faea2300d9a501170866a12489c257f6646ac3dd7ec8c895258c7af0f50a97d18cc18e266f3b7a4f4c66e672d821f04717f403bf846bd925fb4caa126099ef9da7e31489fd5dbf53f71da96967aa4a93572abd2c0c9f55b94a1a8e718a3c78633d733ab0183d2b1c5b2126403c19373333aa2b8d6c0e3da670120e22f649dfcf0d12f09cbf434f2dd6b8fc3c7d60eea97056b94c4718f9abac7ef8ec8facb1ab5de1e7e67838cf676fd49801cd91e46e7be84912cc2345b634a0f9a97ea13ed97ec8776ebe350f55525dadf113561af27d6c71b2de2dd2e2bc97349c7d532d705dd5039b03ab6c17d6fb29b0e7e88e08208bf9f949f672595ca386b5b322c1295fa33f7193b6f4755bffa19e8aeef99c17b04b9edb4a99e01f6ae3072de94eb9b87f950446b215b12ff02260293ea0df3c37f10fccd84762153857a2911f9e41fdc3349857d669d4d4ea02132e04f7b7dab16fb2ee4b8a75b2ae969b21238a785181b291692e83166cb755add1ebb297d07203f90a5d2403f87e5a34b2157e3cad5ae6c995091921cb781f662796a5cdfba0faf3f3ca77b69f4c7f94ab8928368f82a6d5371872136752974d4989412719c5949653a3b22ce805481906346e84030cb782f990fb54f75a5deefa3adf48ba06009a1b769e201717b941ad7670c2f08e87cfec43e3f5d28866aafad630b11a9bf72218552afb0f12a0604e72952e36de13d41da4747847debfaf614c6cd65903ba60b120777fb6fbf34d340109afae24641a8e305d4f768acff5259d6db8c755678a7b5d96e11a5b75919ceb21016bb4131eceb96948600b73ec545858ae64f77eabd0182711fa7a2681c5b8a017c2964993d0f0151830c305701469fb3896a3bbc4447d4670963a1b16ee27cd0905f5b37a8abd18059035ffd1fc19cbcc12a500bd692a990ee03438caac38def6ab3f0485cfab09f496876bc3c806ead45545dc9f54dc1fc3d0313cea78a96c1c1c9393e8739eb052a92be420896bc8b62c283f09de4dcb168a49f55da77bb568daa2d6b27188185ff03de40aba611d5d8d63e7bc62cfb543ea46031f0d7b6c82f0f5826b66ef4f7b8933c5e5c1625297f2c625338e2485a0655d9a258495e90a8f0fbdcc4c67f5252ab57efa68d416c99b3ed9118ffc73a416359b46b5414123175287d0c7cfe056a0e2ae45b5c127c13cfc0a3b8099e24ab98ae8991decfec6ed5644be4842f439c848db46d3c4521ca2e444e835945f4ef5063d9a3e17faafbb76f7011794191457faf3b7e680ab8856f7adadaa4fed99d84f9e7fe6d0eb2bffc13918bb565be637f7a4d5b62d6d0aa1ae6e13ff2be54c41e714c3e36e14cad1be28c837c5be95fe470b50e3b4ccff06727455b7c46d7fb9c4fdcbf77a4500a9405c1cbeb4a69a8389f0975388929b6a388583ff2c197fc2bff86ac6abe3886981fe3a816b7fbf047cb5aeb8277fc77e7ed3e7959299d011a0effe4738df8d800c4453a66f832b13a83c0ad3c7fd0ca4400ff77fc795de4efee79db7a40cf838442ae36a2d63b5dec686844dd36d93e36dad938ab97c063ea83df47280ad9780f35d847987f25b9baf8588ac694649a6963ceba73082a1c370d51a68b64a83d1c6ac9d3aec62b995cdca58fdf03f520b0e3bb4ec05fb96beafba3fa59e3210fe6c94c92840eec43a30b006cab525751122d8444ff493e0e0670aeb7cd3d64ccf7a21b87c1ed79a07af7f7316403ffe154baf2e521bc5cff7c031820078bc8e99b1529a47af4f71f2d6def9fc175f21032a8f8c482a1751bff25c5f70e657bffb8f3598a2562ea26c0cad09d8e388ae8fd7157d23787f7f324f4851d6963862219fe802b4bea172a1f0959c990cd8fa34bc1830bc90fe519f4510be1544bb23dea7a0879097e3b7a97d8c0905536929045d3edcfefccef967a0a348bb6403232ed8b25a8adfa478c5f2bcb963ed572e685abfaff829e7ce3df0d6ac3cc1c2712371d7434fc6e2055ae1060d1abb3e68c2c402138f9303d72d8ec67b3898472483f7751486bb38283f20fd322020a2e2e6605f290fd3490bec743d4ed8d2a55b2ace57d7c47b5d88f71c641df21f04db0e6b62d86ea5e7349a0f4a893c19c5e8e9a6c302121ce40cd44042d35f419df23bf9743f5bcadf0dc7afed6b750aa567b9a9f79c4b58408332f5365f99b33467533da102267b5fe1c8d69f7873a9436fd7279ba9e8936c4fe1cafd6a7c1518121c16d100ab78da9e910be02c98d4ebcc51f3161fdfe7988d37469f180d5824408b2f6c4232ebaad8a762b148f6929f3f647d1b6ea5d26279687b802f3976173eefed0220dc6f47dde6035924cea3a65762f8c6d1dc6882fd4fd7aa387146de3b45561f26b69137a5b2a60ac4a30af3cbf518804f56f4ae21632dbf08c8bcb66b85678f7f32b407fe8d7afd447fdf76b2ef80cd43dcb573dc76d47226b183a9c8824c47f3a33395fb535f4f343af3c4bf913ee347c586284e0039e4fc3deda566100ec99accd76336dd104689b96b789b4d3865ddae1be85f23685b84a156750616c48add191627b18682ba21b0acc5da308979e1c011196258e204cb4cf38a5b8f3ef91f6c1f88e4e937633ad533f2d99c5802ba7b18805b7ad564da27d5094ff2b84ba371213a267210e59646e3794d7a16c13f54054abb641f633edec4288643af9f9ef2d2e2cc118b19d8a8cbb002f424a42e04362dec9ea8f5988fd7fbdda785d7e3da1a000eb1fb947280537be8a72b305f10cbe3e245c4a0af9647a6736ed292647002cbb42ad59b72528e7764af2403a773bb76eebcc5b92cc1b6647bc90b3b59594ee74d73356d93832f33ceb4e6bbbd4bc21272bfb0413aaa1b6ce5f56cf7a3bf9d311a810564378eac67a9eecff0f3e633283cf6cdf87de769f0ae4511a36593e3997f203bc433533ee5b659a1b092cc4a72420a80579a8120cda9254a58e975734bef67e4d9735ffb8ebc2b7654d4bdd1fbabbf50d107398f61108d5f50d7fd37f73240733b1b8afcc8c767954ededed8c6789ca62b2d8534ff8c5dec91dfee14d08ecbee658a24cdd034a02b5de08ed719ac5bdd9c88436e4f013c027de99e0a4ec2d5d63c1b23622db8383e2917235d9355cefa96e788b592cec3a685a32cb8e1bad1c91455d3e14a1eb7f961aeb040752dc822a7043e9522c5757c26a873bcc435e20ef9a8d591f598771dc760834768734c35599df616e01a30a869dfa295c1bda39a32485f84e5de994c51489ab92624c28d18d6d3c4f71060e6ca6be064a11c2c3b47c43b03f68416adf319006c037e9e13eddda2ec1a2adffe102272f10ba69707703cf66b0a1eac23ad4f6aa18eb4ade5ba46c77f4f87e2b57d28a77446756bdde57bfe99fc22ce0a232bd43586e9d6bbbd8808eee607beba4777112962b902a6213df7088d4ffd3df470a6a12d4b988cf7d3206e90e5b7293790e54c7cb08c8b770fd7e2fa316bbf93ec8908057d92f2298dd39b76f3c2b91993a9c30c72313d1e35dd3bb448a8b743fef0aef84ecd397915990c8e8ccfb879067d171470b3fb7558f3a5da549148a3ed948666b200c0a9548dde42cf051698987175630294b1f1439f24d4001dcb8c9d2dba3350f104d664339aeea0777d973a822f976f82fe59b5ea8a833fd908c0aeeeca55338a3e4958dde8d18cb01d5575b313db1dcc9e5356dda810d51b53dbf1c63ff49781b6b67911fa4fd6fd7815cb633e94cb9f3f2a063a6492288a8db20ed5c7bd5af43283aa3e1b0787c076a62bd47d390aedc9e64ec713879f3e3a8d91e64c03037dafa6f3570a635e6033d85855c3f36e77d56570b66e6142360b0864e45a0deacbc904fd93ae4750459945e30bfa9baecf5dde7e5ff3f72608f1ddc8a50201b465a8403ea4b3231bea7b2e8d987cbb5fb1d4ccabf7e1f2709c3cd00d309b892e21d0a2413d1068b212868e68d76aa0292422cbcec08b23814ff50ac5d144a13314eaab63907f0884521fbf72721ee654b96f0b26be737c67e865444199044367d2d8e30fb43096756627ef11c6984a2eef982273f7d3873bb73f5349d814e530f864cce73a7ce56f54711e987f23e032d1b411beb7254375210fd5c02a11bb273db8760d595869424bbe0b2214099810b7a787546a433457061733358814a2bda8072dbf8ceb17bb515e23c6613f3b19687060b7349ee848aeca95934d416fbb5a61b33dfae84410801b8145c8bf5cff51aacd282a5f7e02ed88370bdb68e7cb5aa7c735875d0b46b03d9a60b6287ea506f0c5c2e3ba0cd882b3fbcbb68a2f6a9621f3385b0591fd89b1fee6b07c3c481e78ba229b15a90bb6098c108ace7a73529eb357234ce7be5d35a9ce30d2605e05a662c0f9cf7a37a647e6363b20279c981588f4c83289763130c1cbbbdb3b641a68f78d86a754b2ee9ddad8397671698f974634a2cbee1ac32a0a7f40e57b91372f2e219f6a353a9efba0a6f590e426885774ee7e97bf60e54372506e688b7fe221c43ea5fff97181b3a57abd2aa5b3a1ebee81b94d7a6fabbc7dfb107d0c0b321182b2708de89aa6ead05cfbb6b382c6f4f56160b877aad1496135ec38c8793461afa7bce8a9bda0cf75adde3bef226338476732da27c5dfb045470b3edc66a6a996b3913c2080a8f28bb757ed636e1abcc75cfea2b5c54e3b401509e9740398f2486b51e209d34ce7a4dbcc9554fe66acca30069949d394c7d4fa8a6336604bb62e5351ef1c53afec8bc8b804f11d71a69e8f727390f3a6a22be300b1b49846847a03a75047b0fa3f26ea191802b290fa11830901132e0cf1a6d46bec0d400dc31ad62374ac9786b21de3a639590a6446b1de2bfc7fb33ceeffa39924b2e7f7c4f7b111c5b3439dfa0ef8f5eabb1cebf62947b530ade05580ba60bbd53df928925917419e45df94b79ff57f61e57435a54b226057a83f20d09b987cfb7c09eaeeab7c4e53587dd60ad3e7a0d41145a3494da0af8658ade532ccea562483cb02a7594f1fe16b29e493c18f55724241446ee2168c4940ec9e542970ba62bab0bd77cdf36fecfd666514f325b1409b1548ade31a5c8239c6830616c58c3cb046b2afb594466835d816bd7220c354d07431c1730bd800e0cd5e62105232cc55d33d9c63be604cbafdfcd890468484af3ad8f32906ec6bc99e098f986aed3ad38ba1b61ad9f0ffcdddd645105a5629d31f85d69dde5286104b64b39cd3c7f62be5431368cc89b679c2431812b8bcdb2bda7867b3ddd10ac63901e3888ab661811effc68f48bcc6869a490b5868a638b21f82cdcb3b9f7bb0cc895582f550681bec7181ce2f3aad231e255b729fc704ae688401b2e2e79f6cb3c221669b0d3e1113263c347abd0b2bf3672e9d35986dce6e87d3722bf4ed0c9d163604bd1d27fdda3a852ec608312c84a49c90ed6c2b6dec3338b3b6f703711fa6af5e1734cceea3413ee25c8fd98a3c7150710f22756495f250bdacdc3d5084f56e20b4bbc9cd9ae974f71a1f198c1f3727b9114dc182887c5ff42bcabfaa33c22865f22240f1c9a474f95de1f881d24b1d5156fbf58376d4fe666dac9325b7439cbd76e808420f46fa884e016dcc0fa43dbc1ee4a9af136e7594928f930d81d5d6128506a656fea51c3551e4a6d89c55b71abbe1f3ebabc27d38893bc6fe7d599bc0226fbd28d1dfb5c48e03351e9abb848e3a03db39ba811dbb87f3e2d26e1afe6879e88ae1de26d42fd6d482f6f825449950191fd8bf5073390d1cfc4278406fb1940d8be45cec5f24be26844b3241ffda1b08a610280b675729fa16b0a85d964fc201ee38c340704f0998c95cb4e7282b9a4b0408806837205256a33eb7a1651553eab107d3c1e05cbb5d3c89e89f4bca11e5d2b2e40a41a0b854e9eb7a1c4b89c1bc14dd63a7a1d8c3a92ce2416d8dee19b4a10a9c05db475ffbbbcf9cae880c7f3e4c8bbe3ee0a3e6984d04dc596afd676de974b1cab360682e86dc8fa1add4b200c9edbd427e3ce651da090c4c03c5196245788bb66a73384a43518caed8d172100559b782a11b7282d0ff225861c02b86e92844dad4268c17b6edf2a6e25162aadbf8bf9905573e4ebe92c75753324f8ad524e799517c67c05c17976f6e677fda54c5c8bcf25b1abb627f9ced81ec2130a579ccea595921540d061c71a76d8a8d079a1f69ba3b55bb66efdc2299ef17fc02015bf55336f8b4983f11a0e20da29ce7fcb53aef9dae9aaef4ed25b54c4a54ec3fb2a561d70a659dbdd8319d39bc5cc79faeef9ed5e7a48ebae26d5af45dc202c8c35ca680e957935f7147525437569376accfe67306086cc394efb86c8d1d6134f7a9f296b197155ada24c856e909f78db30315d807e3cbfbb113011d40b81190f1efd7f7c3ae43ea24e807bfae1885c0eaa883ce6398c27a2869f2cac6fa182d7076eb5fc56240a38c903eb6904b62b7337575e284cafc4cf012e4e56004de260e438bf8c6bf58e23ab20a36cca00349b6b9a59ac3cefd7e91f511d52f1224046f4b03b550e346e28c66f17a9fe83f16e18f85f3c8651e3146a1c81f7a74a75de7d65dc11ec35dcaae1443d0ab2a674eda063d473cda485745ddd9fdd9b4c830fe65580f825cb9967c29eccf513d84ec02a91ba5b2203083e3c3f08f5adc685cb276efbe50b2e834a8604e556752ecbccc493d719aa79baa57bfcf629a23c1496a0a4b76d521c0da459701336bada6e805a495955cb1a5c6f22d697bcf32ed463f83faaac66ec30a569d10466036a026cd568785b1bd314dd5d34a9b5b92ea613482c3aa13dde8a4b5c27047ee35aad63b2657f8f950f9e2bc2a2e0c29754f2d9681e31d94d1fa1127c64a9c15e8ad4fe1a6e9b8e71aefb781adfa4fd4fcd0f5ef647e22af2de94fbd76cdba01b9106d29980b556fdfc352527b0479ddaa9655006b6e4d24b0dcb3dca4023681e8eac86a455f969efa2b1451360ed0afdcaee19963685eb3e8c0afbe9e836df3d20df696c96d231d687a2ae195fda8f1966526fac76e121ff7e74a14586168396323345c0213b3183b65206eb444b6b23dcadad5c30d20d0c16d29dfb6365b21e995c02d84b25393fbed2978d5c802fa441b327b75ce4168346470798ea787cb0b4270667f596b480751658cbb25a732e403c8be5a3dfe466b379d7c5ac533840fbe3eef2618ee1b92ba76ab381220f1557479c69257e3b7bb3183ad280e841c2c553adb9200003f7acabdfaf6537ecb6565972c684b351f9d27c88fb3d63272be1bfa76097972945342ddfcf9414f05d3fa5bafaaa637f0b31b1f35c5400d879291aa6955d177bc9599e7a2bbc8149d35850ba16df6f1e58d78f4a0a19f8091f6324067d4af7274a46b301941d776d984362752b56993ef598403b96f473a164505f0a3192a32e868a7bdbb5669ef54970917e096ed1ccde2e3d9889a003e1ac45ac396fc029a56023d2895df8ccbddde0b42a3a64cbe61c99d48057eac2acde575ce564cca5eedf7bccf6c884c3097c60adab3fa26e58db83cd00c82615cd53af36ee61e2efe168543314d3edc222171950a01ce07a11bb33f6fa4c0d60952c51efd036e51acdf7b3502dcb427475d356b9dba7960094d446335e6cefa8166c884ef7e4a30ded188bb0c065153659fd34e6b5c09b0fe936f970625ec8232e0d47f9c578155ca6893573a3d8b6d50fdca4077ee5167b058e04c3e679542d663c0be972e827a9c59686ca0fac2b416176f8cade6f799d8b964e86082e5752d174a008e5778ca2eb59178e02ab2e91e42eaa074019e69f502f409b979426d6e7ffc5ef179b6f473757198b7c72892b15ea9775909427e9c2dec8b98dc31e7d99bde49cd8303d93907edab737c4daa2039e8b041a72721934a68a0dedd6f49573a5120c719469dfdbc022d5931d24b2b7fd2854668705bcbed6fb8621748171834f7f9e031be0e718c196da1499ccd995b3a9ce585dead22b8367f5981b0810a806f9d6d89c6c802fd0ab8dd6f9e13a179f6c7f673ddce6db2a0833f5a55b3fa9f59085a0280478d78559d6787c80fb371365bef41db64bdd4d767dd4eab494c6b3727c1811801aed0a07241e1a2b68083ab467ef23c634f762057c1fda39d2276ab4a7a1c73dbac3ce570a4afd7b7673ffe4241c085febd9ad3989086bd19e58a67b5026830e50731b41dd03247952e3a27d5f0b40edafb005c4969dee6e71971049b9a693c8f867c82f04d8ffdb9bc44e2f8bd137053d7680194d31fa54aab5853efe659f62b2c888c39ef3d9dc0bba10bfed7f870d5fb5bbba5b4f69c079c3228d7fbea1ac0ce78ce911dbeea0c927a5d65cbd5e90e544a0627df15f8180950f615817bf97f13fa002dc011b2dd7613eb5c6652a0ccd9693614f0661a6b1308abfe2d2701e80d866a30d5ba577a4c2119853e47bb9506bb8dba0894bdb16bd62a812b55a57c8855f0c983484cdab74c22294a5cdb329326c21a350146b14a88ea4acad7816204e23764daaeaea464cdf8464e8e2b699ed872684fec82a76d2322d4bf4597776eb5608f1418cdc6eefdc03bbb901786d5ed9a414d9c9a27919acf56ab2b7c0839ac402745bc781ce7888badd2edaeaf81594626152343621bd2cb3ec1d2133789d622437975a32ede78bc9bd6192dd886c2372dc17ff757ce2e0455ac80fb5ca15d137b036cca799f96b609d43b6a5147ae406ed0dc4c3d1727a59db488e9c08e837b148441e3e77c7205673f4fc5d7c5376c91e6e0cf7b0ef6f5db8ac5e58b36a927d42468ac0bcd421aede1cfb1be16b95a8b79ee1c8d82104646dfb8f62f469f463a251f5df528412c1505e67c305f1fd701b4b76d15376ec5650efe1790dd97d08fb46d1215dc096727dba40e9c06a59ad8d80f6699a9a1e0b7fffbd7ecc74a88482f200d280f2f5be6013dccdcf31a15bd665538e850fec417a3a8e087732551f889df0923158e5ead30182b83bc090f84d7f2428587d7409cc99a3e0a4711074bdceac14293ee0eac5094ee3331745cb4d4456979354816bb753c1be0e978361648350f35a9d0473e90dc7d198682c89fbadc58b048d8bd043e6d0ee4ef31f8253d75f93e59e23d2bcbbb44c68234822388275aef028df5dbbc4b6eb794eb157231c27714d2a68b4cfd79451b1426ac3af6950b7ed4f4c3de418470798f9f61f60e99273430c0bf39e73f8fb4a8d2e427bd47f3fb1ef9526875785ee8bfa246b23a883b8659388dd8bcff60c0a5f05c2713247c0693d0a8bf42c60d2d52ef50c7dbaaae5398df523fe948bbb3da6bd6587e58c837fa6dcd1f7985e15c105dde2d5fbd8cec3c4540412bc3f54704ec4e4d92b323baf9ab747cfd156b80130a109d18a8e8cc7964846086a433852242cec4573778ed5b850bacb7fd45d19fa81524e0d37732934bab7aa01a883865de97becb5d459fe82db0c4dd35d3344ec1cfcca790265dd720b926e4b50abb7da0de65c6c83516aaf01823ce9d12eea4ab5956270d4b1ada2dcfaf779a28df657e00261c8a2a0802facd3b71ca808f822cf64b2e395dc7a36eb25c88cc6bb914664a9779e34547158272921c2aaadea2ea67b5c003b60b188127ceee102fee74d213f40c7c612eec6500800fba0c9cd3921c9346aca3540df2b62b194e70ca4e14ede00e430295eb86a44f46390ad94330ac1b3a040b73b1c6caae51ee111fdc554e4a8867906ff62cc02b6a5d8b5da22d9da12ed80326ab34b9ce9d42ac89a5330b4d78b50e07f2cebe5974ddff38fcc6c7b733a29fd1a38ea1a90226301e9ff3e37b57e27759970200e251255f89175059f4e9f207d373d8171ac0d9e967ef9c431612918409650cede8ca88a9772240f7dfc244cc9c0b8a70a197f2d200d6c87288ab71a682663dcd680ce9387347dd4bdb971d06782ac5d38f0b3560365a39844e4f3824c8312be84a067be1731339108115b35eef615c8d3d283fa522f9b6bfa0f159d31877af0e67406676aa8feaabf28381c49562d14bd53761a1875fbec9f6d7f19d6e5f75bec8b00f5e17552fcac0486b15eec68ee832890def4ed7929946a39d098e0e1ae1bfb007d3bc895ea72ad4321004efb5f0d27552a96b9fee2ea6ae168270c93be26d68c9e1ae457a22f864e468f51d03877854122378b78d4fec4eecefb996b8b8c58aa852f4e175bc5ffddb92aaff2c9659ede37a35e65db4e96575dffa0653506e5b719f13144c36e10a430be32f49cb30427b1a02a2bede714a95c2435e257a5101801b830a09a0e77afac67aebb740fabbafae812da900713927770e81026de15c6bf38815360a82aa9c30830988500fe0953a87b79e595ef04db8ab6c2de3cb2687f3e098ff05d216aee7014ba07ac5fd3d7a06a3a659c2d19ee7e2b1072ffcc4c092c05f395b77d1c6278424a29c56d98f6046cf870e6d1ca62d6edc7e50c16d3af576afa66219eca73fd0fb974be42cde17352e2481b47cc8f2b8a0552564e3cd27ddc8f9111868d9957fa1ddce68cab38e7a50b4d013f1f32fdca7a67e42b7c044ee439292325724ae1a944ef5f53e484d6055719d97f29af67262ffd4068fc93f433ce4fa36fa6b26958f20b2e255fadcdef38d6d9de0acab058a50097f836d3862ee4945a788c610d579eb7631aa3e39d77fe6f021c1c4047297150777619f52c1d605b30fbed1d0f64d81986c1564236d5529536da08cfd82c8f4762209f8f2cd44497ffb4a020ec8078d3a026cb22e6877f3e5810ae1c9d28074dfffbfd93701a640ae9d387a98e82c7c8a1401cd01a891d7fa50bc48eda6f6b2f638fa2a9ab81972b64e1496737d9c85f80cac24946d99ff76116a33825f4224d3e085c061defc520bdcc61693d8fbca6bf0d9417a01707614dd4b4e13f9eb40fd70ecfcd6f88bdcaf4b43bfcd1f3b8a182a1063c2ca3c0cd04ef80ad69bd4ae07f2fba59283c95a5f6f57d2ee8a0b41b5aeabd0ec80b5e22b0a87d039fe0c0a8698818555e90e0679d0c6b69fbe1d1537286379b4df4ae7b780054839dabfbf3afc6dc00af7faf9bb8a67e84e137d04c96883b6805321b82410f6f36419d3b3fc24fb0e2bd78c8eae12b1d02cdff9576d4f31f143599917caf355e7efb5aa512b776dfb8a04464ed6b89dee2062371404deb080ba039b677b144c7406a2a25afd34a42c3e802d7df3036dbe6982e8da9462a8ec3311f382b29e3934d044d8df87f0dc069f4804cbdda1f1ef8e86c6410875872083c5657417e4bed53f47b9f6f042e4c633fe311e1a9e9796b86b772bb7ee12a03a67d6c4dd538eead8d04ddf1c8f371f852cd6608ab4ba6dd88a75998a44c7cee3e4ba2c21a0c0c0301f1ed09854729b4878c7cf8ef197168e75b877a3353f95a6f841f14b6c3c25f406c84419bd9aa7c826202a64fa06f34ee3e94086543820d8729fbd759d6f46be6c7ac49e43623eadb9b04521f42f126f5d3e6d9f3bad32f5973552365a470d308ff438db32e988da35416590ab95e5126ffca73867a8b8bca1de5a5ef6ccfc68771bdbdd9c42d03a43cce983766651d3975cc4abe6e5e49d5775e9582be05db66cfaf38525fc704a45fdf850fe82b508fca465eeb31b7262201e6ff650662188354ea1a23c0d7bead470c603934d971557a05e9c0a8b0eebd1379497a844c3ef3ad0d1100eb536c7a7b79276b5e4e7cd56ab4f1edbb071a26ce0a4883c6332c425f1c1d9333b5cbcc7870ee71edf7b67def224ced69b655aba2a881045769021471cd312a7dac1f1202a7ecba8d6b77d90c5abeb08b254b559967487c22aadc401aa8f25e33820b1955111a044e4d653914a1e415f2ae7ba1c1208fc74921be0986619e34e3da9de9b9c7a6e5bb9b1691eb9eb21b816a49fc02dcf4d67fb0cfdfa9177ae09b4418d36d5b430aaa1d42e65bee162face4706715c30b22678671071e5d97c9f68c6cd46e8a44db8dac40c5ae48e2ab0cb5e5238c757d04e2231241dbef4fa27c441dfbd51aa819dd3784f1e823340bf21728f527d2d20b48dfc6e0bc4caebee659e07e2f16251aa59c8f78334dbcf34f54fa0b78f53deb1237556a4ad1af0545d043524314dda11368a4424cec508e436fcee07348c09b49f4b41e27655eb4687513ee085be1de902c4100a26207c6de29f52744e02be8f6c27b95caad4688b161035b4154686cc61115699bfc6abd3fcde291cd97e56ac467fb01568f5a95d12592a77f94b42205c004d0e733506e2707941711c4f7df12bf4e5436f8344082f364f8ed0840e5a82534aa70dd661cec046cb0b44ee47c00b822cf5ccd400442534f03c129ec1747d73fa350fd6c60bd4b391c67fb796505bd4fa62d7fc5a6eb1d61dd4470cfe2b74e8478093afa02a138cd6ba6a6ba160e1437ed0b693b7f21ecb1efb529d8dda43b014d3601e4f27f3233d9fb9098f235c951c6dd915721533071296f312d7feeab51ed1734a8761164c0ccbef1f614564a1bd366d7b34ee027bf838255da92df5c3f811449805435e81243a1c333382056d719c41771832102242904de32577ec0416e142f234ce646550597dc79f7203503f12759152e976b4976ad251b0c36d445995046b2eaf9b2e7fcdb11a096ca5d355c2364b9a6979533fa168d7f4c2a45e6f26677b7199bc828f91e4d9f51de30c76e4d3b73668b169e6db9f7963c1b5ac17ea1b074624eb77663f184f608f5ade723151f5a0aa9653e9dad3f4d2b9b74d321db22244442ff5e8b02fd0f18b292994580ffb0da17883abbd024b9e3e3b5d540d98fef3dc3c2ddcf5757d44d48049d0a8840b7c2c7d2d9dbf214acade8c64750efd064269cb467b3b28fa96be78da2eba907515530bf0cbc62cf512b03c004f46664692a2d5bc05772e959c85c99c11664a4a1dd75d636f2f50ec6f102d9a8af073e061b7d872de033ea2f1334da4bdcbad03835a08ef081f1f7bb67ccbfa729f726971e9d1a9cf457d0025ea95ef8d016d8cf08687d664ad0f73dfeee389aedcf397df12851c7c749baedda05d0589a6632d8f8dceb968099d4b237e6da9cfbc6c750359018667a0e27e28c0731935cee0e39f9594b9134bd11c4d5787b86ae92fea30f951bd6c688ec189c01ab94bb9aaadb5b7591e1689339908c91073801e88f17cb24e0bab43270781247cf7da3c55f7ef37aa76b1568239ca788478617481f455782cdab52ab29176ac5d4e83265cc6dc524ac3e83cc706120da516b07b3ca2e629080a074b9739e6fdde96e5e8286febc51a2beede4a69020a6808f1c7e1403c4f002d95476723ae8b4a5ef9a170223d1b45e3bf20adf761bcbf975e65e9809b25f67784aec30aa5699563c9612ab8b00eaf2eecdfb1056ca15e05eb896131bc90fae981a310669d12dceb7ba3f934bf3469a8083c956fdc924beabf9efe2fe7c333bb360cd42ea8a1ed2338b2e1153b352b637be7793deb20bc5bceb6cec77633875c4f479d83ae6482798a8f29a247af82f07dc4106877622bd1cf53b8d03828d1e8a74e0f5548ed4aa7a24f163730d308260906c47b253b2b78844b62d48dfb7a01d2e2b7cdc2e7e1acf7d37e396a165e6809775fe30bc0ff52e41a891af117590a2b2fe4685298e3a72d25f0b1fd5511375bff09b98dcfe18568746e73375e56123bdcc67935aff71b0e6fb933aaa72cbe1638e8d4fa21fa9357c62e08c6750eea6f32e8caab526933b5795c5db61c69ffdc24ee822098030a65026d3502b0165cc6d215312850c8cea3e5f3c5e809dfa7f08372fc85eb061320370bd52e3954218d269cfc97e79d1b7c2c443e4983c0f3ae38fa282a2de1de1909f589ea1c71f4b27f884acae47a6d74e321b001e1f3b873a43cfe0869d1dea5dbfe18777ba7671ad0130d1ff011a0f712c8b3b621d6561eb5f0f110ed260412a231dc918f08b75cc23bcfb55dbef3e46e1609b58bec78152d854a70a328c6f893f1eacb097504b7d134716a94a88d0f6cb1ced938b37e35eaa7e46c5df6903cd55047f9bb026f3c83b9e49e59cb9de3ae36f44a9f6447eb8e9956ac07b6156877ab92b291bb0590f4c8db9a343b405c6b46b9104f1b363ab76998f3f5cfdc9284d741acb31ddb76e9e35b1c73ebadf26f13454fd5feaf929158b9f7817108475f84a82c04221cf8f37ee337eb0b7ce009fba19365791bb363d55aced2920b2eed54e4f21fd31223110f9a348cde1672f4686b5543ffd1132cbf8022e9919ba00b1456944e3e09ae7f97d6de1608b7ac8b5ea2a1cd2f83ab50d8faf456f2b80047c3d538183efef751721e84a99d784ba9e1b01f142b0a9a8b4f6269b5ca9e2cfada31b85d3b4e546cbdcddb50b8f21d4fec2db890bc3dd094ea3b0db3b011e011fd65c0ea7ccad727360f674b5e1d18c23323a272152cc6d60196385cf7952e84b18088a3432ad5d46b5f4370c240c9e23f1bccbf9a3630acbd6d887a6b9d9c2cf35253ed1c757930ccdf67066734d24b15de75064aa7788fcea2bc2d3d3e467a5e2b567dd81f8b6122bbadf6271c52bd7e4e02682875af5cf90199643abea2718367359b252107372238f60dac7a8826c5272dec1662fade5a3238876255c82cf3a793eb7a9445878943981e58a5ee9f5554c4108a3c679764435ea057b9e1c814d63b8ac0afaaa37e63cdd9e7c10bec4b86fe4c29d48c8554680daa3fbf3a67262bf293df08ebd1fbca8c0ca976802d2a690f3fc4ffca3c652e59390a6484fcfbe8ea9757555bdca711506d125bd553880325e86c87e46892be1846752354df43d04787b26b5eae35c3cc54c3cb34425dde071c2699bc2ba3f79a7d42ba3e9f0441bbae50423a55d1eabb53825b3377e1f315532332404bf2484121b63743d6fc5754c73537aa87c73e677141e8fc6fd537ce3de7bafe96e28c9cdad09f1d5e5e3e4b70a22bbbb9cef8fde6b74e39611b22ca23d0d22af012c38c867441f6018f5c11b4a0acb9f9eb2b53caac863a0c1fc279f4753027077665fe56b0a555e37053bc543a0849c739a7e5e0dc6601dc5c41f7fd55f2d4b50a21ef177f10139ab230170120301564608c6efb0c5a99697b3ba9c810dc3c56edc99b9ac59d61db9de635d9d124c1bd434f54a0cfdbc8fcb542d84a19da06bbcbbf4e5df03f2472db937d8a8288de04da12255e05a2e0d0ea7a356612dbb12464f5b991cf1eed53e32b930cca065818c66166c4a6a9f19dd2d0902ebce72199654f3640dfc592a34bf1078a4b1bab7ce2b771c305b6409190a6023381c75de2554aebf4b93f9b4592d7bc3ed7950f6ac081e57457306ba43abaf3a5b6d3ac550f8e9c16a7b13e9f93c4d93698e2d405a8e2ed0e3f9f957c8c6b82818f4e552a6c8b17ba08b96e1ce0d45c7464e981ee1e9a271b6eadd68a9808a6ad6f48620a9f520da009bb0169ecbb1c01dd6be6ca99710dd358d2f12e756f99f36c71293b36c317617262407e8b1e95533c46f780a9476612b1d021b15dd38be21b08e9c47e5b7f6c01034ca97caf81eb263ec48b66da5507bfb5e9f1547a160d45dbc72b1812bde186135eedf94540c05c8ead7b2f05c300efb8f62887fa4533de39f9fbff60d5842663990ad10ec23cc6c3d91376ff0b864c0c2053f8e47b7a6c955412cc002a3f2124d99865fa2b1e6d143e8037d9e0c8761faaa7a227baf09c38bcc09847cddaba3fb6a9aa3a735b89b99ac6c73d4dc4fb9f330d5573d40da42095570dd3b16871294041a53021a859030821f33134530fc038b7ea042789e90907991035b9659e0cf7e731cd7ec80be36db39c007f44f3d5cc8d5ea096262d633320c3163ac9eeab6aa5e642fef11a666e6676fef2ff9db1a51561117e584231e9eb2826ec8dbb1383876f74a1103a589ebd16212e9471d020afd428f7eb0400fec8249ae4169358bbf1ac8edde9c47fa1d71cc6e94d0825a54d923ea0e51e704c2e619b6747091b73fa7c9d3e520cc9676feaaaf4146f603d4b25f77ad26fccd19f779c38b3b2564ae10af81a67519309a95d5bfd295ce5f54cf2152e1f71cf469b2edebff3af44402bd91316b56f496adef08ba7aa6b86241058dd35729cdfe0f641d21c5e972a729b367db1cc5e7519f4a236fcde4f2c19434411147402adbf63db2aed53847c59c45ed1692567e28293d1e98c62ba0493fca90d396467941a743e99f84094f7aaad5d4ca36c3fe51fb10ae28f0afe172413948e1b34145364a68bf397809ce1e4201486af7c65c64a0d18e751faba66cec7a7ca8eb8cd065725c3aa427017714c01dccecbbe6e633f248b273b62435866866c37d2899ffc14857c3b14650e79e8a68d8bfe8e68eb1a5435011c33c5fcab9ad8da0c74d9234e463ab7abdd4818838b9d71c62489170258b9d71c6416aecb8fa0d367dd9cc4a50a83b0a5f085a1d66f7b326cf76a7d2e0c50ac1c0d00d2e58068edd297002af41d770fafc764caa8b52b23341cc3370db5df711916e0b90af12de0e4f9d15ad4077053b718bf2e9d6e112d1c05536e86f46ea00cb783a48d9f007734f176c17844415c7b7cdc094f7731c1a0ab132b0996d5a49bfc6033aafe3a0d68487a9cf357ca80b6a71569a1ab781c788f6c8c31832e559fad609dc699dc011517eb0a8b8fae1b06abe3698c59afb847d675af5644659fb104f1c997e795b9ae0d305a13524ffb6a59e924b93d8adc8ca2f778f1ddcf0c8c1254590b675680a5697b01d196c51a3121a92185d8beea8f41e0e69e7461eb422cf3aa0860515850204f9203b5d0d3bc60366ec6131a7deff77ed1934c5ca853fd496987aa2662aa3414824a16c702dcda0461124a0f9bad25cb431f9ca32ec5053980818e5d90cc00fe6f83a4279f7eb4454de4d305c7582e81293652acda6b4188b4770ef916c06e1820120c6e526b8ffd3507850a35129ff92be26fc46e66a3874111efb7b7c60c71d3b8409667d48c1e625c8e40bc42ff2fd01bd94797f62429c8bf66ea2affb8996fe122823bd25d39f1b1d6a3732217e1779af0edaabc9658359611739b198115608c7368a2bd24bba0e67f7ad2cb82dd86ceb370c6b40b81754d9da88a75f614ed7ec36e38b5660ee75c13c8127f4da0696e9ec6ffdab5ebde57c24cfc720617adaed7c15fc3c9bd683b62bedfe628c0e9e642b4c89070c697e114ba4bff4b3c5bed733db53882dc98fb42303c12f2404c213f0471e32953272cf2acb0811acc1e9c8d81b0f0884b7651f9056959b673fe95c540236ff6b9eb246ed93445f27c5ce7373dfb16b6bc0317034274fbd018079746de4bf34b2ed65d81d8c8d5d52136f1301f76f5fa68f790fdd05c712eaafc74142285ad356b41348ea4b04a6d453b15c96dd48d28e3f01406ff245ece539a9b4ca1eaf59e3bd6c391218819431f3771775da3dea16aaaf18aa54bc1aa8bd5efb3677a9ce145ec3e973345f132a019fea7884f2fdadff929d20d78575b9bf8996fa7289c3b26292fd0b8cf0b65b06d9b76d71babd981fcce1d4d61e0af34f47288133bf039bf49c3b395a84d1515edd233d896d3671a09ba875bddf331f27e499705c89891ca957e6664afc87d5e21075d9f84c01901c5bdbe00675628b49fb034026cde180849865820ba58e1d660f3ec914b447b4f9135e203d88a892afb66711f1725f6635446eb2bf8dd6ac8ac1fca0770f125f241a431b56b775452d376b2b72e8c96b24dc3db450ee15e5f023fba93080fea069c0a7ebf2352bf9f3cd70cce63258e7025dda4014c31b1141dd9bd18fef0f76b5825b1856fa3f582915ba041899dd3be1248c555297817976afd7a725d70b7bc5e01b570265c82402a0355b1ddaf2c0adcbcb829774960009c2a0588a5d1b157c0df70de929b66e677531fe9ce68ef0ddf90ba3bf9ba6404a045c74aa3204af1c53ce5d7acd45f4404ed5ad1bda4f2d20dbda0195ba9b5ad48ec5267aa2b1dfdb27e6c5fe3c998563d5495e0612a2f185463d5c51c01baa18d4f50a79ffb4634efc4f93d39a6fbfc4c533a58e48b0a1cb5a3b64dd8da959130ccc5fbe9f7e83e434d4e781eaea4a9bda14691ea5da3e35ef9190d990c8154738f50082859f0ef8d4c61edb8e9f7df4eec8c2152bf84faf07a9b58df4a54a5c3e496d7344306fbbb4bc10876161b9031d128f91bbc3977af854ebcafb6028a13a224f8353d21cf6859c16c60bd30da4545150cea71c5dd4f8178fefd38bbba6b1e03aa11e18119f747fd2e4e70cb24f45dc472b58ae0e13db21af2f3188ba635c24c57485ed744e7352511fc5c12fe239c1fcc59a241f9d46c5bbc77a8beabb0f200d6e7cc992a82335c5e75f7df45630a28f44f7dc32a29d76184ef24e35dcd380f4fc46dea39dd58d48e11614fcf902a3f3e3bcbea86c7bdcfc7f936bdeb28dc1d7c6e381deb9057e61c02b372693a0dd9715e7fb7ea62fa7e0603d3e8674f876d024be34fa4b298ea1486b5b91a0e8a794660ccf19d4d4f4a6bd4e698f2350c7249fd3f1ee2600479a390f444e7d0896facd4cf4a971e58e1de66faa52df7675e37fd5b5bb0f8c0441b9b7479045c50fc37e07aa9418589235c3fe0075caf08cc24d7f6a40eb577bc99340f174b184913236f646e646abb58b0fbd201de875260c70c9d5037b0023a714b7b989db67f1c1e218cf0b5b4e1e690b7bb7286c8bf1b30ea5aa3dca74d1930560d2c8cc450d9db921398ebf9ceccb5120aa0e41dbd29b5630f93bf5f6fbdc69ef5eaf687ac33c17f0b7e1ce0f4b0a8e1d21bfe34639e7eb8778d7de86483b0212048fa626887d1106f258826bd77e4540853a268e9cfea6604fabd356731d7970aa750e093b2e021c8cafaaa21eb87f275403547faa6a07f4d906d88f40f0c68115fbfdcac566ac9443e50403bb2076cfe1554fdc364b1e301963bfb4f1336f265e600086f7b9e9bff59ca297b43732d7fcba4bad18d6cc5469134e75df3c1427288d5258795111b9899e42e403d1ce36a797afaab609a003509be1362fba4ef4eb8f5c79828991b64e2f7bc72376557ad185a75575d7205e1a9d42783381c876ed2575fffb4fc4564d20aba4b6790d528991c391e65e0cbeb6dafe64bdbf457ed6c2bce7bfb96f1121cdcb2777ea224f25184f4ae3964896eea51c4b0e98b586f3663b0f30ab61585d27b2d9053127766f572f5c6e872862eaff785487170908bbb5c7028745b2194619f85f3c34ded3395c5ac33f9030eefdad112de31748416d2dad9f843035af441294f41a287bdd34789b3bf96618cd1ca1b5182cbf7ee0e1a532056b20c148137f9d4fd33d92e7ae828e5e8abeb5d42b87dfb3ac855d8526947fd3093783b61caefa9eadb9b114137bf63c0647ed5d61224d75ddb85011ce935bf2be48baa65ec6e7742dae81d0491804a56268690a7bf8d9480cd48df6d999a7b2a730cce5aa5beed78d23ad7232325f38480b8ca5d676aff0fb1d87c017903a9a66d9b0a14f4dddaa00a647a756e727921245ffbf04396ac01545177d8ecea1baae2f49dac945582720b636421cb388a5b2c9fe6ec08c1afdcde9aa4a179a7991719824bedfc9085ad3e8648d1d3368bfda72c1a01497ef2c661983c7dcbb3a9a91761c9694ab6b5f20410029c277d2957215b203044dbd8c06b4cba8e3f6713ed72358e288c2758048f2086563b3bfe8392d7a67f51334de8c6dcca9681485745e3b817d88bfd88260573d63d04f8360ff5ade869fc306ec02f6629ef942231002d06dcd281cd9963bd11edcf8a43fc99b47a9b68a49221d0b3f09f5ebc4e97a422d49f5e46e81796be08bb390a1c62f69df6d60291d72eedd0b8257ae76c2dedc904479b003f0414f6f52829d51b6c3ce8ff2170a124bb2ca9a78607ae259ba46636f025136e7be3cee8795f2e80a366866bc0d0ae3053c39033c98a8072ec4d37e446bb48881933d6b2062c72f6644be97d3c81e3558215edb19fcf564bf512224ad0d4188c2d29c5e70419f08b5021bf15ba55f9d68b37eaaa3c5b891d16a6ebedb9eb47de24a157b5286c146562205623428e58b7144ed9337ef79845d29470a7396d5b3dd224512b08458ea1fd90f56ce09de70c75f5d005206061e388a6daf46e8b2c3fbf7aeff39ec47327fc89e3941fc50066e0b6a0ce6bebe13d9aeb29803264dbfc3df6e561fe4d196bd6778b7a6a45e7a31e61c3a0d58cba98e659d5605bb0175bf195c0eab96bd522e516673a43619e6b30eba495c50099ff52f274d51d3df951cffdb58a4696cad01c808d4994fe2b9731e0b779cc13e26354331dcc90041f773bc6b996dfe15dd6c1d02b4598928394c9761bf60897a586c5103ad942b2eaaa24613f7a21528415ccebfcccb6e15f4680fd935e82ba602b660e91759096bb52a7994ec7911bb89b8af23effeb6261e05cba199ee960ff788bee168906bc493a0429165445f375c4b455ec1b5599b21cd2b0fdaa2bd2b2b83d4544428dd7aa2d62557d432aba8d655ec1d4ea21224ef151cc9dddc034c03d1f1e1b32e7ccfc7962a8e81d58d822941a29955c0f5c5d1164016bba0ee833fa6355db279c5df7227be312195586887d407c915f11fe2d5fc768dd899b600e8ce7422e83d50aa5cdf4ec55d830411b2b8b488133afcb24d5ed93b079175f3e58c2da433c9f4a0c032397bb7e36b32387c969fed3514e4a991c5fee1c0bc0bbb3653b07004332ef02668bede5d24331792ec85488b761249cd533d77cab266e2e8528a5ae11330692bb2ccd6bc09ab831cf556ef1807dd7e3835d87c3b893eb39c56eb14a75ec2ff877cbf419a5c783b61e412bffe6e170298adfba6f68cabc13f6cd401d5dc6873e01122bdfd3617983bb5af7d8214b31b995ac77d139840f1e7737cd05bac43e0bfefed963f52c027d77e0035d53c8595d8ae0f7b615cf14fc469d49c2f94b4ae9e4427a501580caf79f89272cb70f91d7e64e40a2ee1014fe8fbaa434e16f8ab425560a9baea4777d4279b963b9d98d6652bc5645f577e5d9e977ab5449e5e8ecb1abe0f3625a583ecd7040da4779012c827ae4ef9e2364aa50aaed888b22492d49d5dbb654081e9d3e2c8529968841d7279d2899a71b89a68a0471f73d3cb958bb4e25256bf37e55e80e2bc80075ed83663fb3765d028d79bfcbcf17ad1c4909be97520c8897ee724c90a1bb608d5de507bc97d108ba1f62eee418605859e6e2f02bfc7247e5006a8af6614f8d953b5b52b53d6373188624b27d869a2db62cdc534e6a9700f209123279febe188151dd1bbe4b826e7cb7bace40cbd52ba3b5c3fba01b2d37c0f0fb41d3f3cda1112a2babfa8e225aa52d4fea643fff2451008dac140462c2710690ea5849a02572e5125d5bc6d392ba79b0ebb94e2ba4ed5d84218621ef541e6116b7eb7f53265e429eb44dec42ef859fbc391380b26b67dbdca81f1d78eb878efdcc40abf84e31f397f31ab46e4a78c1c168b4471bd1051396ba94c620ccf3a03bf44bcf9e0137e2ae4d39419bb57c59de067dded6fd59f1fdb3711ac7c025519a65fd2e8f60d4ec6fd30a5bbe3bceedfd3b4c3ea4b9202c5233bd0bf533d0d62efed2dbdc05a6e0c53c77f26ad74406bab680729ff2e9b2c4b67c94f574d4a136a409370f4446810b729344bb673f5f15a6d156fb638c7faa6bca9f9b533c6f0625fb84c5352fb5ac491272c9c7f7fc40d9a9af4384e546d63f9e54b3cc4e40f6e2e743781c59244b6850f6c83219bf1f48977af3f04c3433d399b0c4d73f5656a514c5896e35e8b3749cf9fe816249913187fe7458f215ff196e346b22b9b88878f3afab36d09fa31887970ca4df4525d60ed09d6719aeb8fa4717de556cd99fc15bd4c7b1a1165b65922cc1d922df9733478a9261cf0f264253d28834c8f1e9efbfe1a5ec488431e939e2e3edfcab559745794367e60e2cdc07067920a296429f329a1fd8ce9b8d9eaee0435ad67f80f330a965870ea4d5f8a8a7a3135ca822833f9e7033a73051f024d50e71b54e0f82e3dce4bb07a575d50d01121e79bf9730c8d93a7c6da9dfbd97e83a53fc699aa53d33d2a611964f0facf46e64e508c411047e26530953910f6d23d0c61e717af154c9740bc52b38c065cc95e4a950dfe1fb402a48cfe018a81cd893f0550a826ed7f22ff93b2f732136c08271480b699f71a4c1f67278c1fbac387fbbd186cebff77a1f70f5a907ce1498bcb3f95a56fdd48f12eb406469e2dd08f0bd4bfc3de56fb4d53b42ea2416aa50df55739d272a24ed2890c1a5bdc831cf86b147dcd47318adeb7b671d01f3ae183894b63444a220ccd9cfdbf5f24dfe3a842470eac438aebc4e802ca68d98170940f72c534be40935921fd5caee248abccfbd18b6212ce17590a6411bdcc139d41508e18a60e6c414d88feee95432fd2bf3fbac0e66506dda144b3efa18bdf8cd1251ee391d948121678358aa0247a8511e32ce31204a2cc7c65e8ebf166771ade047c55486edcaf390a5e7eb117fb97a94fea52193a3c8504266dea5151125b242daa0cc6dfe9662fc624227836fa6fa21cae502d7048cdffd252611a8449005298e5ed2985f329c42c9e68ff561007880bef7152a9c42554bf8995210b0f4f95e3f4f2952ce00512986bb222d3ac906360a6afa2aa494c219f227237db9e185af0ed4d6e2d378ef7c041bef1df0755daa347c30061e571ecc3ab5a3397d0f13dadc287d4ddd07841e3035a10570894c86266e1aa47622e97bf1efd2a2e2284932af97ebda9082eff02280632c543a8ad70770af983a8859c04a959877dace846f8a075ea7d26b1939640992613a024bdc4d4d69f27ff38e3214efcff363605748a40657a8142d9d50df6a109770fd797cb79490f6274672595098e1431eb7c04a8427a545b2c87e58b0d7dea325c85a40ade27d0158853ad38b3502daf3cc380770afdb10d25b622ac21a09b0333a801ebc4ec9749f2faa8e5e309f977d2ee87ae1fcc7dd7b823cceff7bc1daba6fe824d2a23c289095ea400e98bb2b7a9006eb83d93f164c92445e83376a4e6d0d54990cad15a0545411744ddb4c0887ca72e4ec1bd5fcac2c27ca301bb898a206485c2f7f4953b2ff60483025eb165bc0b7e3e76cf036d49558972062af0466ebbe8f996639097f40aba3fa521344ac176d85836aee7140db0b73f22a12e3b1d961b3815a16e88f3cd3508327273ae039c135b60895f3975a2a923b8914f4aff2b2cff98e1ef27887c1569b78dd5ee4260e1248656e794ba04099a60c84ee66b39ab6ddc1ccf48279da001acc9644cadc79bf1dab46402a1ebf5b5815161b20efbae29dfc5079d3b5b146ee473e08ba555d813d3c1cd244909db2e67da7026ea84641d619449f42740e4ce87233d043bbff1bf9db7fb53c170a2305cc1d69b322cf97a2dab273ba38adc93fb6da48e5ff3a894e829e2cd832267643e36c9a7ff3eaf74d9a2dcbc26fbac841e484dac739b5f1f5f0feb7b811207e922e1b944104ad1303e0d77ddbe136c1ac725a95ae9b3f4333eb7a0491efe24f962c08ba87a6fc96b82c65859684bef76d4b289a2017849c8bc05459de91d995d9d82c1caa36fe74cbb54ee1cac5462a3f503d6e5c90ee716a6b5c8170632a6f5aefeb1b7a3f68c1726dae1fc3e51103d4d90449f4df89519cd47c1dd15373298d4686bc5d6dd49561b9903e8d131be3d663e96bc78296463e7783a59c2c42a5c468e330bcb863cbe0d58ab2d1d84ead224ea30e45aec4b528a75b9eaa5381aa86891ae346c5fc3dc9371150a2ae71a9f11fcbfd181a654a6d0be220dc9e943e4f4a310ea1e465da44c45873053c6a02fdf988c7a9ddb6c789bf0ba9895b92520d8b31d16597d7bcac6f5621f6e9628d202cef27bb8e8f2c40d84a87d48042999c69313c1a51668be2198455fd800277114e0bf92ef741f0868b72371c2c77a164e87777ef3cd1139ef8ddc6806adc45ed4b031a17d1f723789e9431877ce07f5a105f629ef561672e13ee931bd39a95fa0385bc7cfe5e6ef5140f4f0b785176e79a0213fb3ab12a0eb2b1debae7a8b0fe3d945b6a77dc2d0d7361352bc366ec509f36926a03efad2056838b114c8b1bb6db587251a99b2fe5a698d6c99babb21576e1ac564fe14370a4829f91dc3c751a710218128048a097d6183360c5e0cad11cd6b49625b34beeb0fbc6e3db1a1c831d711c6ccf700624fdc0172f3b30ee508f47371cfd55713fbe9f5bc2bd0603a811192e465725a4d9bc8338d450eb16451b46d9acf09cd752bdc2299cd9a34d001162c5386f3b5c9402caeca6d9deba53523cf0b595716a2b1d761fac293b8c54d254aa691d390e7926e73d14c63481ea79b1dba59cd097ca664b7008fbf4e90a4002f23a28a828db25ae0495da90fac26f9049d567f91028d9dc4f9be05439b533733a50072106b1c66fd6482fb4597d9177c32b49cae8f4d1822a099b26c95dd79cc17d5c4e55600202e9f8e37ee47ee8667629f8cfa125fde64e8dd97f3200c23cf98755fc5e6898d03ac57ce52b7f07e7d21a240b99c9da4748584701a0c5b8a55b700ec4c1fb4838708b9f653b4f74126a7799f4d12afe0cc732fcfb4f7dc7209bfbf5da3dce2353074945a3b78cf269f4bdd5a70969317d7e42938fb9d3333d95d44bf950e5930dabfeff7a20c4923b6a28e4569022e8ab750faf8a8d6f3aa6c57e3af54e7d4bfb1fe36b4738d1320eb48cab7a48a53c626cf1abe9f378d5921e29fc97de964a431f64a9c14380627a85f75daf8d7c6dc9d101fca679717d8b23d7e8410f52e27d6e1c6ea1a6d268523099045e37428f959db0d11a1c3faec6a3f2dee363b8ae0bc7d9cc08ea53e4e439c546afb9544890ae6bedc96f0b837049fbccb35454e8ab65615cb078ee77dfe1dcb4aa1a1e121529931c7705ed50d4c7a763b6d4c2c3c099437115a16cacf0a61213789561eee231f6a456fb0ea485e21809abf4bb8bb5612d1a8e682955a9ed9dd4a2c75a45d79c03d67c50e1b94368761aa754ea057059b6dfe5cef7f0201b4738a1a794ee4605a2caa3779843c116f8a41ee0f5c9c5b0d5f742b43672e439c13e07843737cdfd28eab0545ecac71dc01d1597cc059f6857f609cd32191895ced35fcf9684f44e7091f0b7e83b63e4b6428842610afc4639e8f2b79de506cdb96da107843f50afa4b45d5d5e3ca3dba6247a0f9f24f7ad27bab9c547946f431452cc89387dcfa1c72f9a625863502f9a9f588f182e2672e2dcb20ebd1695b5c734e758c62d9ea4679ddce1111bf0909e7bc3b3ca95d4ae554edc8bb868844a1b038b2469bc3694f921eff916c80bd070d98f60f5d76645f1807e230f407b8d98e175ee25970627cffd47472fcc2e1cb764a5e3a330af860bb1f58d20332d43680a5c3fafed00e53da82e00c0d8ab5b6423c6be6f6b8b3288c3ecf18fc597c48f5ed18a2061b69c560b59689b2261405c1d486c6cdc94494d4a8ea23bbbdb533c45246f9f79a3fbb17c41a3c3a6bbcffaed6cef8642eec292b2bd9931127c8cb9f197b590f54a0837f2515793d4bdec214c2dac07952b3556af282973409447605647ff6148917e928aebd5fb9eb5a0c0ce7f5beeb08ccc1b23feed4f10b7081744759038c8c431525c6a3d2d5cc33ad217701317e863eb56d3b07ba2cdccb985035e67bde6aec75bf3bedd312deab79df9e576edca571306dfc294d78c5d050f61d00dd4c0f47c1a2fef7f7440919daa8d56c84f68b7400964f309b78959e03844973b15fc3245ad12e7e72b02cddd73cb43a71f7c523a6e43d4a07e99dcf8f4f8ef9b4bf242dae3d23b48bbbfe534075795b6d4222901008236d4ba539792e446727a3d99ffbd049199fd5ad51dadcf76a4e261a330f2814c76101e3e2606722f564bbfade0d032eb826117699ef27046cc7ef4129c7137b04a9ae7e51d2bd4c03292d166cd99b5791dcfccf778c5e96b88c15f7083d1a66542784fe414ffb8e6527ebd828179e6cd37ad541f1ca55e80135a4891ec185a5ab83543b08618be87e2eb5b793fd6c507decfb651c0b26a46c67298fb3e6343e9d56827d8720baba2c259564c3c124fe07b216c35b442e1646828aa9eee8b39a6ffb46e37f86c07712ae07419d3601b633b2a5800347647462983411a044cc20f8827208c3ecbd11d5e4bd15fd626260ee77113cb9b24e6ef2a2158f9da4e274be6b361ad18df5d4512d80273cfadd8a1a97d8b93aa69680b82b3301bc61d3c9d353d1385fb99ca313130578995d533a8e62bc1c3e6f94f8119a4023b56e1cda60d97be0404cef2865007348b0d619199520669a692d8738c207de7451ec504d8e0a38b1cafa27cf78cb4db3a0e7d8217adafba92bd1e691962c92db25050485c48e56e7b7d3c8c0260fac59e89e9fd265d4354d59b1ca750a27e8fe75d69a972f1352bf9e8f214156e9b957341a4d858001dc4a613bcc4d97d4a7471e19c2ed90adbc259d48f899021b7043427422bda432310290b88a9831b073c3441b26cdf563b16ce2bd203319bcf40ba9dff80efcd01db2010b0df123d328c52950474d6036736454686d22c08b405c526648be438a44f881e2985a7272540b03b4e469085383cf1e18742d4b6d9440dbf7506d9d35b1fde987c3047022e25f342f4c1f2e6cd2e8587d0d181c1442f5f345228623c00bae23ab5661900efb09d349b90c75c35b41c979ffc80f4ef92948b1a2880a444142b1b08183bfc4af0b606b1b2b14c1156e904f993857a254193f6b945b2ec7b1b84fa1d666a5686cabf2d97f7d4aaac5a2f793af96c3597ebac8f1254f1877aa5c86b4924caba791b13960d406aa37aaeaec27b99c7155c49b75c01fab6a42ab7c805295a602fecc15b2257fcd0d9e41abaec3e478351d62ba20094b5649ad66e10e1b18efe19325d1bd986e2be9ba6bb924248f1959638aa3c99684128612fd6362462fc265a225ab4582e08b716951aed4ae2007deba0c47272548c4ee2c184baf0e80271fbec09a4258d1ce0af6e0cc5ea40a8a964d19c6e3bc68320fd9894f3623198f62c12dda9042ed38248eeeacd9fe6da6901e665aaf7cafc60510fe76acb43caa38963cd1fc365ef4921a86f2e6796394946191d45d8f4d0c6458afff13820f752add7c5bb072104c68a22e84964a7b93cb35e190667ca753c0ceb0ec67d2591b3b09bc0b39ae4c7c6a869bc9ca095c73af2a05d395105a044d7837d1687f8519ced256d45a883a0b71cc5d433fb6eccd6684f3867911bcf0c370d140b8d553182da16a67f4c620d8093c51a609fefd5fa095fa0cd654abcb10a3fa6dfe20c33178b6f645a2e71461fe2a65000cda78a386513de8211a4be99d8d9e9dcd68c6610258aba0baf5d401db0a47ed508fe3e8f682724c566a8373fe11571ef2684387fe9b326dc5fda0d236429a1f0705ed0535886ad38ff261b1d820d2bd0c66bfe58e6d7d049cdcaa45ebc3032935f7ab2be8328dc6a74dace44175f32730e7c58e97d570e2d0bd026b0de31207a8e771fcabe79b133d719b183778a0af60189bca51e10ab23897d6cc20a6f5611c9cd4d84062fee8e719375c175b62c5e7e14c2982d142d7fcea7121f8eb91effdddaea60487d119e50eebe35e44b197b04fa8e5f4190671d11b00b927fb859604be824c6cd5649efed767ceeb7068d7c636a5d9d753c45a221f720e2f44d9572370c62f77cb46629bfe3e95cefed7fef8480bdc3eb8de2680ec9fc51936fc3b033e47954e646ff476c81988b35590d1ab15f3db95489fe846158942dda755df84b1761a01befe200e93713b760d118e368210779ca730dba61394f541d360eb1157386716eea4c2410b2034ddc8bfabac6fcd82bbbcb02ba7940ac4b250111440c5d8a081f6052ccae7b10196ae116ef8b6baa4085211a14c1546d3a09d2caf0d1ed33771719cb0ec50614aa2f08278d5da35ce6e0aeebd0c832513624ad574f5ec9be153d92db9234d5366a1e641e6934112ad4556be8f209a1b7cc9422f39bd80ebff8178cc35fb77b535cc3c99f4a10d3674d7806456e4d5d1a5749cc599861ff84b22cf61096bea3f60924355b39ab258eb61ec85130f37eadc4d2c5ce1adec4b3af249b51dd6acee1d58a5e0c87262bd474457cce0d2c898a653f6c941f9a0bb12553d52b4ae33cceabb3bae676bd84104d8369b62b3522bfa973ebc1c9a1e12101c13aea907510073066e064f6c17094e3a554ee70538e2fad7aa1704a0a2a0bea99c7d765769a8adadcf9112fc99b22edda0eb5f99fb848723d42afa0cc1a2661d16ad68061440fc2f83820a9b539f910884af175896692d65d87f41d6ef9060107145bfa918ce81038a2b9fe0e232628641b9b91aa774c541a9d32a124374024b4724d12affe640b462d17ad8c03fd55a24a108193075c51e3bef4b0b41b8c388b3aa658af7d5c69e99bad4909cde601ef20d943930133952b3474a6dbe77ff346de7db01f4f804eeefc40a7b65007a509181819338a5517bdbdba246f1d5be79b6ed89ed08d6838fa32eec70b5489c94fb8aae30d7b0b7c7c873f0297ea18b03a995dc1a8e37528b89504b59ebe54813cb2e4fd4bec3da6419cfa1d1ff6155cd543d3ba9c1e05c9333155706fe46addc3282fe903670a97577c67cac119898c74771d7a8ce764857bc9d3694c2a288ce0a94639676a4f43a984356dbc2e4f5cb9ab0dd2e35c68e2688178aa78dc0f3d306530fc90566cc614db6330fc6d358558123921fef4c4778b8b6140843f63a5568ec0c031b017a2206436a8a746cb6bfd1448cc892e1554fdb9b20a6a0d7febcf24ee6bbe19f11e0eac1036e38c7c9f73942c18ab6accc02f6de2848699b7dd3279a9e5ecad3aa78855479d7df61f310b94f8a3eedf98f4e44b49c9d63e166fa947ea8d4324831489b28d47a2005f0cff497a4048ebb0254b063b2c520bf4906d1270e763d1262e89c223cf79e1890f8df0b8b482875b141eaf5ba23cc7e7a4d5a8cec02cb3777a77a7961abff7b27a57c739ce851d3ec2b0a75721c61c8da6e3fb43f7220136a770b24e5c9c8776a812760af61d310f3c16fa35e3afa61f54cf2da9284e97bc87995dd452db40fb8e4cead0cf6f24299e5fa7966242b7ff8fd2d45bceae102d984f672e6f7041c79dbdaa2f0765b5fa9913ad0e7ab1d66cf850f4cb280254c36807c16cd3607d789513c2f3b2e49a9b5cad1507cf41bc010a45b7393db282f8005cc86ebef7c8bd8fd9d0e5bd7cd49c9e8453f3fca6e0f0ed9d6cfb2f975798de868d25e9c8a50a903d28bdb468ffb9a40bddcbafc96c28197e41a8dca8ad3af36db6d8817409e35a981774970c6acfc71b4096ce23b1cdeadc44bd047285d867c471c871f9ed04c55bb1a7c922d5fd80e6f2dd48cbf660f7a6411cddc1468cb947c5205be7261e9862977ee786a4c933af2c7d772db30774b50da34a1cd1f51db2f587817241fe578214e4b3fd37da6020ae0f40f93e8e4646dedd72d045191d00a999a194e511d0aa4b31d13a9592f68a26b14b018a47a64692f63c3dfbe2775e294e411873e3d54f94636c22f06e16c861a023dd04ed618b5ce3dd6e1d8323a2b4a8fdba2d18dac7ed35d815335f5368d04f6322c0ca2c92bf5fe302baa2c04019991042adc06ef4bc074c91a708e3d50f062cc9834a2084e7d5a03452026038239a90d72dfb4e816802b1579d7a02a519c99451cba631a9d526907d360b410d6d4f1dc3f370f550926d0a1deb4daced6fef119df2bd47c096749a2009992aac15045e8562be48f5e27de31b63a58f91721766400324026aa410dbd30635fcf321ef2b4b1c3f30f8a2e1c10d3e6a5b1473d928299135e53213a65008337fe325b6f48166ee059510dea197ecdad7cc4d1173c94621a40b698775c40fd838522def73825b0a7aa3e92a21c9cf114ddbec6b983b368442cad48631bdcb475bcbbef902fdffcdc14a9c86fd0966967542ecf7ff102037ac71a5110c7ebbfec3c0b2edc22647bef826fa3f80ace99aa79b95bbba35b900e0e956e12a072edd67de50e7d09d77b049fb4b0e9ee0052f2c5df2880a7224c3dd3bbe233da12d25c2a246b18985d369c21a00a4548c674c26dc74e2f2484c21399dbed021850f2957309ef752195e48adf2812715cfcb638811b28f4318417c49fbc5e4a863a06bbf94f03a5cc2bd998fbde2f6e9caa30c4427f3eb59957c074393960e400c30c19dd2a9bedd94a485923ffc43f5ec6177b57b9d387618b899cf72e61291bc521e065cc39a2f7e6bc7d333b2c147216e771ba32ae99094e36ec798cce35e762d7a67c8b271607780c506f89340c795b17b4a58be63b92b870888a859c30defc7d3fae701ceb52d784c8c28672abdbb7b5fd6fa49674c508a5eacb9438f96f01bea0d8d4cfb3be21cceb96f7f5f7abad66f1ca1cf391880301f89968c72e6d033ca275d213bae6dfbab185fd59f7bbd5a3f9e3d1ef7e7896e3096ed961714bf2aabbe7932b76464484d38efd408c62c8b9a57aa933444983d550bfd709388ff2ecfd9df675a8746a82c9b2ae9c2a3607cf58e26f330fb08a7dfeccee5e54905840c9f8e987ffe0551e414f6c4f5e2cd8b24b06f53061307d18d18b260bf5cb7e09de5a7cb7795825c21a62b471011657ab692ce3449dccb0778c48851b52b5a27bcd11b758a885385a29b3a8cbf6e9b659824633a8099f3d3f28331966c255d8b1935b5cf28531da220630571a129a7517f854987ad901e9d5c30941beba9f72df20345d08b66cb807a0e824b583e87ec4c382bd347726b35f22b409dfd05b5b88a8614e3df64061ce9ce61aeb210f0334fddd75dfaef82aaae80aa915b55ddc9e20c54760f64868e0005cb4d98842b37a9a8082ed6d248b4270b28ba8a8887b01a025fc9360556a024d33e0b3280cd64424d2c9429da8f72fb275fd1e612fe3e843b6403dee83af74bd4b4cbff53992e12c9cfbda559ebf4a87a07897b716e673c53d129994bb6bf7124368326fa969a4a04b6f12539569ed102086ad54e52a547b70b946235703809039dc1c9de333df25cd5a5ab9232ca4d857db026fa480c47eb7b888a80f69bb911c2f35fa08ccca6a4cb14ee899be15e00b6965c190a9f3af9787c586be2f7ca14e21feedc1641e2ea32289afd0fc9064619d96c257f396db3cbec5aebb0515cd078e94c267a9f19fffd021c9e94d7bf8a07dba6c96019075b787ea5beb86619115e82de9bfbd3d436ac834bdc9e826f0b7abd04dec13aa914640eda93167f5be87602a152aa6062c4c3ec282318ebfbdbc93c631e5e8a07488c1f905cf258c66dfd29029124cf3ea5d2843ffe4621c22c945512c7792c49d125be5f79d0022c97f94d68e2765101ea8e3ee44ca672b21454a19193d1b08d94d38c80518cf4a96eb3a176ccf2c1a59089bb7aaabeade6e0642bcad0fee1d7f8f3b461d07836138e170e39b038b57ef0165b0ea3717b01507f7d45b1264976ddc2d4f90bd264a35ccc6eaa591f962a4b14d113a5148381e925e4ee023acd11c8fe91e73c486f86ef326dba4afb6e4d1c2d07e62882a96b63f25dd23431808210e6eba5d19962638b3d5c0c52486fb13cf4c2f59bd0389cce7763fab5af2f7523a9ea81d052e1bf69f50d9b374aaa0c48c07fabc752e704f467b32667eb05bf86d46d06f5002056655febef512aa4b7040c837a9578e67c8320f62cd753a4aecc88a951211901578d16fef2994517366484858736bc89a170e78497329fba82636486d9414e9f6a6ef2c3b7f15e778cf66c677ae53d1ecdd4dfcd23adbf1d168718ec9bb4e8c37c73bcc5b61a2e8cde6df29fb4f2a92cdcbbf8b76aefc21f88085ca53ff4104e187ed598b0cbbe6d90adeaf8c32bd58ab396a908a70a13bb4357bcb9c4d647eed9b6c350d173bcdbd28b45dd2e822af9c2d8d44c06bf671e88d9eb435d2dd8db8065abd3fe1759874eacf30e3bd602678231c0d0cd0026f0bd0880a6cfcf420895886ba38873a87c2d017c8e6eb77eb27778acf16985b8f424122d6eb68c06a764b407b5bf88362f2a77f82b331f6b0380c31b37de75daf056473c34a90d9b4909d24a3859f479163f9d47bf184ed18601a354249eeb715bf2db969fa2eab607478766abee90b5375d550b8b8c00a5c07ac3b522fe6b845f986063d64674c0221fc8c607f707799d25064183be3872888638f45138403830c127a6743b830ace5a217b1931c5526d720b0bebcc4b13c8b71f1833be85bc6432ebcb8e9b3915a025832c2a558fa25a519b0048b716b453df72f6175e24cdeb6c9e9accbcb45bcd424bd7962a91cbdd9ace122a94a1fc18008324b4a790478d8a8911363a78a8617f35e1726e4a28d98a16fa20cde7e94b670d42cae1033e257e9c62c79311e4811bd0d54a0b35cfcf51cd26777c03dfc4c9a06cfe32705fee105cae01405eef89c392132b090e2f9e6fdd1f314c41029830489614f00518a8bcb40ba74ef8924a9cbf16a753bc0680a0c298cedcaf9c2b15f74245f708ff62c22ac27f8be91422c460458c781fec4f5b8695174ae97b454f204001afdd6c84ab95cac294ad945e7e6247a8f695157296469458be0d7a3661880c26bc79781916cb4514fb5305234aaf69fe3aa89bebf8fc88c2587c446d01149e2f8bc6746a11e1eed9264f48f90266e6f5b267474e39ac090385d2e023f70e6659c51c1ce2665e2af6acd9927b96f30642fa8938d96aa42c2722cb782cd1d086a4c7ff2c7ff160dbe93541b52d1b4bc9674d6406b691ed95ad65b60b4fbc31015035e389dc1b783c7193202735c81f604e20bd581764b30b84efc9fdefd0e36e656c34f789d6bc5863a03dcec91c8ebbfc837b677948ab32c6d32e2584fc2becdd494addb278282e21038fc84fc653ec4cc576136119621d6326318e36ad302ae098feaf686add07361d1b2f3e37f3461d4393fc493465f786dd616b97e073b9d5d3e4ee6a4fb821776e3a7aa6f19765cb71e8860033be2badd2bdb14bbcba2052e4239f6b8ffe90c0202d9bf38ee68afed51e140783a1018167474dffa52ac64e4a3538fe498763be0d39b4a5026234742c502df5ac5bc29571989d7d4313a71d7c2e57a8934439a50c5da05fbd81a3f8d5c33c7c80d2e405a586b891d7dcbbd3f07e8d7423c474b6a2b1d7e3ad4cfe8c3bb83a863b6543e9da3131ebc765b95f9085611b24975bf82b4e97f2786ca3c56cd38a0cba382e04a0ca5ae68352a39c21bc63d151a3b31d727b7c313bb50e6b603a693239980039dfff023fbb348d8dc057c95c27674bb17a976608860c864c01fcc0f2e339eb1f10bdd1fb4edcb1d095fe62ae64b226aee980a9b4d01a6f6df4d04e9fe81d6584ed7c97c186891a202b6d979aa0df2e9145001258c6923ab297f7b08f483288ee149948a2bfb36f29cb084875ca4c79a155eb673eb3ba2bb653c86c907cb109408ba74b879e8f797eaa9ea587fe45af45217fb7c649201b4013186a19be626c34c735369abc6f3331c3e8ccae5c768f6f589c07d02934c54fa8dabce197249071ca9960fd2c0106f3f79220141b9450a52acfd47f04b42004edb218f23f1193df95d82031a8663c7d6282b4fca88226038b2e3d07b800f6ae66d5910b47c345de117f4c8173e0399f78ad78d3e33e9efaa165d40a3868f1e7c56a261ce942932479581095cbf8f94a387b9aabb31fbaa731182655811a07f9d92a9d892f7e13b842795119d67590c661edbf7a1400a3c3f81c589e47ef75b8fe17a288ad15961aca2e4b25f75d8057d2f0d80c6dd54ec4e6e8ca5d95e02c40701d780671486dacbe16e5fcea280a23aa3706cf19cf7b7843d138db44471ba6a0d3b16fe33bc0964b909392d25ac878886bdb4e2310042214cfa770669db6aeb6a9a620d789a25d6deb96f924154e74a8cddeb072ae939fcd79e674cb4583262bfbe13cdb89fadaf27739873e6c6402064f58261187bbf3551b43e4b3f935b5621423e4df79abd5a5d60eb393ec34ae684f015588fe1c50345e8ec49073978449b00a198e658c602186b388e1bc108cd022574f4220ec3f6aa15f2191c55288b2e63e64d104b62461ff3b084426dc7648e2e847251dca4ea073bc808c160eb3a156c9b6c1a091d6144bc71616840b0260638ffc024d3545c280ea7169fac15e380005ca6fa17fac91ea8edf2649aed5c285efcf5f26a92867ec77b157d60ed4e699368d18105fb7bb5dec1c13477ef7ebb8246b6bb28e7a4bf1e1ef4260bdb36c7c95e408409f71b39624126cdc72c0cdf4eef04ee7a5dc98530d4e5227cc37bd971d6f09a66e61e01f49dde84f41b6dc22bb4dbc0360f6310345eb5f2b3b53411688f9f77f5b577ce529677094566db85e2523c3b46be63355ab34ea384fd0ed2ff147342f73dd4ee185ce67fa3682415a68eee6344f34d275744f0d33c200a33fd612d990a73db93d14f3ff77a40a57175b8ee0f6265b3efc621af73ac3fddd63e88a29957c54663325dc2d4bdd4bd8dae17e118dc1b9ed7450a96a62ba968aefc1c8d6af51b425e1435689c7ecc2fba5e7303a569dd60af1cd7980e4035453f8cab703b205a7ba0b032ed2738645a308b7aaeaf6161f2ca06907de720b68d6875b846bc58f0db4aecc30bfbfb49380226f5d842d23fb16b49368aaab82f79c183bf303f7be08f26124247f4170e655bd2205219dded958369fdefb231ccd9eddef21ea93bacaf3a6056c0b305d0bc0ec0bebcfee0ec8bb39fcb501e0175d970ecdce3d588a768257e73b9b30c346ad95db4f1d59c65bfe78ff266772229aa30279af8e3ba47dfe34f68f3a0c67d2e41d73bfb1c7178cf576741655f2ac31bd649979c38a46bbf5ad4321b8af71c774c27129c36ca4ee3fb068548c0f69f20e2a8fc450ef471ac708c7e20dd98a822744fbb48a40f994296a3594f6c094648aebe2ebb2720eb11435eacef61fbad8de57a722376aa3f6f1250a7aa7d31ac1223e419bef51925bdc3e169f12af5589c47e4c6347a46d3727b46bd3d4fef2e3eb47069009285d26b00d28a2d7f7443852f91779a55404c68f0372343740b040074d2e1858e2b3e88e9b538aa28d2e91e20c0d2e0aaefbe354e573664efe425f2c86335f548cfb0e68596032eedab8a39124c0c0976253ad29ccc62ceb08c77ee1211a35d4c1a8025617c78daed5fd55b84638b7a8955e9c9871e96189fcf7e9a5b63aaf8f88e9f0ada2afa5b66594508e3da2e85dffc269a4af7b05a97e7a93becc51e82ec8771f2ea9ce5e88d96e409ca09cea2053b443a6925d732e0e77ea9b7d481b0ef0f2b83b5112fb1221c4d7a09f14fb0524edd5ba7e600bfd05124ddb82b413deec56cc4ff2e277f49f7d698efd673dfc706df23057594919c99716155089004cd728e6aab43d0d9d55fb8ab9185d4b7dc6c317bdd6246d0579d2f128bf7defab3fcd81a9fbeb34b742b3ee0aafae72efad8e8f1636e9e8202b0f73f57fa67cea68879b896d1f94b0a660e69c55055889700d5fce79de21d03b591ebbf93c9991bf6fbd8aa9473b4c9c892034df72f301a499860aab96f29abcb691e33ae727e36cce19ede0bebb120df95350d059e93ba069fb15d11172054fb554ad1cc9f448b4da592b57b9d556d7a4eafe3692ea5fd6fc00381cf7b81557d16c68a3d550beaa630db57617e2427d747dc39c6f70ccb11a2c0afe5a4d99471c08ecc937151db8478a1d14a02872ad2a81e1eee9f733a196291b9bff282f1f83c9c098c77a3b14d47baab770c127aaa612c7d2d273bb433ea20de2c64b57d53439878099f3a7c1bd178d5d1b5531ca3f2995fe9f87f2f2a6ec387354aea64e81ea5f503ab4c98754e0822fdcbb636a2f4480d9f7d05bc5aa0a3f5ac202dc8455c46ef30d6e3a7175bef0071f608a23a15580c8ea2cdec57ba86e2cad5f960ccb630846c39760284b0686432ff3573163071200f99f9ea9fa8e119b67313aa42971e7ba05024c6a99e3bc3f98f46ce62b07d1a1412defe4ae539c2b2c80eded70390c07ebac7836bb11493b407ebb4cf0300f392bc19d480bd5d368b6b2e53cb2b3499cbe8aa5490ac5d586cbcb993e162152bd26137d82d5e0a219459c848a1687c97409724999797057987976ca797d131b105d073a53a180aecc59596d4c1ed0810c891809bccfc9175d42ddf5992bf3aa3c3d5bb996a39b16f264aa521093a34a6bf0e7e92d73c415362e5797273852f58e51e89b5ca65fe0ca532df64c6e2ac0da5fb86a45ee37b24ff73d1cb48c2a56d0d2a7b9b16483fc8f134fb533341d4c30ac134ea59ab4c432d803a4c8e3145ef03c374fb07b44edbe5043d18e74b84ac0e38069daec37bb59fb06b38155d05dfe9df66e1eaa0d5d5e8ebd61cb51f3d8e21d4e3660814b0529675f27acf0f105777a91746cf129aa858d48a6b79007f04192924039e0de120f2da19eb57bbb9bf54b25284f290fa85b30c34555d766bd0825b7327bc15fafb957d0638bab68bdeb36743f79e240cb6cdca82c4c4e710cdba09ac9ee9b55cee363fded32a3ce1b8918ded1ac29c8ce89252209a303855319bcbb27e3accf86b0427e8306cdf48808f8bac7f4a3b2ae835b6b1516b68ce43533333619dedf17ac43fbe23448e1a56714150f8d1166eeb5ff63c7d6df45341dc4fef14449b1a12ef5070df23bd3bddcb2e7961efccd4b76f8a7d5b6dc004751b0f07b45ff4e47a8d24838931fd26c9773ce69d25159f6a645bf0b30fce75e92ee8b59b58e286d128992723c908c9dbbb45c5eb8985ce9302e59d1f324da7d4075c7f3afa92f3728098082ce1e68956af2fbe764804a1d6463e79389d2f42935579f60fbb72e71dd7d51e216adfbb610f590cdfba5c364a6ca6a52809581b6584a7190679622e6cc03e67e6b7c33a20a97c3045645bc3e4f241c8fb6a17c11d0ac10e2ab6a678bb224ac769a4bd82c434a7d7ff7bd92441508900ba99689e6f383c7e79d490d3f857820c02f0bbe640122488a37469febb5aeee4d921a69ca3af449894d5762f8e585d49cc9c2974bc5075bfaeb9874170df35030f7d9c1811ea23c422fa078918ebfd06b7fda636c0e0cd9ccff8ca6e0e941a024504bdb71b7619119b75fb1e43be963dcbf777bc432b8fa96df40d7d2ba6185e0d740e06a9e295aebc6457aeb784b0d9bbf99e03bcc84aba5d4e84a5aed0920053f5d19028c7edca194076242bf09d3eb6f91dc310810e5d99d10e0c79c4f2dcec3bffdc43c9d9b4753cfcd016aa18a6c18ce2f87cf91923aca3753e147e2745dd64a14137ec118701818a9a2d7791b3556a97876eb4d745ee48dc3b17555efc0ab591cd655340dc5a0c86cd31915f60e1fb532d436bbac3119cef3dac4092d453e748c66a2ac51ae695d46312d7fd5f8ccc18a2fc5236efdfe9149c1e9ea56e3b4b280d0cd1d66eba34b203cb6701bfe1f43e9954796bf181e5377df7255af0357398b825b3ddd1371d88c38a8f9009ae10c9833726930ed2e6a763cb3b8f4b6d81427b56091a4abd7295fb84bf563a043959955ac3b77ae0ac0a3148d4975b719561c4e8b771e1844a68e2ae3321e8951b21de7103825116cdc2e195738f4aa360e0c8a2a9f3aa185b530171867c7e17d234155b1fda40c99af479ac722b4709c41919f4c0e39d6b68ea8be988890f70e07c21404de0185fbf1308c91536e22fcadeb062017d777196c45494236f1d2d2db86230998ab0a25f26864604114d4635a1299d711e09b309783504a6b6e2ba3a75651f04e529e72a6ce20af009f661a020c0ef305d80ac0af90dc4e4413113af5de69893ab773d29b2784bfc9174fee41f2a6e32bc7eb5412158281b630b9520afc3aa8e624589e17770a6fd82c4e869316492511e1eb47f74d180aa9363e00092da49c93f7c47afff09b36c30c23342db816723e8d33170d8063a82a91d353a97d6bd3492c2106e9eb0b1da23275a1bb7c50f60b95aa45a7a55e5368fc24bd69f93a1c8209b5c29a66249e17ff177dc78a8866619686090c72d9e3230736f2147d2607adc263f69a1e6e1a08e979f4633d434a3c2a0d14b5076d9d36ba48684e7cc622e09910eca1a54fe9cefd51f0b396e85cf67d7db9115606750332c8ba3a7173ce38ee02ea1bdca78cae52729041cfd3e68c56d5aecca57f0303c3235aa92eb7d662da4962d9052b0f40c7c7f89d7f84a2dc36bf5d0e475a920edf5a9ec30dd9b4312ea26b8f691b7684422079faa9ef3e0b9f3eb212c9d7485f53a114b6047cfdc9172c53e35b8a43ebdf8f36f02080794ad36c8d9cb251bf8c3e292d6648f767ab44aca127a1595086bf54ff1258970683d2a73148965007d421ec346746ae9d6d8e95a6d51ed144876be3e61fdd9a1119e8d15a11db906e8732e8a0a86d515b00e14c6bca04264211b6a5f218bd288399bc95b123f580964fccf694306329fa82e66e8a03a2d6ba0156e3a9194c590aaba9f5adf3ea558e2c970b94d702bd3ad698a3273421f95bbc25c6dbf7d4626d442932b79b0ec8b089355eb82d92e0d3a40a85b07ec05a045819291ac0d5eff61dc95ee6a98e1d57f29df637027621b8edf2208fc3f5505bbb4e29403abfae65712bec4fb3affd56c19f1cbee44c0adf4b64d2bedb09ae9b39daac3c614aff7e0a2c89ba3e22e7864b2caf3a3f1fca59dec7c20a6456efb32d0e40574a70387207a1e0fd05dfeee31312f907a4ecf72a8a2b99bc0ceb8690d7cd5d5a299e2f12a93529081714faf10a99a26eb499c606ea5a7e85c7c32a7056f8a954e286190bec00bc6e134e6400b0db3bb102af7b58fa81351d1f61887051611be64d7cd48d26a7a4300410ef7f75cbc05af6a1376bdfa2a531472741ed2fb220d7a502f73b2d28719fe5184c2afb411f30b2d4ba90df9f443e1e0113560be2b5cee9db1e6b670e03b03964ce3345394e9ac760d06a51a9b2a7a7353d4805c3184adcb9671a9aeb114963672791f2162148c3c2696fd413dd4392afa28ac0c6fbf99311eb6fb92ffa94d3a557b753af33b02d17ec16e5ab91e5f12150124bbe5b6de9f749850fd93500305d4bff4c62b1a0e8e0edc281ed26a4a36aab0ba224b07028833e70820f65f2a25a34beb7d1ae663e3936bb58d0135be36ce8aa46b9310645763f6322336358794e9d57e7436db96ed1b5653b5511fc07a183f74f2e585a8398edab978a923eb56ec4b94ea557573033d2da993c9bd631c7c404650ed15f11ee8ddcfae264936464738d9cb18e6c1c111a15b15ec8deefcb6878bb90bd94b031f06cc595f5436e3fa9d677348ea94edcab9b5580ae19d954070a576fabd37161082d7ebbd83b157c4465fe31ea89e4931b2bf96766ee7d4a4a81b837aea6cce8b141c716ae870b5de3e7bd23df7997134ac2291ecedbd9e9881ee567620475b0b8249fb11ec4b4ab563782a39ee86ba46e12018cae1cffc4ea4af34ed05383740d92785d10a1b77582b6212e4f7765b9b78d17d2644327cdead887720f5659a8489ccc9ecd64a32c4e56674245f58632d949b7630928ca4f81b9043794a86e4d85e75b7741d816ddbdc1ea305f552ac0a61ea62992b73954c799d701c1747ecaa04cffab896f72f05945ea18091f309b113f4280dcf6e58df68e950b1b2be00c8a5116e66e317d344fa5d44c71e98242fb53020a0a3d102bb582afbe14a0c6efded44d5e660c494a970e3579f14da69f7ae492c7c4be592142de65578cf5ed62a651b92c46177e8a8486213749c1c29dcffe5cd12cabfce1f8808bd839c244e5f12ae7186c44b3641f65204c2ba9f544e13403c53d34465d0547eca28110905b8bec6db79495a5722f6b1c4a7a4b09235938d45b8089fad85524892f992b8afab85e148c16573864e6bbc8589a3f059c15d5cde3b17ecc42696e6593d020e101b1bfdc0d35aeb579105eb4b48b8b95564ffb5478268e8c7717852c4e2e0d5ef53d2a725fcdb011892a8113a1dc71b94c79bf87deebaa7648b58996d4c84a44119936c62910a3fee9e1eb5c536aff0dc64ed395c99b46155de20fbf2d6ae1e959c4322479d98fad23bf98bf4ef303b831d8153132523529a085f72eb6fc3c6995d05e7284b7ee66b7ab520eb123a68a7d3b8a005eb242e6b1599f5149f086cbfa16affcc0ce7416edbfd9ab4fa112443d7cb082d43dc3b927df09301901d122a3dc1b5e2e02d6d1d5ae5ccd8530019fe3ae7b86713cebc5be09769aa04ce538d794e2e29a75e2482d0aa51b223c71073fa1f37f2585391599c8ff040fc4d7dc1985d986993a1a82c5124276115be7359ec812a72256291d2d643531bb02fd51148fe5aab4c38d395b79a375e78552cb57e7efda05a29a866c901166866746fd94f445e2a70cb01484d7b972794f7852c012dae7ab7e9b93c63e1f9cee8698ca29e3c1eb91389ea52b388fc3603d9a5b48b595aa29db6ecbf65c585d72745d1982b95d930020656f2431017e2e1dfe726ba2bd92c13bcb75267ba30c7d7c4ff863be995cf725f4c57abffeebb63252d8d686d2a195b4f355686e7e965b98f3e0b6f81a9db9f26196e3a20a70d0e5d45565cef23d6bc281a5340d808714d5aaafce3a284057a1c264bd92a75037c36348698d24e06ced415ebc77cf54a0a3dfb785732ca4cd6ea1a66212b33359d31b68f40451beef54115b8f17470a5e47bd7549ac71945e59d785bbb967051e3713c9fd1e6df6e2044f1ac631f67f7400dae1bb37d4e8dfdf297e8256c6267ffa7bcbedf37667d774cb0e7b2d206f0d93f8a822491bdd3385ffd3bbe59462bab0c8b5f394c88a77fba7b2b32b643042e3b0b8688470a674cc9102805c4ad7c8c7b2a2b84fca17432af891fe18bb48020124627d175f0dd88dc456a895bf5c64daaed1310fd592ea4e86ab1b3d0c31739669a735deee4d09d8cc86b1aeb6ffdc611024f44d5799c70664562b3106c2a8a2a98d90de4bccea52b4c381d67318538456a8dc0a962b652a7ee16804219ebd5185a36850074e26407ce0784d1d98d425398f911e68567e848fc32cc3f161ebfdcf0c9f10afac77f19550c33b24f1504206769173144deab6ea84feb32345f9ce3dc1346a9211423b0e43c47cf29958bab49a5bde775c9930b568b2278a2e7cf56ad73bdff9120a3e178387fd08cad926a8ff06241085a3c170e2cbe8d2e03d00504fe39d6a79670b5e0fd1c31e29bc91b11459857049f57227a77a924314cf60b9736e7a007335a18e10e277713f7e81719fa7e5e3a6a4bef4e1f15ceaf732b61c8ed5b380ff0b20a3e46f9db0fc62602c379714565c442486da7dc4b67ae5659f127c9184c1b1dcdf5c9557e158d7cde4fc0003b89b717ac918563b04918085060da71f12346c4b6d3218af6a9db8ffda7d0a2eed2c409fb98ea91c3c379aa17d65bc5dd1a8682f0a371edc9790fe2c4ae78a902c62ecdbaffa1d57f0f0df68a6f69a9057d72d19b496dbac495955e5bd98adb28557ae250f0a6bea1669f244f44c9e816c7fc4b60fa04d16408a88db0b7c25991deb9077eb6d1aaa3383c4876ff2705e8085457c5f6cd4519fae8fc32c04783028bad11ac9c95cc9ce9917ed72a0f1ec21b5340d8e7668b441c80de97b8ec51f4b7ef53c90ff13567041917670ec7571061a3d4e287bf09b68d89a2357244cd16a94e620ddd308be852a6da3cc1138273d3c4b2fe66b4be8391b4c27a6de890940b2ee029d4044ad018eb201e5cc68b311d9912d3699f8a6a480b7b110777611357283ac357189428505877e532acf143c81223022afdbb648ef00ae808d38e162f0e32b3624d5da9f5bd9ee0a7a3610ff11023f566d484215ac8349608d3cfe2f504131a1cc173b99af32997471b079ba4899ff9331c0c2f55cdb7a3c8b847998e107c1a13477aa1b02a0c9b85a0ab71c9fc6b9fb6babb80aee2f6f680432fe817a2585a9f102cf6bdc1e96533851251528c232b0366ed01d5eafa4d95362ef963b6416ddecb50d1e1d90d0e8970deb4c1a46787eb406c0641fce887314ec1d4e815fc89a15cbfaf2f37888d6d033e8b2348a6fdb371e9b8f4961fa7f0b697ffdeed3eff5b74e460c2660a939d6a212244a9da6b345dee78411b2c2b5c751f43160d9d245f9045ba56423014302e4e93944b4f116cfe997ac1399f07e807c59abf4225562989e92d76c9fc046a73062f16dc7e65082f44d71536fb3fa8edbeee22aa78a8efa0cf0e25d85876ee434b915f74a27881d51bbb317046fb55a6ce0139d3e79bdb87d0323d366a5d4ee529e5bdf155c1ac0922fb71a61426612acb83ee78ca8dc9367966240e097f50dfd8a26fac37c9dc80239a4201c361379a671fa96fca1d662818ce427c3f0521be49d82ca09bc668de4762559dacff75d3f3501b79f9bd8b9aaf8a90e3ca0b59c695d2c7879c404733847e5fac9f85b87592d09141bc8829fb21b1c5d58f038d81a3e24d6e00ab869d50d8e0ac6e8b355af8413a8c31425bfb29ad6835bde87ea149f6127a0ab0c4787e7e16ab4c610afa175b59ab71395cdfb1b1878902f18596cdc500b8b726ddb1bb1ced9d45ec3fe151a3c03195db734a8518c77e2262c7a5a5b57810d022e3b298dc4a760c173b3a251f2dbf759091e7796a059bb4ba8d7b32be7ab56ee026942f6ce3ba962815993754e75f2b7c6dfaf5b8d3cac5421c2b7c2b0ba2dc4efab9f63739ab74b8f3b1a42236765e1a73ee818b434ff19aa6e59df1456c31b009d10718ea43e1c92efe62c12ef761414328ec5728ff31a6aacfe0adbaa0a20a0d3a54accaf2d0206d81d3c3414d2abc03040293be7553cec73fdb3a37da4c229e8697028d857e261184fb3fb3c4d52bad2b08c560bf2b1a6ba0ebc8fc5c907784ea5f46b46bc6e476b3b3f51407936c50fb5125eb79a0a6505ac9d65f8bd34e2a14f772c828068c9861126b089e9eb2c866584a1cdff474dd819a5f4e619f5788430ee56e0276959020ea80439dcc39ca296064bc2a4149e3cd3672c97a091d413bfbb834d59a827edfc8088233a962046a337a4f2f34f43adb2132a433f11e71ddd4912ad2a09fc1fd47153031af5c13c99f3c2278a85abd6a8b0a9096e846392aa170e44ab691567e746239e344f789a452c6f67b9aecdf651661158796e30e1736f5767e9c26b0c95aa9fc987b4c7ec43ba8dbfd49b3e9766fe1b2967acc3899b89fc1e792268fe9044fa63ef6542c4e9e65720ba327de740b2e108664ff6882407a17080eac0e5ea93578537dc07a8171aa81983b722c57f5438832ca34317577a74ceb96920bc35903e17722739db2ab4f3cd60dfb9a9be8fd689d84a02bd5d1dd0c574770a78348aa7411c68cb3ed78fe3f4ad27bffa10f955e38d1d328f8726a9cbd4e68ca6af5dc6e6fdd6c19e3c000b50818fe94c99ea099c1c7b7ab6cdb21adce15e4d6c0992c22850d1dbbe9eaba068f3e3b819b384d1efd097235b1d030b6b0a2fb4dddcf92f349fce337eec4521d9d63dbabaff5309dc27353fbe472de43c5cf018113eb283a96b0a3d631171ef6e792880e2cd98c35e2573e87061ce63fe6a34d8f950d9c7bfc3473d501d0903d3eede2906c92479992f087be02f2b21cdba0e815eface19b7c92abcd64872bb895187886ebf33561d724530c4954a9f4dfc0473dd84cd17c5eb2602869e25eed5bd14f89033cb11a7ebeae526a47a6356c87ddec4db3446cae3a0f0baea0259df09827f79b6265ba3beb732812a254affc1786f3e4b0b80807f33960b83c98d159f7583f55e8892b66e44716a002cf3e1514d09f42ffa18187f0c647c67aba10ecfa558bd3bf0e050dda2f89a5009822869ad0674dd84417c0913387075b6ec384b29f1ab87f64c5de0a23942bc17d1fa9a98110ce29a027d708c256fc39d0efdb896e5004306d4d4ea0de7e5f4f6cdc0c4396fedf988f5f95fe1825da972ad3a78d785055d5afe4b238da257003006772cdaa1d86c1c39da63c93205187bb8756c4c4a02e3cf681e2c2e5041aecd66e01365397c0a0d2b4b3273e3295eb3102566d1ebb57d30770c649d43f53bab57537104cedaefeecba0edacd8a75c30f1bc806cd920b48fcf623444ce77988e591e06d24e4df1e9790ce2b64139170a845991c1889a4d1f6da1740895f83bc182ade909f85643a131f096d9d88eeb4fa85debf156115abc88cc8108806ecf5538acc858e4228320b9362aa7d7b0168c133fe9a01d190eefc8611b4af2f3fe6ee0c078a23cd5ad32be6e9cbe65fc340eb0afca8b44a91b788759212baca68912b04a12565842bcaa3cbe846e7b890d515a5780fdb682df2e068bb07dcfe7fe3106a25b4173600d51b5686a13ef0662c1bc5307c5cec8306a7a2fc0bc013540b4fafc5716ae4042f937033fa5c2edbf634e68a0422f7b0f570c2af446bbf208e23c95ce1dd91b20de89aa3746b84b0c2a089cde45092224a92d144a36876a7fdcdcaec018f99d61662f7cbf87b6c3b9a0a4b2049dc561d78b966a88f674c5e473918f46c6428572496d108bdae982ebad10a7bb9ed57209737c089f1d4f2ccf6532e97e290cf4d57a67aa3cd4a071bea20298be24c847f56a22c040ccd8a2ce63c740c60b25eb219703ea9a0b597edf0192cf2602414be3b7c486a473f95508ef6db1b2efe0eedd8a42e8a8187deae0205d13edb86109a14a8eea5d8b665c4728b811f2b33c38b599180968b6d126ae6c5a99b3c23912d5acf382358ce9f43c9972010773c0cd11e9fbace987f828ac848877cd72f021a8f78507ae6fa5d579c5e5e8253c225ae26d5d62ad04274e71cd9dfb099d2a74429298e4ef1d3b5157864225995cf9c759e1aa5c1d58f428b6a210fde5cf6f9552e778fbd98168f3e567930b0833e1ca9e613ef6b3ad39aec8364b96d39c4cd2968fccbbb36af92254258a8235407f94cb2c6aac9aad979510299e1b2f8cbb3a54cff0755285b94334e173a42810ff0c0f089942cba448a00dc4a9f468857042e3286f05741510064666ff21e5aca57e6f4a8ea2533e61416a76bd9fd12abfa68848faf84194fbdf94a10d9e279572bd16e4bf2a27076f60e48eed494fc6a39b41f099f6e69a60c65a9ff798132a65ebefeafe40ffa057a3239b01c85e58526d30d420ab3d40cf5f20c3eaa3a45ae8e94727675ffbcbb01abf88f63595d591cfc57e3e93e290b4496b68c9b09875246186c768d59c122608a9ab5a8b1f69a7fc7e744db4c54e93f75019bce8ec9f99d51fd34787f5fd25dcf678a2fe4da8fe3e3678053a4203f1be7d8754578544af06cc2c3ef471658a1b7d6450f5875ef3996da582f6ecaae7f004155f05139f5c735394af50948227e96be58a1c07ec3668daf5c2f8384ebbb9eb35d257126d1109a9cc265200589e0a5c734ed27cf386cf5d3ab8a229008ebdde8377e4d7e75c4b2f4b7abd45c2823192d5edd39103b9dc846246b1085be1a21f4edd755764043191262a1c2252ed3e9b06adf255cd0e948ed82edeaaaf6fb311f6fd7cf9ae22c283c3548ba9799da21b336644315959f3f97cddec9c6c20a643ae6710604b56088439a7794cc140ee917e2f4138b6ec7b741f45f72ea3db93439c8bbf7095c07eed2c7dd9f569c1f5a43255e1b4b46279a01120891eb0f67f718b6fbc1e17a5425d8bce6f3c36d2c39ddbf13e2cdcd4a6564f15716310feb0263404ef86c162d56b9ea91506f2b8bdc8d5439d7939a35717f94036f27532413222f4104cb9621a0db7f7d2a104d3b1131244b208a21be30f31bfcd6361e79cd94c699a465eae78c90140b26c03beec5ce85a50fcf4eff0e9965d75fb28da4b4c8fb46f991aa420fb4e9f80c7dbc05767647d0ca0db97ac422d904dbec60bafd9e92294c7d40a299dde3d5abebbefbff13d3247f599fb224ffdde4ec3d24f9b4c13b13062e83919589ec4f8745a3bb7027913beb04b47f13da3aeb3328e4fdf75546fb3daabf5dc11f921e89b1360c1b2bb8838ac34a002aa0155a03a4a1bbdd8b719af446e3109d01f610d919a9ebe873a9eb046e65bff8f7d18dfd14ade52003e34a69081e3cf04f85581ae2f95bed71ab8acd069ad382062945ab821b0bfa04c9e2598124e2cb5cc5f5a918b1a9c8a9fe40466f1adc5f3175222c20f3754be66eee98ddc5ff5482d2b5e775acc80f6682df10f262b68da81fbeb0fc30b2f067a596f8658a58abd5d9c6b522c6150ce87c962556daa5c931625d015b817b902d4295a19cc14c2c4eda69b26fbe040a11347bafd400e1bba608a0d084d1c6a89450cf89ce84a123790b48e1a3eb95102a04cfd2290379669e85137bc5e063d64550c807c3cdec127d4bc9908d5bc212671cab3fef8b1f2b9320af50a0ef20c9b5bb13a4020e979f4d249cc6d6b537e84311ccf5de096bd2bfccd2ed4d4e7aabf1879566d23473dcdc8cd3e3b6589a1cebd60dc63ff38813784521a74a2042ea7b30643d8af7c434dc9a2d9c5a72751ead62c04f1781ca5f1c8f342638b4ec9dca06b1dd485c4948413514ca3ce90f484f86accbce1262ba3a6b82d403905d3d5b98d403eda1d6363f66fd7e3de764e2b732e140d0b1be84fe20e9c11eec88b5cb71f209a187ac6d37ccb97448430b1c03a92a6c216f8fe05368b5e62bd69d1ddd54e3a5b79364cc28e499e5bcc68be5a7ac345f7e9af08d24f65289744dd7d530b32e7c2e79828677a618fffcc858a4664bc372e9d0f0ef424ff769fb8277a9d6d59016f1c3aaa9b9a2f8c551f47ba54cbc08604aba8b63be6164c82d8f17697b070325018f090863e6f6afe913f1e9547dbfc4d1065c9f8d32957573b194a13a0f89244dba9d8edf8601915b1503f9ed31c58555bf473e33a8b728bf48e259e0dc9d38b047266e60c46d31fb2d54d2bd6efe25dd7be2051a8d279758f15db6ec885876b18a529e685e51c60e1ccb697d7e43a191ad97afe79ae87e008de9a42b6afd44796c5966fd373b78996f2cc03b75e4ecb4eaa43a9ccdcea2534a9a842b1d6424b416b2a99cd975b66aab4b91a2c245dcb8264847576b6e00d96b78f73ece66e165dabc0cab76a66890ba48a21a984f8dc0c2744eadc2f1a5b40e4e594647205c11445425b4aed875543f1f696e0585251d04f6838f533c2c3e6205671618273b36dad4af054c07b76192bf131852132c92712baca383b8645abe557978e4804b56335a668b854d2ea8d2c150774f4f25f9c225169878831cbed551523675026a2538d0186d6ca1e3b84db6ccc88310da2f0788d79de024ed6c6c363bcc7b8bdf4b672727a48546d2db51e0907c5cd3f12c0d889252da3f09e8de972efa27529a1acd61301931ae64eab7faf7619feffba14db2f1a26671b5ff048c31d83857ca8b17ad3928a9f87762356c66e2e15768ab6343dbbf303e5657d0388b66ea2728ab68db9d2a650cfce5f18346b4b6c7e66a15b6ba32291954a8ff1f965da1c4940ff4f91903c9ff5f3055e0758153b83f3ff4b284f5c559d3ebebaf716bd15347353c74c355cbd499c6cc0919574eaa7a42a66ee270447d16d2db1f97b7310abb1e051961242a082ab3b88d5e168681864907411d32fb3a539a2cdcdfed3f81877b380637c768a1bffa39c08a3ef96cf3a2e69b9f3d582ec711bf84c7d55febe694adcfd6d7838e6653b7d53befe3392f402f533bb4b16fd6961fc680672b6e58353ad930e5d111215590089d8df2fb4acf687e14f1a16b769fd0ebadbe53bb0457ad06957819d34602c27a44a90315d89bb3f6e2c3a0fbb8ab5cfbfb5380f0e105ad06128af36f37b79455276c054a810be9b29111dd1c97973c589dedc3913251b361b0633e527bb313236347e5cbdf44ff812bdde3783d55d86079bad8f881e84403217daca8b87b4f5aa1807d3e9d0e991ed045b10726406123ec8bb40526fc1d792c7bf71e30b90fb083618b751249087ec9cccbc1e7c2918b9504d43e7363b8740ee33e93c022f530cbf969248d1b238a3301a957b27abb6cce28df77c8f186101d53e48e168a2ecc1e142878ff8c4ee7554e39ddf6ec1dcf717b2b80ff8488ff64878fcdefd648ed5cecb27e3884f8da5127076ef6fde72d3475d43de498b720352a4f3baf4e650888061878a2f4e81799205710d900df51e578e93c22646990a7b83f926f98d3998e8d36c64b910cbdfca78caab8db418dac5b320755171e5cca04a9b62d1ad84ccc167bf57075bc7548dcded47fe3e3f11cb1538b71e27a9c4869c275716ac82c1affd826acfe0f7c820142bbc6f04c28e55d48e57b77ace0eb7e60756ff588f5d11b3788ed096f569ac5b9f393de4e696ca5b6017e3f2ca635be7303a685e8be3ce2f9d4251f559f9034b83f4fd042df2d51b353a3c4c0d711caa8a7f7a780023b4080ce9cefbf970628feef77b85cf28d6c9a8323da50133851491991acd75752eb65e3db63a801368297558d30de890b91d03d6ce59568cddc3e3ec53b659a82bf66d3717a83fb9ac11ed497a7fc4489199f5578bbc55fa7e9e846f3d38d7276929b5cb418aeaa9bd10e921aae638578870e69f6c9448ff8d427a1f6fac25ba836cdadbf2c86fc264dd324e628bf1bf4455f4a2c586c6db9fb45160a45621b05a02c8890583db17cec2e88b6f95f3f07ff666ebfb0db3ff4606b60bede2fb33fdbd1b7d5d4f6f6472fddbe8db7f7c3be2731e9abceb3c20914573a6a7aa0db2feca67a889e1b4647dbc7835a736646e875d8151d2cf4a8a1cfbd27d38032a8681004be0d08a58378e1ec416edb1685d07a9914e91b9544a2b36ce66c4525fb725e41a74348c7b45e5b09f047676ccdb0259d1249a94bbb8748ff8151b6b10d0e2d8e6908c8ceb46031445c800a9d6c9feab14eb01e79b30b34902e3a301ea2ce0a69f81718f3b656c7fb16ac026df73f01a7f5d238f8bb4c13328fc2154e1af794747d6b839a06e159b339212a40acd6a63e0a178e3c67a86a19c84a013721f8c25c158b42f1315a5f99f869a603375d4888d1feac81a25e3fa314f495f5b19038adfe98290a71190990f624c5afa4a86dc92c2a94ab79d05484e9fb0b435de95adfea85253ad45828af3304857dd1d32edfcf3efd4b1328ec19bc7f7e3ecad099daa292c6195292e6c7703268e7e56d518c5a1890b95368ca33d7723ab7504fcdb4bff8da3bc845bd2887ff03bd76a831c7670cdac9a643449203d2c8a2142307a68fecadf192e85f7e87c987bf774183fea3e42f173fb43c27b02fb02f1e40327395277f4d28a25ff3a9c532c4cc89b3f8e56064a373f4603237a68f0c8349e628a2ef600a0d1d4af5b5e92c6e1e4b97dd5847c34f12ed2b4965feb6e3509fdc7663f1786ab6135e62e4ebf1bfcf10cfb33b8c3bcf5a8b830f0063a3be48481fa847121b0bc8dfc01adb219b52fde5b807c2664f369940faae6a4d78f262de0b6a12a115e9e9902128799130b40e957e704443481700abb24f1a20488874b937ab0c275ad745b399640b51721d345c4c6a58e502134c706d10641bfa693634a834beb94a60bc8f103d25410415e2b0cd0659159034869d2ce62d7fdc8db731c042003b149634272c87395d7672ce57c0a4c515bd9804c78ba204c7b625db2d0143213e39a7e1dffae4e8d77b63945969f726f0485519781d6f672be0552a873fd4b1d1ccc75b0eadbebfa11c7a58102a2bbdcdb1e63a5b0d1da99ad792289d50afdd663a93d937e6fcf4573720d412734094d155e8aa8f40cf0e823b78ad291e553d787e3afe81a54746679f3b519ca5d2a82316e21526bbc70da482b56a01dbe615db680e8df230a35c90573c5f45da7716cd5ee02022019d11f300e30d17dd7acc0e1c54d290756c5f9291e367e2277a478e041165890ecc9426c0c03791ae1339ad6251f7c92a2d17c9e297657815ef0e97e274821d2b50eaa5f9f2d0d67d450b2b9234ac96b270d9420348206252a789366e09ad34813a5c6045538629777b0be8d28c242605981b5fa1f190388e028cf6d9aea54c29ac06b8fdbb15606f50b6f271d9d76b5d3e1f6f50e0832e279e93a3a2bb42a95ca7c960dad5d4bfc28dd83b71d5ef73b61c2ab423997d3178ed6b12e19a73641a15517390e89b45a2746bdc1c0471af235a2297edcc570b93a034fd2853f6d9ff3c08a14940212b091a182ee25e01f03428d3f643d39afe36fac0e8a547de36ee6c6701e678f7be7a02765ac8453dd7a1721bff82c65632b8440c7e286c22ae7ba8f3801aa6fa8625eb4f4b88b8b1a5e74dbc7a452d124954a1becb042893aaad339b648d4097a87131ca65dc019d1a6c2f0a1cf855455f3c4a17d69d2c42e6db35b04586852af16ab60ccb55aa3fce6130ab016ffd07164e43195c2ea6f89e03c686a5234161dbb08fec9f9b4517a113490a070be7cccfc7b0637a57005e8344b3193210140f0fc80bc2ec41ef51b9363f1c5ea9df4d2d878415ee98948355813ad0f0991395637bc21eae84d142b51ce89776e319ebb5c05607d1cd17e10c61292c64b67f6cd63c6f9ef703bda17be03edd84b48479a30ac12f0e9008c09b82a50d7c3487120ad421634c27fe61e6aff6092e693890c4d3cbf987fc878b7385933ea78ebf85514ae500a8c88f116a0ed45d14eb9f1132e4f85bb1e350783b4ceab034f8b7373089bf4a71294c0df8c2c5332e0800fa930400af4e38e3973ae8e99b1eb7b6a1013418c8f34b4d7ce8d59c2a8736226bf343c8683502e1aa6f050dcc5a711b9fdbd7beed5871c84a8adb4d13e1b85992754a89b2cdb6f226c69595197447565b9c84641ad8f0403f3c8827e3534147d22681027c11219fe5aa87ea5d828ba1b1ca799a2a1455797db48daf3481f7aa23fc15c061fbe39748cb2063a668271864910b983a9febf74a3e55b86e0e4962366d96df5c28e5980aaefb3fcd772333e84f0674b5eec20f69e5ae830f7a4d59b843dd218c50ba0056117636ae84b545088bf709ce45ab1a82af2fe5c5cbe80495dce8213516d6c6e36fbdbc9ccfb70d0200e48d0d891d4ea96eaa1a1bb82319a051baf0aca50c5dad8a70901f504545c2f4e06aeca5fc6c5c7f227e02b840ee84c5999abb288676ff4ab80765ce0fd9d176e1cf186eceb5a6f0ce09d2fa2000603ce7d16ebf3971b7e13041aba3463e944af34efcb8b63a6982a0ae39ed58504c669c557776dcc35cbc7b7a31456eb50b13cc39424d8a4c0941f6ef5f0faa3e65293fff00261437c0e27319f2b124538c64f5e8f83e1fbba461351cb401c5e770a1c3bce1828f283d08541f452cc19711da17e9781bb273a06e9857458ed435f86e9c23d0c23e0e6a6bb3604a9f8e31570fb7e2fe23adb8e89a0fe1bdd67817f3c210c3bb976dde95cd8f1de8c15cbfe804484e4e671744336488b514c40756208764f9a186dcd721a16b13027b04a23e89af0c148505008a7a650feddbedd9913e1351471e4a553d7fb824c6cfecd05d390570816255275e247535cfef75bba2673ea0ab322425ebccd016295185af36e1642075f29ace00bc87bf408995665fe12cdb9327ca807083ea5675ad7751094b931e9aba279ae9ec774513846c5d116ec946335e07f4da97a4d9f5d44338f854eb6790c4c1f8a8459646053415219b823ab6a37a575aacb522de7109757d60a81491c9cc073665021ecff5c23daa2746cf935ac4598ee57a5324441d75d60b42b6fc39059148865efb2c8a76767798c33b42e92333b26f38eb3fe2ea26710cc0de21884e536832d5c5fdfd42bb8913af43b500fbeccf1c5efda9f3aa47400b9f1136bd533b5df7890944ae766743944a9772f72c74718f5f6e6627c62393cebbd3e9ea1a635062cd9409de8e81fb9f11b48e7edc6b93eb406406ecde2b04e56b11bf691fc026a9758ddf1f19fee0a5124efe6ab7e16f7c7c54446d06ebdcc7c9b33dc555003e77cf03feec11b70bdb56733aa6932b5abe7d3164c084fcd104eae9bd13d35e6b6a68340cc4359fba40c09a44fad1fdd28b93155fdfbfd01e33e2e00c00340cbc261880355ad59f1883020bc1ab17e304916d0217a9701675101b549f42f957e57f87cd52b42fd63cd1315cc5954368e2e481a64eb0b5ebb6adfe608a75108364a6dd0b3190ca2ea8395ed7f81797b7d6bbc26d99c6fa058e0d64c2fac476f576a6f7ef97fc7f9cfbdc54d8bc98e222f5a93b0d8cb9e4ccd31aae8339f6717a1deb8e1676798837a1bc4dd50e2076e90f5ba7374d5de4bfbad817ef886899c8ca34e42e23912a9dd671fc5239b606fe89e5d156d3706c5e8efb06c0074bb6c65f233f6cfc67b1ecd6cd00bb76c68881f63f1d53744bbf1d17ae09215e56264f04441636463f5e8beadde8fad2a8e15a2a71e0a7b1e321cbffdbfd6dbef376edc1a9ca3ddeff2429e891c2272eecf683ca8699b2c3495300ab5067feac3226bfb57bbf6d552a96a5bfc5dc001c566e39a1d528b7f67deb4f4eef8b7740124e8057d3ed9b026ad985408424bda56110787e058a1710125d002a01e1ad8815cc7985bde21ed75c9989e25d681fa44be5c0ba42496fa9bd4b18eea26c9c827723b1dc80e1b329689524fba1dec43ad1a53661b80e4f0b1280457820f4244a06efaae4a32bdf3f786c71b149e6148ab63de656abe3ca81adee13e13d890ed2a6d92b3d9d9fd2c68ddd5217afa6d99d884276547ae089f12ad048bbf7ea4d147e050c3ed7f5ae19db932592e0c9bd7f7228d8aaf3903cac1294c6b3783b82ea06327e4aeb1715a3c67bf502fc67619212bf4805364f1a8b7b3f0933f60fc4fd4ea6070b540549e199f7b03641dc07b6075f9344fb7a9b10de6c2d88d30609149bf147d6f670df119dcdb3284467c9cb4c317b882b6148c54f212d7c7e143b87323424ba156815665c243044dfb64c5c1c5ae0399f563602a700ba8f3c7a900275579cdbd16eca5ea2efecdf81504dce7f997daf8b598bb974d45ac88f364d46142b293671e293f7c5b3a95109ece5b6941bcea3d14f817b6ff9b8baa96813f50f0280a8a809f614f640614b1e71ae4d4953a23ab26558543862581c586b7d9e5ef408ef25273d29d58d3e472d5fb8475864069490c13c85512bc6ddb426c45052049a245df35c39b4be0b0e5ca326ef5238760e601f9ef17d40162e27b0897f1c48aae0c2536972271e29d01f93bfbeccdb558ec83a301f8e874ebc93f93190517fb3efa255baa337ffab17c4870c42bcfea06038156e41400a7063e568fddc127b9cc98c89698778304d89148910042385070e0a22fc297939892c1c8246a1fcd665fd033d3e8b5f485d60af800481b31d50a539a7bdc5afdbb299b40b72235dff6e50f7d594231b59d402235a60bcbe5d8d4fccfbf7be739eabeb513f5b6fc4d0fcffd2c4bb5a58e9ad076ef92c60f11bd04165c0e55aa6fd7dcdfede92155552ce14f3fc308eb6570cbbb899d53faf2f912a3d2cba6930c7e04405771426002c0d086185ea7b826977548d25dbfa672f170bcdec99f157aead013db33fcf90682222895d7425bcdc662a098c3906e2b40f4905d14853634d271ec2c18a9a0d423bcea6c4438740c2af21fe7f0564007929eeca64467c83e54a0dc6833a3d6e8f064d25aae7137b57b3de31958970e9e89477858abb9da90bc4402fe72187d9e9c9dbbeba05342862cae7060bd8889f96a8d70343cee55da958aaa8aa6ef5cd1a4c981df325fab655175a2fcd3c43bad3b4ee84f61fb1e324b9e55d8947ff69fa65c79eb3f654454921c6f48a78951641e6290e2e96d938ab592df2f11948fdb5a8b9031460fffe2b471113ebb286e39331b3e242361f239a14b651b50a924a75de399605313af2d4f6395b9b5eb677498c8c7668ce425e86895be3cfc196cdea43e5e2ae828f3e8ab7a135cebac08229dadaab6fe75173aaad327716c6862d2b2e9efda892aa798e4be7fdb1aab57f67ac2385f80b26e87ccd2de0d6c1272eebe08484968ce301f5d88f06559d4af6916a1fd5178f855a677133a2fb659304c04f923b71c7c6f52b749421562147222b7a85a0e8a3567f837f0dd8dd43bb1d891ee8729937cf2547f922850455313217dd2dc6b4b6ed43a96d6e65b05c6ee1e63c76f1e467093967a8cea9244a6759b37bcdd703e4f84993cd7ee3cf5e14eee385f6d48496d6e108d0725580aaf5846f5987cdb44ecee34ee32ad24cdc177ab3831fd00f07b9553d5a81dcb96faa38b04a6845e2560118cd789efeaf4c2e19b75c388c14a0d3df2d21c6815125923da1109c32561bfed9f33344c9a7c589109d4094d76faa67924646d2ee5aceaf833da886cccdd6d16dcb3234d63445964491602ae4b0a0eef718fa5bdd094154324b28f4a4d0b7bb43ecfaad2a9d9ab14d7f915454fd38d63e2dc79784f4e97e8aaf655d982049d7f9356cdf69f3dd1d0020368edac84c9c32c5fb9304d719f8252948f79d310d88ec6c861d7c7d9396678a3aca20431a39b2ef00e047fc45f6fd3fb5322baa1523be6a85e5c5ff45c78f5d115bcfa6cc6fa3e5f81c95ab07605b36b075e9136a4777709db00546f491edef5fb9ed14c71673efd68bad424c0c3c26db986b6d5742bb683a354b837dcf73bda8c27be43fc7bccf0600308e75dcd299399355ac7857f3614b9bcacb4cb4d181ebdc6af781fe73a73b76a663d9bed2702c5c6d65390d30c4f1a0e825677e12070896f3622ccedb7c5da6166bc22fff39d9c7091af783ee5e34cfe2f0cf8750c71a6d03375251b8e0af55d428517bc6274e3b6add0c05a900c125e15f869ff576cff3ac0a6e5a66a36574fe49d3ea58b9292e7424c203a29922fdc20a9eab46184d93bb8d012b43d9c27a1bdd30c93f21634b97b4071e302a9eb2b1e66633c5b2a549109060c462e63c9daab45c4d0dc3a219f068914adc2109fb320fe9d781f979bbd5a9f3c67dbebbd3f8a51a30eec0ccc62dab31214236f21624a25044823958195f0ce009906c34549eaf8825b3e0210c8411e63f20602ca5d55f21d1f4fe21e43975a7da4d4197f8e07e5a8332a61d1ecaa2a6dabd690d5dbb3c398581770c53f05e51d3d073a0d1e78e5ddbfe2d7df925eb95a5123f9afae6c22d8af72f9757af971f8e1998193a228ff3f1cd54b29acb76e1778ae6395325e765d64585e52c82dcd76ed01868ce778a694ee4d6e82b13ae07051a694f75f5b48542522400355bf7ff22a59dce2c92ee481b404f26449c14284c14bb606dfe3cae5dab01d3114a752a6d3702758b9f9b4141fccebce75e4e59c77cd355b63a0b768293461ffc564a296e5a9176df933cfb8de5cc4e092495e625423f51bf34b0ab17194baec38fac39c9d10d6269b7cd4798ae489fe64a9e1e4e7a83a49d3fad510eb5bb7398db678f1621c8f440d7eebe0ee8d030f7ae6b10d649f0d11ba20813270e425b3919ae574c8c72f8b50e7b19ce310570974e989875085663cfe28848b87cbf7f000a91fcabc8b25f772c56de087985fbabc3867cefbcc5822cbf0e1385105909a94dbf0b0dbbf4f677a54b4705774f37cb935587ab1999ccf844dc016e66add658a47f33c17e859eb9bd82ba883abf8fd2db2a0e6bfa5fa2043b1870d99fdb6c976cb5ffa3f084debecb77a7e324cc7d1b4c5381915b04b6543cc965b6dc4383a145220f4c8df6ad1207b44058fcffe0cfac01f90a661b3469e10e77ce7110922a9a304a96e0635297b4f792edef7e87bd6ef254d8b5cd926693a365d6bca7810eb4fde2a33d58fc5a7bc85c42445c3d06dab264f5764166914966769f4dffa5db4f3b3822f2159bc978b0a6bb2910f4dbb828e8208e3cb431acfc388f3c9e3c5d747a83c3e6ff8b04a13a472a3af81b0f523df4965f26740e79794a24e704442c844d151166c1a8d6a20749ed313ecae6234892c847479959261d4537e68c82afb0bb35f60470e143c8a29340ef4a3671fbf6fc1bf1081d1c765ca3d8e89120a20e5806f03976a5955cd6dc50d5438dfd2e48e3ca93b41da7b2cc577f99d23b578999fa20ad4e121ea6f837591922b612f147196ded91fb43fa6d14fa0f67eb82b76c6f40fae45c8558410e98b95470bc0e937a29ebec6f1d748ab4b451ba17c72f7e1797cd741cbde509ca36e9e50ac8806df36bf6b6d5b6ec9bef05f78bcdfc2c7c8b1db97cffb02d22fa9a3cc17252b704ac87f648831186f9042fa999ab9d0525630280f9d951d29e174e602bfb85c4bfacff48b16a49067092efd5e5a5b32da7d4d89ec60a50c5aad101c103d53030f95817daca22f04c893d25e3defceaf387b155f59ddd47199983e99347989866d1df3403edb88ec5b1bdf600dfcbc0f31d66b3067eb84832f41074061f286f4f21b80fab72fa5a9aaa0f128da545dfd669534eae76777cee0123027760e3a37e94d378fcb4bf398d13ef19b5fb5c5eaffbb6e6d52c4537253959367c3216da20e47c5ced7f5314e80fbc8d9b508be14ea681213b66211b978a7934148fcdbbf991c9b5d74a5d8eed13112878c5e08f0bcbe4866caff238e3a49c1f4f4f20d80e12f8386a2711bd07e261617235db5a850492d87b0e143611d4366e73cc63bd1ee08061e3f56af49f828048807732b3e2aa3569e9f73cc2ccbdd6bd94d7aa74a5bb757db3218ae33dab4f7a08696d699aaed7098b861cb3382f3c89d83c15867ce63568b1cbc4a9b943aa4a1003a0f7272e96864b03c207c70b3362f6600d2dbb01d3dda9442eca8a9cfcfd3a9128e9303fc24755a98f9ab83179d5c3c095d8301b4d5eabc4ee4732fe32d76152d5a153e8981a064eac9e7191f6248f0e7d6e5b8f008ab4bcc30fc8587c7352263d365ee423b1c9a2cc14b948641fef844b041f2a9efeede6563f53fead818baa4ef2acedeab714e46f60a6a76149aa2fa841db0f6756cf7f9e932cd517987b46e9bf528a443b9da31a12e71260f2a97af48a8ffd9f463f0bd5dba7b64c77d2643c7e5636f11e3d1d71ecd30e8e3071e02f2dc286e0969a105f354bb38a6f784265f610bdae1067285f0fd44b68b7e3859671b5f894b9d89eb44b2f2d26e9ca2926adf15d2a7fdcd078a78b451019390bc0761cd3b0167e00fbb2d2c9fb68327cd97dbf9ea15f7a16399b83f15ac3166b6f50b264250492335fc2a7277d6a009e89c7c9cf16e26332d81f0aa7bc7b4088ab638cece6686d8edb8914c1cb63409c284e764a2ed663ee128137e3ee59aea2dc45a9f76c7ab5742d6a99f54f4b005db011ed8f2f907ff40fafa536af92379f5c18b7678f240034327d7d5cac64a20db1a9878d34c9e969afd00eac09f59c09415a38ae427c5277f069dd08d3e9fec559489aaeffed2df93d6b35fa3f010ce8b575a0d9a58f794601f9386f88ecbf14567005f95ed1505295357b3f1e6e61e1ab678ed300daae81ad5366dde22914a051a88b241b9bc59fcf29474fd9f67b90a021eeead39451d69d68db81de05ff31b12c9c7af0713e799db9919df96adc5e8969dea6c1cbb92f1bbdbb9c817dd0ba614c5b89fdfc7929248e95a3a574dcf94a19935ba4e6b762e8ec2d737c1062c539705ce50f95d05a273fd0918e436d7321eb36ffd2473583c190ac042604814aea13853b11bf44028fab297ba111258ddc7c0742d0b1f403b7f85ab63cd5b0107d59e748e856bfd7bf7c810a9c59284c79677701e3db0ad5636546cf1d4d8bad7270f6d67da0f40fd6c101685d23af040d8a4e4a53a446ea4c13a977e32506426154c702e7369896e1baf8d92305ab07c22f35e37a4206c5ec23bb2a58dc7aee6b33126a2d98182dc7452760d3d0c3303330caabaa74aee735f4b63028495b80ecfbe4ab1d2af90cdba9b11cca35f3b8fc08b53c6b2ddc69a0e85f52484b430b66df5daccc39e5ef7f1a6571822b0fba714b77e5698f5373d9b85c66b8de2be5e918d5216fde11fcfe83726a72821d8df131570f490197b6830ca48ce5cb43db10c58092ba0e0401e0dfea2d604ba0d3c9a630d467660d760242da8d1157b38a440f1f6d35cb105103f42fe9638e6426aa1dd1f2e176a71c96f369f1dc46b60e443cb3187325484928148039ba6c17bfceb99a015b274524893c79241a878801b3e28267119ddb9cbd0b88988d0a1953f1f1ee3a77543e3ea65cebeb9ce2f355a5a37248f803d6ad5ec2ccdffde7b8592635ea7eccd2514d72c092ea3723bf3363e563c215c3b374e7300989245fe723f917b95093078fc8073c9597d50817f35247e25cd6cb2341655ba026ebc00737ff557472462acd171da3c79a41301810d4e8e5fa15f15ff074244dda18f0fdcfe7e8925fdb594c79a9a8ce55bd493a4e6daf1ee392623098953fa7be785bc27d7a8d33983a1d42daabc18a8962d01bd0f6e18c7f3f70ec21348e8ba2b8dc3238066d3927011a2f78bfebf6d3c73ef988f154515d199c29a2ebfc712fbaf007756c90c8bf44d8ec117c1d8f5bc1d8ab403d03ffe2cd499126aec18d6f7f80e2deaf33b905854afca1ead6f4ac6e83336b87f17a8d54fc93e19e32aeda448bcd7ca9b87406b553231bfef71896a7ecfb5a61dc5ffdb073a2c1b5f293ed7d525de349267f6175c0f288b611c986ddd6e5f0ee543763a912c5da9950008b396d3b26743f79129283409245040806daf973b50ad33cc949def7e52a8803a4481148ebe3538920f834c7f027eb8a58d770b2699176d1cb1d45b9900494e285e80d8ef24373b9a9bd5e1446eb54369156ae07023365d294c83ee80a88447690bd728e7e018e628268cc221024bab987f54534b3ee3b76e6b05c42c73307e67a452e9211bcf75befa3fe706fdc88ae399a08c82124beb412c5a5ae0636d71a4d3f7500bf18f8176aaa29f2c9373ec0b725c59341bcdf744a4df63ea67d48a73a15b0bf3c98438b73eacd92d2f69dd4ba0516ffb292f63ee46f9ed27d225fff44ca746251ab6cfd815d32c02452609b8ea30f6531bcf74c6667fa70dced82bad5eb44e1b7533bf7bc53fe160fa0ff9cefdd81f12420c3f757c2e32b5381aef6891bd3b584582fbdf32b0678b1aba5e8cfe84480835004f3a26faead97bd00794d9b949c6abde6b7b831507fc5389f0aca1077ddab06db1fd269a53e3561e6b9f736c2ca7871b75ce39ca36ff1ff96ff1917fe9ff80663ca90dad569164a1e5505cec1c31cdda6b2764bd2bfb24a1ac27355d97852bde52b54d4dc0b0cf6250c93f36049714b32d155b159ab4e407675bacc6aec6bc0058a74e42707c7f45efb50ccdea94de044eb493d5734d49530f2c1631c8c59677b2ff2b7f84d54cca16ff204517db6b1e5a310aff6bce4ca35f9c099322cb8e98b6427ced557283692d7af075ad2aabdbdae681a5ed47d0b615be7017f69b28e0141f256468620cd85ec724d5ccabb06a4ef638bb915b4520c3c0bd2b5edac1fbe1ad4a0a098a2471d6345ff7fa0ed2b78c5902eb25b0255fdd8745c5f1b5006d0a663fa0f202ea281dbbfceb1101a65eb5c02418de3d856b6eaec25a481a143a6ac7f0e2c62e758d2c376e0105f1f3e5b70da47eb11b18dedf23219827e39adf5dcefcea26d5ccd421127a9793443b62e1f9008d6ec26c52b4e3841919908193eecdaac2996132c87b4d51307437ee6a73df12d96727c091a80fa2fdadf4467a4d8cc0c69aad7ad2c629a9136c911ab166d1a9fe5b98586485f965a351dc10a462f2965753e4f67b044d0a8f52718c2f47d39c957730640c92656960e764ab4f19f92175324dd6701ccb564a33aeff6e6188a62576686466caf929ea66012447ae7dabcc2c688157e19153f08344d8906793e3d399d54ae18d87fd74bbc77f039f838351497a128c2bf1693097e6e59c469471aa96319a3a947e69275e71f6e78a77e50657dbfc1d3c90e48e48fbf228dbcb8df3cc2b95e5c3d52778041a88749e856c2a4056d9f80c6f036a0a64322fae35f4dc46a821008943df956b3da25cf38406e379c9254778a56163680a2d67ebd42e45b9e2328e52b2454d440d48738e68f90cedf8e7516ddd173b945126688ba5350af8d9b558f9fd0e3963798f870670b7179d40e4a440b7c4ddcd594d8379670308ba7a281428212f5317006c59dea0decdb55ec952654eaf2eace5ac198720b574c4a1f4425040d1ae0f935bc37f4076b0e35e062a6bbd634b064d47be91867018ebe853641a0128b9af05821e19704eb2025c90a7f91e3675e79c3c29be9e0924bfb1537de602bf34dcdb1cf83bbeaf46a8e2cc3f3da20397a8b5ef54ba49b410dee98b15b5d3713d3170abd1d477ee9f08749a16fa4ed6270b950f36ef4c0df19bab6ff47f01fa445ab55807740a0a701e4b2bc91a72cfce689164dfef087e0f41b66a0ef5ebea19ad8a0df74332475f95d510a9f67bc233ca27d34e0e9d5a624d7d612f86457d0d8560502ff9bba1647ab9832523d71eb4e334e97e7e0e2d49f3fd2a44fb102e2372df9a4cf5eae8f88d4bab7ef7f71298cc8dd801b66f484a701c9e8d1bd911a6569ea0930d66a1da79ea9efd864e9b32965f2cebc3550d977c09d22bab073a55851722c10a251801b409bb3a6d329d35e49924adf2d3543d24f70ad9265a41a8f4c635905b1bb6d101d026047aa77f2b9f52f8b586bedf982940e698083fcd0adb5e378106ca03588c3b8b83060167efbd9b8ab44e6caab695311b24973c7b296f653f3cae22d27990f1eaab37de51ec3b287c4682b603254319258debb873ce8d674f83815f57652960998a98035d37fb821a9030cc8c23f69c991b21b2df9d73bdb42f3006f8335db1916715441eae385915893fdfb427f15fe3d0be9970192302abb23e10e762211b59275f78d2eb0d70d086503cc3579fe835c4cb8095c854bc763b34bf40a46572b9f4a28be1b1b4e1112922bfc3805624f8cadc0ba3ef7fd010106f748fd5e4f1df4b5161441f0af588e28295b9652c235d41fd39a0ef878eb8bb2c591e258b80cf205874e8e625bc6092d91a3124732802d077e0bd638c9df7cfb93854008c6d295ddf1b27f6372196cab7f6e3484d2ba4f2b2732c80bfa21fc8a033be19099f25e5393d981275576211384e5acd33389c29de7b7920c2219f9c9e75b66895f3a16a28bb31d3bb4f613f4d26d4481af32389c9d68cfeee9561777d86d494ff924ff2acfc0858a8e273f679f60682a039c7ffb9da39cf8cf46196e2cc64cd8621511f7e5b29ec2a7fc05017112dc78ad106c4ab53a64905e18cd3a36e7757f673ee1e108d8e3ad2c24c43535d2303edeb9f9bdde2be3041002f1c73b982c56f3c9e6a1dd40248c10b76d57d05d6aaf06c2b3f8af94d5beac5349b83fe9e02f511c44efd50ab621aad0172853a1e1272ab58d41e3a3a367f810ad7c69ef656c917bb945056db9eea8a866e3850ea556c26b4f00d425117185b8f598f90986629f8a8c71345e945c7f9fd372462c1746a38866a33f7d43a5d6494a49ce97c56d9adadedc85817a2a87ad64fd9e8924e53d5776b9f6086c9397d547852ea6d9fba508ac2ac89281c0d9aa2a99948b62c0af6858109252c36b3e023df8b6b9f30b658c97668d8ba2923d2e9e18667679f76e9d92ff2eeb711e56e69c1d2702e421eae9f61d27e3fb70506edf221fb10bb05e1ef73f8432a8da0a52f7c953dd89c1c0587fc0a9e73aca65cbdf370e30ac85d86997531a31190506139fc7d8e1c61e343479d3e034487cb91be549098449a60c84f17e3258f742231b0f4d7cbd5b0704ec6c0d5682ab9b4bf8a10c83f1794c53da569fd2b2fee81bf00bfdc035a76f728f043941ad0c215b361a26c10e7134e61c71bc257b69817e07820ae69c5c2e0006c44a3d8b1a931e843d306a2998e38d55ff566f5ae4f499d78caf7519fee93b41c1065934838dc9be743313bad5e3e720f8895d7805747565c55d2a3b0e25c411d6f98b5d053e785000fb90e0bd10dd0ef6d27ef3fd600c2b83f8c91b6e79a9768517d9dcf2c2c896e81aefe29c3b97e4dbd21f1856d1047b41ad7258c92fb846aeb21177b8febffac04c94634212fe3c5793e185bac35f3d35e204a7e2160628082e241de337f2dd662836598eb734474d8a307e82efa42e797b082f419a2db1353d8e624f6d2f9f4b47a4a5cef8e44fcab9deea72be618482a20b2414d04b4afb658a2c0e508d782ff7d24bc74e9bdaab063cc3d41d6beb1f0ec1fd7c85e8d900e08ee9b1bb8c1291901184a89a81a5ca4edd68104a4141affd14b665625424f113a019e83870bafc231d4829c22b00e4b2d85379c1e7736a89432f508b4366be228ef21fbbb09b2e2b2aeac62f21eea0d040421c2760cdf58a5888bf9618129793890f285e9a616670667c8bfa19f25e90a4b33e84277275a964b713aec41cebfce629d93354bd088a280a44974a3f0910a092dc48cb58487224178658d2bd8aac8038bb69e5cd8b0982abe2ab263b5ddbb5db236049e83be109ecba62eb4e5c33f1b81e915d9f4abc0826ea2f404a2d2aa83b9fe081c04adbd5b892dcb1f1e673f311aca38184df8ff963c341ef971b003d346cf17656408cee5f687c2748d142d95e0f90394e853cf98d11148f0a9b1f48f94489ee82c68517a13189eb9e25deb47889fe04fc7a5ba0934601b33800d23aeafbef6e2ddbbefa119b52e656a5f40f308a492a5c1cae1fa2b1eff1f3840544e4f364c869306a7df31690af2d746104968b0f00357c9c4671aa811027096e73526265f785775988e8ef3f8a92eac470ec4f7ce539189422a1a453dce55082295446e04e1f05cc3c2b0f80e973513985cfb2eaa5fa1d945b8dcb886bcad9abe448ffbaa5014f852973f95c31c768d2689f3e74447cde1e9a2c71158f11788898544a8bb03d1084f9695a39598b26b0339e44fae746e07ccf4d1c80bdc5e556437e3af7c293b99eb9e5fb58c405d3bf8eb6b2bfdcd14e2cd3b3e1d9e6f8a089b1495d66d5e0b71300bfc97e6cd227e833165d42fad5f7fce9187fa3540dee57943ee29d6cdcddeb3526548d0378f9ec2a914a0ad766cf0ca914c9ff06a286ea1f67b2f16523457c3ceebf42247f1e9bf860ff814f5d0b9e73fc080854fce25532b2260bca0cab6ee524020c7b77b2ae72837edb0026db652bb01e9938c4d8bf2dd74217eee91add4d3f88dd7c853d77720e6ebdb5adc9bf67b704b3ba7592d120b67730ae8edbf138d5356329f84984518aa6f39ce01d965d355a22017449e0c62bb847743876695b130d656aca9e8905ec6c57ac8503025cc70835dace76df78fc70230b21382a5a14c7f8a9f2abb9c54e8f16e577cd0f9d0936790b2aecb526934d1cae4b277d9fa1c212f93ed0b5b03ab7edbca3af6f54855884457804c677fdd01ba5ba290f16777884acf94a3eb8fc5a8f5fb3c3a8a0825773460cad864e5957ec862056ab1bffb6a9f6f24adc2d63fae1ffbe358fcd3b75d42f5c47e7e1f7f066276a72dcf2edd25f972460045d3be6ba8e3464c1d8287ebeb3085cd0949a40a5a546651411f983fe518846ffb1f7c6b9afd1d3db2631f6e9cb90b744389e686416a4d31d00445876df6d3a7ab2817e911438eaa297eec0a66681cbbd6f6fb3e9705016dea90a0ff17712af695625f8de0b5771e5e3df44a3531d43857c9d11eab00b78a27382ea189cf69fda1d5f8c5426e2e209dcebee20ead49b987dcaa305f8e57946bb05cc4e14a7755daacbe900207a6579ea5bc31296a01f64bb2cd1763b4c1012104e253d0b0815b5887ac86d5b1ecead56f7f724a4acc2075413813d44e78f6aba49b539225a87e1501d78720dd7a88c77b9f1e45c5af3c8ef9401adba205a722f05eb97e6018611301e4163264c53985bdc323ffb52032b63d995ce473172118ab19edd237af5f66e1cf21c0f9d01c025a7ea6778998a8ab93429f609de5c72e72d60835630086e435dcab17ad3e59c9e0e5ae436ec5c0963fc0e2055f6b6567da7919a07a8e5b4145eb4a3033aa7b387e1b8bc49c3c3ec58590ccbdfc2c9a3a0334e19d1dc7a9750216c67014f001d9ca060db0eceaefb0a90917809da09d607a24b43f8df649968cb6b982edc1df5f31a6acad7b9ed5529af3501c874591cbcac82650519202fb58ec3f283f8086ab12fea74fd00d76bf593ade0f9290ccc6984f895e6fad9b2121c21403db903f03ef067fd281ca8920d09873969d0b0e38851771a22b579bc0ff31dd078601c5dad12f8d497555bb9e6541199aee4424e60d74706cb5261db9fb37939b3d207c0a702c323713951f89601d238065c822fcb0a171d87463ac34275722309e88f14c0e5d51e1e99653e4fd7567484c533e79911b81a8193a9f6ef4fcccddd6cae093c0ab625328bb0621083c5d6b802a92c56558cf160dcef8d88e45627e5b643cd8dae3461a164c44806a3d3c0d343c5aa463489ff6e81b6e5694d90bf3ec1a718b7cbeb3417fa1efa9f6dc93f04a9edc73ab1294e7a9343c97ff0a751fe962427eeb872718f50a4f26f2cd3638ae9af174c5adadb8e457e30c83412c66e34c181750f33e94f7797aba1acf02c5b038bccbdf73f922cd9cdbf0d6c9afd1a7136a1e549369cbe87a6fe2ff5e7ae8555b3c85d7707530a9114b4154f63421d9aac438fdf57d0328acfed823734124755a8808329820f18e7ada6cb568b8384631bc2e78270c8f324ee934da9e208b32db17cc839bafc83d83ebb78044b25e280b04bb994ac940a4dcdeafd5f9bb1f20d55531190c54020126e9adcecf0aa49cd5cf866a4795d167f2e9928d27f5e6d967a4528a377135e7cd948b08dfa91f510bf1be10d11bd4ab38c171c4dffc440a59b7d30538c9c8e6f105a708ed3615bd61c29e49d405b847ff62ae5bba7b3baab77f37b51ad64be760840311acb84ac1485a4ce7689cc05c48cca5023b5a3bd1bca6e53a89a47f5570a05edba29c0e4a0bdecca2f4510b1052c946a791120085be4887f5a7bc53792905c4157921d0f6fda734c1753c611272c98df4fea64f4543938dd29cf7d823d7e96bb6e96ad32e558898363d428f3e8c351798136d5408e8e8993b6f5fbdcb6a4deda9c00c8e31926e9d645004ce84f6f06c7d10d87ef424ebdaa4b8efd303cdd7b558e21a2746d0c850dabe33cd32530df8cea6d83594cb8f61e422a9fdbac733490ac5d926ba15afb919e3284c6a46bf99da5f6fbb9d893d94fdc060d8b0830bea5e4818f6c9d53ec1331cb1364b609bda6ff66577cbf907888779cb82cb35d99ff4aeb09203e0b4ecfc3ba4170d6c02bb05710931bd0f7a01b16c67e971f74f8cbca6b332c52f45d783103ede4f53ae3fde5a0c0d689f7079e355e056f6ffa105189a722535718d30255c8cc2c24cd780e1b912c9a6ec10859403b936a09ac7d0aca536bb7c27207d13d46adc3d80d4de2a0c0da149fedc9070295a8704c9eee69d044f12b0bfb62d9873be31ac2ba42a1cc213192882c0e4f52d19df280285198167a9cb04b58bb9f53b91aa4d5926a7c66daa496898fee55a2ef4082ce1a751a6e8819b9722984c6017acd7002d9a071aca371da50a355316c0ed85c0abe901ffc5cd87a42c738733c0d4da7086f44e985f836d69641f85b58459d9f97cdd18a860ed1042fd0799b628385a25519c0395ce4ee934b149854d4d1f7243da2f9c262e79c0561d8001199ad8b4821e3596b186de13b67c88afbd64d80d1f96e3bc685f97c022a1c9cff38b28d89ee285eb3c3b623f14cdb7dd7da055b4a54967d7f8d80f14069b431be365dde9dc9628478abcd9bca527ad5b3d560459f784c3a9b78a4198727d4f7db239089d9350c0c727f9276abebf893e67482ad07cee2c59c4db78401e2832be46bf4a1ea688c9812939ee2249c137746eff226fae4c021d96c245ab44f2a61f35e9b84a2c9254331260f77fc3ca8596dc8baaf0ccb0e9d720475f4314e624f4e36e2c51e2191c4632c2c360e28b1e8e89dd6e7244ab07586cfc5edbd8f00e8cd9fe20146227bcbb9ca6ee0a102d00581570ebeafbfd9e83ea0b6a8b69917457572703107a91853c7533d53de85a17edaf6d3124ae587aac8b493e6a637c955bc207bec13c04dbc336c998474cd8fa56c148051c5081ebba69f3b50073c7ea6021eb87e4c8aecef483d36c4d1bd6f4284049724dbdba970dd2f73d88cf2f0701f96d87d5274dec22a33f1f856ae124d54dd6b631a73d6de9fdd5b31c55a83ac4d722355d7a2166a86e9f6a832e8c8379ce2ce7e1e1552cce44a6f26a4342c87e106f34b46b997387b0d7e4450f45e1212c07bdfabe6cbd522973bb3b98440147a6d68db172f35e6baeb22d8cf88cc7b4cbcfcf3e36e2f67c8e9fd705d3267d2877b9a472db1634cdd355e7caf9665a35f84436c4d09f92608bcf0259c3a6737f3af8355528cbda8f7d5dda374931121bd43bb74bfee17fdb358282e5c57f36acb9c11e0243eecf0c64810fca2a9961bed735f16b76b2515102d92e17342c310e9b8945f56e43bbf89f62985cb6ea0bd16404f3a3d60f9f48b2a5259c909b45e571d5251bf5a43aedb506c3fe98b92104b080dd60e3d108d7ff867511c850516638d9374df79f5fa32e4e542edd86d5306d540cb7b00b80d688d631341218a34d25535ff1528b6628e145d941d18adaf0de4931490cc0f546e33e7171733907ea6e09400bea6b38807d782620ee0617219e51eaaea920ca48fd6c7278682ec13660d8bf24ee211db2ee16852954363670b05a9be3f535efac54cb7cae0c703f960cbe7634806d0341be05a07e0544fbdf2a8285292310dc81ae293c3b91b54a95ccbd6509ff02a91bc4574648f234b2698677cda49704a6b146966a6b9d3a88cf9401566d1fb97355e44085aaa006b9a000c1f8f1d309dd991ab791f7fef9d3f3dcfeb15facf75436520ed17f140fbfeac875c1588143b1b3b7ed14f40fa8630b839b8d6b8cb6a1ea42b5cfbb652ff0fdd6098f740d5010312860845c6439ffeb539f4002120b6d300e762c1da44de6d0c6452540bc441570c617ecd105b05b2f18c736655c6f98abd4380350af39d5804f35bc2d0603bbfe4513a987da6fb3e222bee1145e28e74142b963a0cf2db65adb55248ca3684414f8bb1392f3a6e8addc5091b8a86cbdb1527267e8877f8978053e13acfda851d1269983c83b984a14b45566f7c6da059a2a0878fc57da66b32c7698e3df13ce5f2fb415932388364d4f4b80318814b0509ef1222476e1517654c968457c684e45bb4a9e35e19673ae6552dfee48ccab166abcc8e86f64d0dad802c284a5a6290d30f7b072d13be5a535ed8a0a912e88af8903a6056423100346843a2434e451237128f00a1e218c622175fe50f7221167ccbc9f292895c22a5946697a172dfe3e44398e7eda748b1193510db167d9fb0b5ad5aac156987be7c3bcf0143c559b5695b9edca1c379c63e82590f25bd7a0e55261c595b3e9101d8caa009d04dc0cabf164f321c7aa3f8a949ca3c9ce43873830ec413f6ddf2c82fec4d18008162bff73f0895fe12659eff7d781b51fbf3767c3b62d360a7381f1f353112829be05102032f60108c0e2f424530d5d45c891c8f3a0d68108c9cd8b23e40a6b44737c421a3fa923eccaab46bf11a2d723d6b1e53324d0c1562ca53b9189315da9a062e93287cd23440002d79b57e89e90d15c9a8e712e4132621f05a2a75e4cec39bcdd810a56a92b1c129f4597052521b9ab58a3d85a2993e223be24b066067aef443509b4befa2803e4d60b57c60cf4f4b6ba03bea7650269fd53746170ae185abb46a08a08a9b1e08bd86c6d5f4f81f12d83fd9cca34e3732b8835a4b3a0bc2b8904dc66ab3e986e79071d36414b634e70f6f4da4270db3d2957dc69149d748a2d47767e79157f927a50b1192990267776162506f38dea991bb6eb59696470726686c203f10923a111b490ae63b7dcaa132d6fd744681445091649aee8796f47642c10b8756f7b2ff79aa7f7c8667345aa4309338c8fd9e8b446f49d99c6e65878fcefd891d3a2ce97d209735cea3d07d42ef5490f7e48e89b7db04f8d48421e0c95987766b3e66d51f1575b891e296549c2d462bdf8303c8e6de5237cd53680f34df0b8431a2feb1b7618a1694df16fabde2646352798c38c122bc0ec89164fe167777da091c2ef2cc12229523bab2c200a6229d4f2672683c60c9ae224b7750160c5d3ad96ff95b16db207b83504de53d68e941373fa95a968486ec5f7f3206588b8fb14fd1c9ef76e27e1b5bd3a9315ebeea7522af33e69d1d7bbcbaaeb70f367f71719acfb4ccf13592d932fc4cf063924930d19e452e3279f0b3b6dded0d3c083cfc61693498e66b49d3d3307498005e2814f0c15a722c532dd21cede9df563b32ce9118f367f4535137dfee923c0340b291c58cc3857f7abcfc82fe144b2f1011f0c4446cd3e7caf7fe0e4494cc7c2d6138fa8fcf1a9245d5d07b16977680c8092a898d3acaca436f36dfe2bc40c05a5112047138a5e35c65a757284bcc06ec6440540fdd88e5e3103d7b05b815b45977a6c5f638c953ec65528caad8055cdddfa9e2853ced4c22c714e7ef30cce84af3816e85ee599a5490db1eaed13dd9c67263cf3bb3b76d2bddb9ba6ddfa74ae4b570d2a73149a006627c8b29f9f8495489ebe8249c21f8342b6389b6eaa9972ccf957f00863ec16b9a702c316629477f2b79ec7e3adc1fdc38b50af2de255b0016f1cd8b730f04322ef0f0794f1e603c73bb34bc2fc7c2c541abf842f93945f9353ccbc293a15c05789073b3a485fe4ba822841b87a7f5b23692498844c7bbd506bb3d423083596b12d1ab791e0d85c143d58e790a2feca7b37fb0b8effc708123ee3430a4d86a28e964f978978bb20add7dd2698d89521c8fb6c53f0e677b169e157d42843205b78903466c4ae5210fff8f593923426396a6bca455078fd4f47117f3068c18ff213bb31545965d6787e2091429c40ffdee3506ee8ca45d2fbf0859bfe40efb48a1c53da73d45e7a992893bc50d92107154b3df1f00a5be778a37e30e6baf2b829e726130031f4b3c6bf2e1c8ddd8f1959b53a5124bebaa31b849c9d52f474df7090569b10ec9ccd0543a229c12e0c135c3f5296bfd2ec8f8e1c26d704918ec95a4ffa50229355f0374ae6b3b52d5d64905820d8332c5756b0c3b9a6275a431b773b39c5e08ce09073859b2bcc9b2e3ab7906958e81fe5693926d65c68a8f4e8704ad287e4fb560979eb3a9853354d77c2a6c9ee1907e8bbbaaae80a8b51384865dbea4d226a36e31f5a7d7ee7ede41c6d5259226d3ecaab28a6446e47995993298acf6dbe46f1f112d250e99729d7b3c1d871e450b2638970a2cc9fb22c3979182acc302a6048ad59ac82289c38a1a968d0c41dba81577ade9f9975e496b19b8e1a2fba53dd790c6e1657a805ef4843d141b7f5654d432263eac998e39ec11fcb0ad7c6c39ae44b83289bcfea088e244c743c19efd33d4d469ae4aa9d02c2601b3ac2592b7c9e5734b3f66c779277ca2cbf9663c1a51b05aac9fce363eb6b9813065d2f300d03ceea29dbab12c432e862ed1fbb7c2ce968f1894d9aa878c976fde7d8869f16963279ab80aa092275789d6f1d56f0e197b306615d997ea1c31481ca58bd66202c6c6e670d3f290d87e0e4dc72b95f05f2156ea5379cd1f0e2f737f4f285055e5796f270899e04943513482ef5dd86177292b970db57a3b203b3e840fbc3fe0d99e4a50f1076742ec2225260b9fe2528c7d9ab2936f78be1c0aaaf69a28c1e901835333db21c47bd7c66cc0a53e7e89bab3e7204c1190e1f1a570e9495024b710555ff1865928bcffc181fd53702e0f8570736eb0d24ef4fd49c7f60bc7bff9e98469d14057293e813313e643c5edcb79eaeaa577f468037a13aadfc09fb90038556b376cea7106e132b574fbae8f42446801c8408ca2c12dad3b735ac6662983104a058bcf6ecad3b98351551f35febb122d01f9211d94547b8385440a09b47f6c7a107c5cd917725fb6c74dbf9d0fceefcdd849e6124f7bff4688dcb9505507743aea2856756cef30ba06e215db65d97ff0079e23a7d6a5d9537f49114fac1a8c6b1f6b7058d3edf1869ba50baa051f554f3404aae8d858dc3ef889163263a2e1227f5857a84463466b71b9b57acfa4228b8be93e97af7835c2a6c250e7c473f19c127751545a13f719353867592f3cdc27906330daea80e15b55e46ae7c571726b5c40b7ec4fb81e5bee0739493bf96e15861bb4782834478710e25deffe47bd68ec67ecd5875ed341de147394635a64ea32bce8168685cbd2c2e5e34adcb26f2b034bacc988f23275e25cd87964e01f291a2de93a07cb76a93500c7f865f2c4d6e702b48a42b56427e98f2fe605e23e37f2916428ad338d2a609c29f2cd56d306c1bdebac9ac2ac3446758823291b825288e1f5048a901b3d02c11f5c47724d0e87e57f1358d13e5794499f438bbd5358946bc422807797894fd95db9ec24709f53bb8accc716376d8fd47c305abd9a48120f1aaa8b1cec7b1fefd77b3597823e8f31597b0bf79ed7b32523c1becc27bf48853f261f7bcf4052d3cd2396a845408bd4eff9c2e11cc01797a593778a14fcb889082bfdb1bb46400eda54ce1e63a82a050b82a5c78ea4eae421884eaf7e44d7232b881504e7e6bc33d9e2f2e5a6745223b57dd6663b7fda6f10395911c51882d78f107b081ec904c46a4c374b6f7c3ab976827c885be70f40727b2d51a7d4293ccbe371d144a23ed81a0b37aaab468f3980d665ef62d46bb33a0e178456ffb3652c3528b250bb0f3d31fa7f0246e62aa193fbe3cb445b9a89c8e3b57c3f6605139cad7c1b4efb890a123a1563dcd217b4ed0fd8d99a1197a3fb46ae40a95d39785755353a4c373d99cd8c8d7cfe5ebdb26182e2667d6326e146fe5ef8e4de680838482211d86cceeab6ee18d21e2222565b54580a75706b2a9e1ec7b4d6ff3b4561dbb7683265fe036b6a00d72c1f3e90cae763bfd48a78798c287a5a43deb08642ddeb34a52af31d67c9575561dc21c9e59f9391c061d7f1c167534c48fe1d386104c9430ac68fb1b2dd024e440001eb67ff2fa972620841668d9432650f9349a0be5307353192dac2c72aa3ddb959c2de05c6c9dc5233a3b33f72ab395bec071def13edfd6e13ad25c9dae43c6bc8805e8bcb15adbf66a9ac755cbdb9ccc219584ed347f5b18b7afaeca73c8becae44a5d8362406a2d57c3a86a3cf9479f685c67a900774b232d508c670dc8990e0d4722204bd2cf89a7ad40ad4242190059f6f8a0280b7e443764d3de289ac4c13a23b4420400e7961b92148c1ebf608c754bc4aa894350a1593b89cfbd61941a36bb62bd86b2d7f42edbc9040d049511e67ad4cac4e6bee44f0416d217fee11038e8ce74aa56b175023ac3154c5bb3e48b71860d801f1b0d616cec227086867a05271102c73a85d53087232ff6d3ca5d8149b4001fcfd2c5252db0f6a7ab4e551e6bbabbfaff45343459ffd748f07158455d4f786a9521a8921cef4ec17f74c0f90614e581aa53f321bd89ad5bc80299cb812168e701b5644bd908fefe552195d94f40d9b2fed593c1391b4b69297f90dfc098149419dee131ae5b4ab6bd6783301f2888147ff6cc72717170f152e361391c9a7a1d0b967462d02a12203ac94e24ae4fbe895dde2e66691a0bdf34d9822c2cc9f5b3b4c3d664d1e3480d3121119c1d6875edc87a97ac4f0de513f2f47e9efb0f920e7512f5bd1808845a26fc3951431ade7fc196c5747fe80d3895b615d3bd937674f4ed81c57443d9598bcdd90c5e115c28462f3474f367a9e9b08cb2a71cd610efb92ca3b09074fa1318228ef3450db1787cb767b4b85108a35f8496dd7e499d741d3bd14d1c3e81a69644db34ea0db6bca0a43de77f19e59925ed8785b718315bbb3b44ac8c15beb5f37627aa19c35561da737a3493849becfb014c9520163d7aab123ebd90c51fe677ce1bad0db24889a00c0a11de209da2618ac5f012b53740b415b2bcc285d1dbff82ee4c79e0042e23fc72ac55201472d2230215520909297f0a46608d0c1db35ff7b9d9cb3535b895688eaadf3bd095d677c832d1b001890514056e78a15f14b57496434d71c8d422360d0a58f8f635a51398f7cc21fa12f72c124d7dda0123893ba6b24cf8228ece31db59dcaa1a2f80603112fa87c4e88c0ebac7e3973d7525da352a0e638d240bea1d2b55d86da781ee191d504c5ba71f64daa2293d9dce01f31a2a549636f0e57a3832af942b9d32d365ce409b570d92f1a0d2bc64ad81f35b58ad3c46d89d4d5e3b6ec98552c9f884a2bdcdd11c45ef6e749f801606e9f4d0f907531c92b2a2792593f52504bd8f558ea9dc6cb69fcd81258822022a003f20c903b6994b73a5a912b34382564470d3fc0a52c39ff6dffc8f64e980cb95c0cd8d3601a53dae39c4d2f830e91870af12de1936ba08e0b044e0c541dbdb0de38e3d5eab3d4e4708f16e1615600f8ff2872b54bfd8254aa64cdc6b0cd81ffc7f672321a016f78b9d5352f1947b81ad0709e84de8e032fba9803fa5502ac7e3bd008aa049e4e8090da2e24fe614805b86ddfd247c605df62ee9bbd79c48759d0540be1a2397bf5ff3646cbc00afb4b7e4ecda5915ae238f0d2dbfa5469af7f937670b92f47aa283028c6d09bf0d922d931bb3064cff6f1a961e7316f4801a39e8c486effa47840c43298b28d8992de978de66a956de265f507ef9bd17148b7acd213dcaa507272c3221187afe5a13e1351084c6dff9bac1b1cda079a6b629016c88169028301fc25f6e63e4777eb6e9b7fe8cf4c952fdcde71d5a3f6eb99197435c89ff0c2ee7cdd0303e73ae170385af32168e01f1bf677ab19e452194a45e42c9f8de434b6e193197b77cada942e834d57e0aa107c8500ecbc1fb6315d62524f6d1f899bda4792626d6360944867dc7958f846a680534a62c8ddf4ee1ffabc2f293f79cdaf0285e30df0328ec305b4ce5f6e87e144f0119dae7dbb26bc34f2541ab705e1128af6e96e009c91779df64b9e582ce0fee4f0b59201dc78a57cb57960434be13d3aad98d3ba467b2e58f56423d4c9fc59099e712d466189aa722d9aacd24571447f5c0a2d8653446fbb2f6602b982f23024ffe30e0d6fda7fccad7e8d106971410222dc1c3382d77154c908b60cff38e0a29a4b6bdcf091987c0046e9a624eadeaca4679147cf62e32d0c8d69f14a15d535b7873a83cd50604d32677ff88b5e7f00db6b4b5a22a8e642be650ebaab9a34511c0ef0eee09705c8fb3b76fe690f7cbf5ad7906fbd3f83d673ca7e6977c5c668a357c57b7a6c11a44f66b75b8364adfa39b6ac14704057757c6a4d6d6b995a21750927570cc3660885f5393c59ed9f70b0a054b363c24c2102ec7a524efc76a7f5d8b20817a4602d26f02d441d4b9844e33ad7a2984b53182c5f4d6be0ea66934f3beae3439851a1ebba933c0f940b3e699a4eb6cf1982c43422fff08ec3a6ca8e2c22dcdd34445acd04bb37ee9990eff30c4ce81a6b7e6461c43fcf3b1ed0921a52b9a422cbdcdfc8ea10980e6bc5b8cba137e8c74b3f85b8306741c1fdf88f7d59d4f62f40248c27e344f0874f0225cccf24feafbfaf6c75e63a4f77488956968c9ff3232ffe279bcb7b1dd49baa604e8c4f55abb1a6b188d3b4a08d751258bb2fa939d597f4ca9b7c2f03da145b9e36e76e9cdcef4f909bdd6d2e77c0ac100365b9f98b813f6d6332eaa462763b20432f24e0aad86368c3ad9ec391a7034681ad5be06d60213dc258e5390502b3b9b8a8ac1ea5db80cbacd0dbc1ee1304bb12e441a34f71f215059a85b9b5b8182303bf3215e1cab34ec13a7644cc644f56a6f533d9db6f5d17c1a6cb5df558b814d5b549762d9fea3a092cb57fc5708fdd1d203191082a573db9063bf83be6d53eb1344315d0bad4fb720645aaf1158bd1ad84219ec335ab904f23a78010e31c9e75e6b50bb9d2222693acb0c6537a0411771ab23d62d0f9e42fcec097d78d7cb77e691c144763a609172f8612b90497b8c91bd8463b1f10b31a0d9d2f7eade08ccda2869e7f5ed170f469af9097fb6b7eff98da520b887da30ddb1b9ff6598235f2c33d1d8c44e376fd2e1e286ba9c019cdb5d2575140230365fc7a23657d9411a55d2841b59ae4dbdb822f563fbe497b7aa5aa168735e8d82cdeeef0dd526aec1e3a092e802c833d572f60369ef48350ef9bbb25d0cc6b306757910bde565b01a7ef7718cbc9e47ec47eaea0137be4f9d63295dbd680cce8691d4b617dbf4546d8dc5c85e6918a63a326f71e53a9ee26973795f10b7f9c40540c53cb23d515a23469d70ad73cc89eb3e28c5c206a0198cd53623ded3b8317a645a8afb8b83857f97a05a80084b70fde1dbec188e69cfb9c1a16cca8d757d530ac0765f151bde6736f4ecf5f80efd9d80c5e1bb961ce09439faef39ecb810cdd3bbcf9883e430088c9b265cdf08718926877f5f91e4e3f7419e145a7e55b0902490d4c6a35d8755689c9a6a4c07740379b049e4b3705a642ed2a7d5a7e06856c558e8e61f48f04a38c306b35962c4b026e6717e8fd0b9e4dc1601cb9cfd254e8b8b695fd55abe4290fadb208e122aa623d9bcd88b8928338493fc59419e16bd3713221ac843bdd06a4c174f55374dc4cc31cf4a15ab1abb73a93f02d89f9088d4177b7b47070b249464594053031620794f55ec33c74b41becb9679f840d8004e35d0a2e0fb31ba36cbc4c414c8ab30a3de79e06de3fcca9a63c03bb1b34b787e77f0bf090a09213c89d029e7953ae7cb6955d82f03890efa8041c02eeee27ea853c2f7c5d5a660082d0e176855d72b7a9de125143e284ec5807d001ab43fb932b8b98b58deca82609403b3d3f28e93f3d0674a5174b1a2547659d8a21436e515dbd49278c66a6d0f3dd5ad63bc674a6a23144e0b3c25a8a5263e2c2e031ef7bafd361fda4966260a8b598904d5b674b6cad8b5affbf4a9d276039184053873d3ddacbf0e71ad48106cdae6d012254ac4cb30c2c506e38afa2ae15a45da62717db29ae466653247aecd85bc9eebfc05ded8895285b7d9a8927516ad8757acfbfe6d3b01556c969bc0e06b1bf797455d3a988c487b702e072c82127af2cdbefb214ec6919e2379b813e679ee4d68c98983168c870193c31dc82a036f66d744cc8b0c11fdee3b666cde36340f3025a77e23ae831d83377a20a828d03fe3120d647b631a3b368ac5f8bad8749bd358adb41ac7ce0fa9e9ae2edf27e1744772e7a9766233ab38efe62194e05c51915a4d3f1df2c20b3d1811cea8d45059c0a6127c1d6910e9f64976926a1949e71022c32b53825ae0ada1713ee315e0e04b04956b2659e57bddc10720e6969beb2a4a6e83d954d3a00f651ac904c3b8b6daf712d4802559e3cf11d171a4a17bf192d687469396b85a94f06dc8057aac2b3fe1b19b9b974bac5bcd4581b420c70f3993efd22ffd9f626f23d1c49e96c326eb6f04ea8d81231eae0ceb92e4eb0dc64a31b9308673a19a3d74c1437972e31dc277e6766d02b333dd2c7b627deaf3234baf1cf40a21b2e072c6e8be85f7563ba206ea5588fb9bcac9f811f7a00e4c8b02ad431e849fca63082fccf4679a433f64c6cd461bb65b155c10ac481eee7e3151c7dbd79f933d19080b208b15039d130bf18705524847a8e509e8b2be3d3adae8e2b49af653881bf525dd5e5fa25947c1a08bf8edeaaa62915678f9db4606184bc37f30baa391fb909be1e3620328a7cb86279f45d09898b7246ca8fa63ef4ccfc251c13ff6eae6e7906573571e2de9773abf3e4b6734371742ad017ef19a3dc3626df106a8f452ff077930d7f779e7a7b76e68a88d7db5b7ed208a16c8c49f0a2efa6ae88ae6160207261a63b7e446d863f8df1ff8b72ae1433bae750d5c86b6e1808a42586fb8188e3f2af7a14bb45ac0db568516b553f9d441a179b132a8ecf49457eae009dd06169c8e698a595253d39a2df1d33e97817743b98d4662bd04d40e37e005c04bea8e14f757cbe56c17341fa2e2080f9acc6e9fe7e105678ee186c36cf06774e4308f5849917ba00397e6fb32c33f8568cda4ebe3960f0c6bf76e5f298859d0bd70f590dfe2fd46430c2bc691e28df192d41b59c5c8a2327a418738d62f3583409d860c30eebdb7771073ba2c452b580f67688ddccc396d423dba54b4aa60bbdac95490f14d2dd9fe0e89ccc94d0cecf186bc437f32ae8718378f62a22601c6d1dcd2282c843de86a029df2ddbf00a77190c39823318a4f34da30eaf5adfadd8989ab4b5e82569aeb274f95b74e41bd50b384e4671154c63a9c478f6e37f2b3c113fa02d789b46ea1d672f8295f0fc5edf3d381cf1833dea89dbeacac0261e1b5eac66c83b5d6a9c2cebed4e804d7712a40463dc62764abd42b30c59c3f37cf6e4170cf50b4ad34682ce82ffac367022d78a9c509334a374f8682315509faa196744e2dad7ca3ba6e7e4df89b8fbe2e656f5b351ffceb0fba1e0a8e7d0c898668156d770db318952d0e7ad7365bfa53cd843825d98ee76308d3133da2aea9f5aff3cb34db1ddc8becb9fd477b2b88d6ac2954fe15fe1b1d30562e2474b83b12dc244293574eccbfe8112daee9429d5d00dfdb781458e9abc5424aa9ae2ef04fbd9e8911f353eb4c584f63fcfd9d3be0e085eb2e572f1b7e97d8af422cb5c7ad76105c9f21e92ab52b4b2f4766185d92e4a4c15851aaf8cc4b60c48af9cd9b5106c966a729218bda5944a3c64094e09563a993dbd80857c1fa1260036194417fd2c8921b4e88d87dcbc2b0188fab9df040b5c9b5775586aa6e7f0168befd45f137c466c2ebcde3d8234746cd50d45d3fe98c8b027504afdc97a4f4ae70010d69768b5e4fae835c2a5343b1fa4d6ae036a08ea4a427a9b27ae7404f2c535cd634184337a600045a1b24f79b6950c3c4c5ea367763da1761b76887fbf134df072dd92d1130917b82f7e34ff35f0df4a512f1913ad99b22724986d835f708b10f63b24cbff10f57282f2a850891e802b7e9f99c2aaafc603828d21f1df89c3b2dafbc7ae6579963fd35b38e88350c970d61b8429192a033e00a96e70bf9be6b896c23fdfefbd0c0856ade13c83d7c906723b01bd4f75f7a3af12c09821dd2375b8ec89a5cc631227cf575e75415bc263d93209c275a2c51cecd3f86af2a30834008c129c78101ef7e5dcd0534e2b37feef15c6d73c58e13f42473c1d4c5267e0a0887e1e0cf9683538a3d191aa17d183d54043474dbbb266cf6b99f2dc020cb8cf9aa149d94b4296a8c2ea7dc75037ca7bf5d8423cb2b3d6b2bc3a73c07311a7dfdc2eedd660c396e18e473b8aa0ae30292fb072a1eca9e85c2b1efb9c9029ae690f0a3c4812f9843a8d634526ded21ed7ec16c64994e4ce41e1bc0af3e29b317db53ba8ae8f160cc1bc928ae2f9c8f78882b11d61dcd886da152ffed008e38e806a6cc91c08d40baa861265e8626fadf263ea6676d935995f39ebdc9e18e65bda8bbe92b37c09b7316b7db877f2afcda65003cc101494a83325d48eebf141eba8f90fd88691939ab6165790fe594039d1b5fc875431d522eebf0359f757d631cfdf27a28eb13c42eef02e5d05fd2f325575c002f1918a0d563301cbe7863da6f04387fab643b98b447b7defa14a4ab240d88809ca7ab264cfd29ecf80e2178f27925b90c74a41b9690ffa8364073df6dd96d15f9905fb57e32e73326f83cf67a095dfa512cdc1e44d2c8401354a933c07d86b7206f5c09a18714679992131e2b53605b9ee64ab5318fea46d2682dfd26d71009dc9524eacfa2eb1b51d338d3e4c3fcb62787ca9045069193004d1d800865d23e610609c555d6070063ef8e2ce071e32f2673bc27b870c18597f9033960f3211b6ac24e42b34d88a7d397457960cfa980f33c988b214110784b128461c3d5b80eb3cec5846df8af19a480311b1739bb23751377f84a39141a5a5793453c6f6cbe8d18b4ba36c90db0e58dc75eca78dfea8ce9e0a6de72da202108ab05a7c2a6a52c12239f765ff7f04c0d62e4072eee1cd92e2bdfad308ce6c17e8fd3c445fb412ddc8e36266b6e0892fddb1c7719bd638d8996819bc4cad3321406047436627b23acf729165992ef5801945e59e65580285ad77b207e4fa8e900109a7848fa412e26e8e8a8de0a94c313ab22f6bf74890d60a1d94586328d496b7d481c981b46f5fea30af501d2294540f4283cd15c185db6ac251035b296ae0d99ac6e01ef64e3f4700dde778860d360a4e6519f6073d4e5ab2c793d9871b8138024be739ad2f39ade4c770f93728db3086577b24c1f08876b1d20672e27076904f312757399add997d1960dea548eef375445f2a7ef42e83bd88230ca80d236afb13e59c3a5e5a9d11294e18ad2335deb09b6b5c96ed75af8077dc402c5c36333e4eb22072065eba676f8243fa13ac9fa252dfe1b0996f7086defa25d635209ad7c1f4674cd56a326592f2c523596930b8bc58d7068a348d0f15055b70a30d63041b79d1eca258c39f925678b833a9c223991f28d5ac33b7c3823ae7ba80cff6cf510726437d16915bb0ae264fccdc1f7031cf245abff3b9d8a588e35f36f323e99d280971bf00af7ceb88fdcfc38e0b2d38445d94cbdb4e10d651167d7c7efb46ca13eeedcba30a3690da9e3af3f3c7801f60c64d67d5fd147b32f153fb931eb618f4d0ca5d6f28e8a01b193b363255fea024768c76ffeeec3d2f0054505def649edf54d476b1a729619ed81f3d220030f62b23bb9bbd2f137163e8bcef59a5e5b34e9de39c598b5c6ebdc17fc162b85d466bad3619724e6cb897a1c064dfa47c06b69beb9214210577729a29bfdcf70eed6d866ce73d18259dc883ea80cc27deba8bb479d4a9475c1418bf8969da429781b2b9b0981fa1231fdf2975321709f96cb968c845ef595bfcbe8f94b32aafcdeee25c598eb225f28dd2a85204f820fd7295819139e92aba285893c8a6e5cb3e63a0a8649ce442cbde5cd381d0660d0af9e7438097f9b11fde67bab58d20ee2cb3b08700ad28b0f15f2d82c945a740ac917928d23dd9e5c59ec293c64a576604d1b5f25edb3eb2217c5053a92abb4935ea359c867d29b31a5aeb84bc4924308a1f150f374bbfe4406581c9f3ffcfa943dbed6959d29751489eb0db67376099718e66c05b2dbbd1261c96ae69974873baef6774274d1cff257137c51291deff58c0cfa466b96ffb5d49e5f0c24a7e3a6aaa95e07d448449ecdb901f2f55c202af4c6b8bb64657ff9b59f14a1621eeaded7cdc3feee6570f9148ffcbac0a223294e786e641387c36db096a80238f3c279c693cf8d36b8e222a8fc8a8a4a429afe58881f76eae7ed92d7128bc9fce66a07fafd7068f05766fb33cc663c0f58d87292544c69e7d2fca029e507288a5b7d3493a6b79f7af9c0a0ffae4762d62f95be0e0c0b21831472b9706ec556cc72a11882b6b30990099882ad65822749ff97d3b291bd477e4aef91ad1f546fd5bb83779bc7e6ecd8bbe7352c8ee0fe15ea73ba3e682ff26b758f0d3e89b12c5e0351ec152192273fb3a2d91ed153c1191e84562bd4130830bb9d3e31694690405fdf4e70d53b62a650b3155c217d9cf947726c84fda26e135e7e0c7883f92aaa5e687246d5a0fc8c578c8bc281e9d4c4577840fe149eaf8c7d94539065a27442efe216951ddd393cfd7fa45d2132d212eb8f92113e3e992c66e9e37e5ec52dc95d362f73343c8db74255e4942b1e56878592920f55709de0c026ce647e1ef5b094460877ae958d668b86afbea887cead3718b38ceb3cddf870cc393a4e6626a1494aed4cda3af20e5195d9d2c9886e0345c2b06ce8509ef3e5b33e727928afb3f06d932a023ca11fc6bc4a13116365894b528c2ac8b1f469df3e1c4a7a704a7990dfb572a8313de058dc70efe485bd66e18fcc4aeec3c4f9fbd7ee89bf678e1f20fce1d4c3518e24aa7399786bdab64edeb20ea11d36abb549c964d7ea694b59dba62421bd43e9fe0a336070585fa4bb54dd0bdee9585d767b07229a215e60aa4bd609b54301674c6aa4934a3da71cf8f07b02d421fbdd2cad512a5a79f8300214eb56c9da0bbd230b44dc2684aff00067a70158beada774d8bcae8241db6465a2be1828b95fe1e1c6f01018ad797f3616568b2dc9eb1bdf0320fc49b40be07b0f428736a9b7e18620343c3d12ee981364389e446c08c2478b0e3a47aa37e8172e45708bdcff835fb972433a1d03d42547494d8e2d5337146ceb556c84d8d2d5bbf31c8cbb2b0f28d7dd097bf823e354d211c86763acfc92130711cf14902e87cb45d4aedd799780be61cac656259f5611b4f3a2c51012abaac4f59e1fc123d4a886006bc9122a279c72371a236836c72ebe59843433a2568e5d361393d41aa3621ab01860f534603b3b41820099c08c8231dcc7968dc2a171d048877b81d388aafc3da0f657ead8ddddbea30def0cac3fae5bf55f4fdb6cfe4f729b0680c3662f62ee5b212c0324079837619294d3f31bc6092043ca4dbed8d8b4894b3c39679a85795426d4de0dd11899ea8935d149f0673a8b10148be8e53b28c34b00230ce86e8069dc4256d3fb93aea6801388de9cc1002da72c4f313769a07e662a04a25b98773bde2b68e49db02dd47346558e55494ace37391ab111a7c26f1fa9d1cf061e27ec0700b345197ddd48b59298f2d89ef970ac35d13799bba182d7b5a26c5a642e3500a23d6411047116f51c245127113251a6f848f9c8621af619d2a0cd59a1abe93c55208f903a5a447985e215faf046f68788ed3ce5baa0dba3e6eea7c6b8fc4fda594ff8762f33dfe419e9cb043a771ee3894cb014c445ee5a99899e985bf073ed646d774415498ae4cf4e38be5045c0e1d5bfc8cae1a90ade71a51fb80c4ba69dadb5697dec2a300b151e3942477267b1bd58075c67f83023b4c0ad64872d3ca6f1615a5468556970ecbe5ae9d7d028960c4f5366e1e267d9d6146a1f6d05b648ad8deb92dc72d240fae5f0d4a121ff4fd4b35211f688e29de62d47ce6be00d40d8fdb1dfc74e8f45683454341532fcf637444aaeddcab3d0fa1fb4d1ffc4649c26fd8af55de470c94d32f09283c30cc4e37261b7747863e810c416b072bd2996c1717e88709a6a83138a9c4db04c21c4dc17260808267b453650dc53cff5d1193838c9de9f6a51be02e91a6af63a241c29fa901f7510702ce2ff550d9b5af27bfef86bf50e6a67df3d962206180390aa8ee69c995954e68cac40d7c5942f9e133a2123b671abde297ac40a6df34c209620c9e5ff5c1889a51c96fdc6ebf126b08ca27173078991e8a232656186a99c9ddad4deb5e42887eae5760d7e7b233d24d1fdabd987e0d56b2c8ffd8e6735e0ae4ccc1f3bb34c58d9ecf40d0e90d9178b97da6b52e72d3f51a8897753a2a6e47f07f1bebafb876def79fb50cbfc8fa9bfb01b59daa2ebed137116bf2b642e2d9dbc63b11ed123595baf9b6970e94b720d041400f8259eb672b81946dde3243e60b1f44e06e89d956f3954105a87c421acd26388d6ff5f691d02ababe23e32c21f1dbb25c0804f6b9241b42b2207132a8f450731982302efb58bdd8a7274317a496613f19fd0f2f680ae3a8316f0fe711c758ca1a914e20ac15482e3d3727951ae84d54f74c14c4585afd673b08a8bcf2c8114265ac7bbe4ee645609f2b0b3cbd3a349f4e158456fb8833e84267e34f670cdb22d7e758cf93b4cd039bf2b8bd10dad5854a22a3efadec1561409547ed099303b9c747be3a0aec4437432a28508c03eea7e9d200622319ec9d23fcc896d50c761fc8a16d24b139b695c71381dc68fbcca5f293add963a0b2900bb02854790b0dce468a78219a9643ae3f5fd9adbb603b2d3c078aca2940e13706883da3a0143fd1ebc071a51745a43a4956c4d105d8291bdbd077fdbf18875527a33d491f30acf9688522725d2279e3ee73ca998f7486bdf12ec23782f0c20924f215c33bdaa4b4f6b5ffa8450b2bd5c83b137c29c4de03bee422554ff8c6502eb7721a9565d3b4b7ab7d6581e0423ea7682af4fb007180721eae56ba216cc0caaec3b7ad7701dcf94e261a7d9749dda5038426e94347e52771fe8b9a2d1be3f87d2cf4c357e9121ba6fd217f51e152ba044ff0364c3ea9bd0eed85aadf8c3b6df9221698df03ea30a9a4ca7f35e233ca3a183eeb44fb1d2a86bed69daf3f0c5c5b421c5dfef5e841825187039044b4ce6db7669d88c805a953514c5714b6d2908d2ff13669038e640294c4a2498e08b523e1a396d37a6391660286f3e274f4b8f8ac56f9d2fa31dbd5d6273801570b58684dbe6fbdeb5927eee7638c1bccab14e0a86ef7359f4101bff9714c420bc0ead339a84dd826aad4bc4a2d63947ac4ae0e28134bc7969d726b4f76fb9852fca46c86ce3e9e109abdcbf25bae6be0cd0d0daae970de7dbbfa3e8902a71886838f5d3fa33cced02d2d6d14d92f93b2d3a48ace575bd7bedc08637d04b631a632f5c42e8b628bbb18fa43b7b86f3f353d85f26d03b7fc5bb0ca284a5effb45758dc52380dc3ac1f367c4aa956b7d735feb2071a343b4e7558bcd8e96fca102c3c45808db92398d729e50d96560674acf839d464996bb274e9b2f7d5adc68df1872712914368915807debdde20357188e09fcb8223499460982e0038810974953c13102fa9967355bddd70cd07081e86a50d79cac8dafe7e9cb74c1f770db02530305cb920f2f26443c08af4bbd813ec8bdbf5b0c84b41b83ac7fdb3bff050bfe4b44deab9093d5a76bbf1aa425b1acc4cfd1fb31e9935899c5f71234c38418c7142225160a44d2010a631177d659640495150010c926f16dfb15c74dcd635a447c8d710719a03e07b1263e1c2e5fb323d7a159fea157d8decc4d9521f4a3e453a581321b98d6b765b44230e6937463c34c3942a4e87afa151c26da4206efa06360141e113c3f3eb202a840e91b51f2347a929c4f0158031a17c2aece43085a44f2398badc11b5da4d757268e79013fda596b61a75ff4d38fc960c45e4877871b6373b3b9942387ee37fef6b305d962033c21b4535cd86288d26ba31c758c430dbf12ab248b99d817d43b41410ca061913a16cb1650182c844fea07af1040f67985dca01b723525241b8770fc16949c04a4f3c18b63db26157bd7a658bfa29d4184fe1c4f594dd45904548dc86beb1e90c3a91c1ce07b92d0f7c1f3aa84750f8f64ef6848042eadea0932b099c6ff48aabedafaef129a8fcf7c867e5c5dbbbfb963671eb40b9521f8a8ff425c45c7d82452fabd0ee5459aabeafb9b812a6909207cc6fe42253cdf66c270a742dadc14d0e4b521f3905b266d2067c5971daa039f7f6ad5e9df43f9ef622469649e8c48e44ad5989f3718121164ad2563d53a399bac6a190bb01428d52222a9b02bfacda096e12400b0ab728dd4e859c61e717244c36f73797914581b657cf36c4ae46a5d717f0351de1deb9da63573c157219782340c8fb17db524af7e71504452d2d1b2e5fcb81f3b18cf70999e28b8fe7ecae5f1d6cec344aa7a1e4d2686f0dc34ed387a9d692b07cbd6c85b8d883e4a09100c08189daacd5fc6640306a532388c8134579f0fef773ca5b5840a0447bc973ecf8b5d165050fefc45a1fd1bbc171aef79cf995254a414d2ac38fddebe786638997ba623074b34736a8ce696446d9b4ae8d381d26aaf5549bcf7d305759d006650ac9ed839694421b48b858f41f067f51bd6b061245575fc4f9b676871cc59743cb987d520b5fbede628f008b62fd0434e1a231b82b027c7e55b2abac14f8951fd895040d3266a0582dfcf6dffb89f355b933cee5f5b2f65a65fe9cff8a1b1e8515a27b836ae1cd8d543677e1805df33f4bd7a153eb9ed3a01c75f3f53e8161574bb6634859211d4a3e9f84cd06d3a20b311da8e33124253c1ec1b5bfa500b561723f6c3d9e89360dfb3dffe5caf51d28120ed8bf05f81d349aeb492ef37fb62fca18cd79a8b01bce246ef0734047baa6e69b001ef60e52fcb2fa285406a5304d1b8d28b5a1bbbaa788d9921b2d66a4b801eaac74bf9f67d4de408444f6aa86a5e1012a6d93a589f5a03a1cc75847656a4c3d4a57ebadb0cc56ad67945890ec392bcf3524d6fca2a362dc4610d2843d54afcff2aa0de7a1f1c7f2b47c91cd7799211f9517aaa4feaba917059906b6b71096585c4c8eaf44d915b74a7617494cbce8f5a8d8fb901ed9868880ddac1734b85612b44e442a894d881611a267a5b60106d87e68da0f3c8bd67d351765bd173d98cf24e2ab653c7c63d697c2a8e43c454230a4b78de883a40ad147c9ac7cec5b07923c70ae9a503fc4cfcc0b3ad79a57eff4119be6414232e754fef8d077a8fe553da913b933c827d600887951c54f0b4907f68ac261b818c23689dd442157566309efdb59caae82ce1ab3840352f1c9a93ab189c79a0ee88046d222f640aea58f108e7336a71c881167e4c802b9c4cc2a5688a6e153488874e6d659051e3e1b8c989a42bc918dad8799dbdbafdd45ffd64635e6c40466e04b1578207e26ade528c67286dda8fcb41a835016468008c65644647cd53a173da62be8bbdc20703ff3248468f396bb24391a467fb44ed70040796803882a0b5753ed606b00e911ea551fc5fbdb8c1a4ecfc9138925da9387d0a0f10676ad73ca19e3493a1083b4c778f9e20d597cb5a1cd3cf1c97a290589718d67f253287989f68c251b76a6c11dfd05d53f87c334bf2a29c0b9921228ead1e7b3d406ade420567597c8b99b8e60b6f53427b6ad340e9cb62c40fffa8cbb1dce040908a2e75f334dd5f7837167e27dc85371312ba614979db19ca3e2057e93fbaa04f3acec27604b61862c9bb0b68d71fcbfd805cbffe7e8e241c8e4bc4577e5022f57addb4c4bb18e6146f8553a6f9828f2bfd73f284f90d7de452454b2e752cf6e216ed599cc6612d5410440d326c66a919d5ac13c2d580c56c1ce71d0ff80845a88ca78d2826d14700ebd403405302e1079b10db8166481c80bac2d24509405ab37ea79f783f48b5fead295dfa8b71f8f6fce9be8cd16f4d72bed75a1d57b6c28ed91f8443a3ef23b918485630004eeb211ea2a0d32e5dafeabe542c19e70e35a8c317eb4b7e297f143031b7b06a79f51219fb24dd36f46ed8e0ad4101ed70bc66346ecc08003821e02be7f1153c64c249a99ffb9be3f27538a338ecea9b77fd735485c2756dbfb824ba84be40c5248e93694cecfcbcb3b7e53f082d179c27355e88df525bbfa34809f11acdadffa948a305c8331a1921df58a2c4e365b3599b1843d19fe14e4893006ada1dcce4ef75fc284b339bb327ce5c518c572d5112bc8995f693eb8a9914378b738fa8ae00382cac2067e496f5c2159a34b97fb8a47d162fad13727cd4c998fad545a332fd60a473ae525872da88599979d43d4745f5def16af6bcc6fc84665d6c2564389382d238b41fb31e6b8d9187c9a97ed5a5ace1a31d2d9ec010fe8b3ffbcc926ede9e3e396a94bc03183e10c89f153eb5e3a1989de0ca852949f3b49b5b6cc4e4849137fd324515c73d7370424ceebcc9278f227095f16b6045a412afe9133230c87473280d679ccebe84b120bb50858b0b9b5c15078e756bce4d88c75109f011648b35868ddc3c76906396ba208de8434fce1c2e327793435fe37944a0172b7fe42d1b07229004db830da87e248c8c5f438c4c1f689656e1c35587e388d255f8136103a4202e1bdc6791e20d70e7b2c9b29f489f7a204c48c7f70f8724da036aad45c65e040e6fe3bc5873c177f25cb2fa21ba6c8ccf3b02c4d1430b19cd09156259c5b92c599d2a0bb292c1cb01575c26c1511735ba2867b890b80b26aaeaeac068e0d7f9378518ef31fdab081b6657e5905e1e6b3aac1cb794401b342994577dc6f9d820c640527f031ad2ff1921fdd4abda8cb6be680c28b10ea75a69902e3b261f7d939ea6328d105a914771f26b753bff35da737c2b72317a6edd633b989a12c53086d35cbc477f6ac8ac9177e0678db425681a5335e3bccaec64d4175a6127e6b9099c23daab46f6353f4f10e3cb06137a8dc63ad1bd39d2e10165d93b7ae1b26dca53050ab769ba4e265999859d0689a3ece9e9e22fb0adb9fc133ebcbed4731efa004de469d025e2510568b956367f2db23112ac3162d0f45d654777c641ad78ae36203a675854869c73704d60b4ad46cb70cf13a83a6551f5a0288c5c8f45c46b988417a1e033604e3e5408adedc5e2a7aa2269f7b36e8b6ecfd6e94a8970c9b1734bd9d8a9255dbd2e7bd2b418a14a2b3f4a0b23122c52c1ffef984b2d6ccbb0237a417726866716907a572b2f7a8f48234ad3382e6202d13f66d06ec8d8cd7fb23e956fb79fb2d2b0760719262c7b7a60c98d266f1e9d4c4869aeec781dab8851fb7b493775267bbc8d0a856ed1e2a5965cbe709e9c2698ef21546c08471d8f9d9c0efc46cdeeda641ccb5d6e3230dd4b698f44ac1660387cd81cb0b136f389ab51711c5a259bcff77bfc3db23d20143ac767268771bffd68b4435bdea191e769fac76d1fdf80009b88954c46ed6b71c9c7f298397e53fb1e48fc5ec3ded55790dcbbb5a0c62adb673ef6cf46139c469d459811d3ae97be803cd645bf61dda187af44f7eac7d7bf281aae0d0a2d1a4e0e5e379240c195c04d6b327295ffb6c423d222499dbf92eada6501dd2c74ba5144167680d5c66527578f3f5083b529f191af3db7fa87375f3cebad492d5ee7e4df82dcf9c650b732c7de6be5c4e7170cb152a4b662661cef0ba37ef9475e5e6d7559618555e60223adc71bf123ec42b019855d0c6d09c2beebd41d4e781391839c4d1f12e54ee720d3c1ba6384387f97945546a23b227ff8513f7e86b203d9273d8348b11fa9af64b7664fea554c801a52625bfe8321dd5ae2ab35ceaf478e9c5c526ad15ca95ec696b8229bcab0fbede17237e5300f6989fe4bed1475b737129ddd6dc32a6f27f08f76c6c740f542a9d25fa59861f0ac7d96bd6c092fcc2dbdbb974be83938ae7d67d360b3a9c4b791b95d464f00df1e9e6c56fbccb18f9f9cdebfbc4c5f6071652049c32741799f44556fcaefc134eb5c2f65f4d42f83d17349e2ca42060cb6cde600c5e4079e1ec83c86861a2ccd3df0a386fd614cb46f934a45b40d09e31905e1c8f6cbe4a92f23f1dbb1b2378665217365f6497b9f6fc04052ae95d23e2a08ff091433ac700af5d7cba6630fe8cafade67966e4cc5f3fe3c3b674baee949ed819a46fe704cff13b839e66bbb9e3e87ebb90fd7aec44397e4f0c497a764cd9e93babadfa287f83963fc34ca4b292ff41edf90d45476b7048aa71905186c0e6e61a375de1143926d55ea81b7c846beedff2bb8175e23544c6184bec2100b79d8a54eed9e8713e9550431711064dc1e1e7fcaa29a9815338e251d50e6758176e460166401ca7f3bdcca74b8eba499649b0b15abf6adb1828cd1135d8c3cebb2ed8bece97eda686a41c95dbaa4e41b5b9a51a63bca7e36ee3a612e270c266f4de1af74bcf2fd26a534b44d5e357301258a812d8dfdbb1daf1516c1adffb37013a8897c7ce7ca974f1051f598bf0a8772a637c08962f72e7a90ea40bc426e561d1963d490118ea270a5ed0e6bc55467d229314a96257ac17c4b4a571523412a0c644f1ccdf105cfc1cb75c51bef32e267a87ed37394ef85cc13aa6b00f287cbca777ec00fdcdcb54e5b6d1d1faff59915eaff3f2979e8a5758b84ca4d04aafb0465270ecbdafb81d644da2d0dfaebeca76729037ec5ee297c662c8e0e8dab8b78751a045e04e7bc5c5bd3dcabc5354a232b7313da892d34a31738b6bdfdb6d432509e7538a11b3606dc1ba570e47df8732d07feea91a2cdd66386a38bc6c111248ea7bc2c25542245778cd7b194cfa689f1c91ac96bcacfbbf8d51b6d8dad42b82087486fc0c620d1981eedae9e5f0b4f7b73b6588fc51ffb97005a0edbc492b0e791b48293132b31748c03d8b03e50e563385c5c528191393d6f17999c5b8388b4ee6ff95ba49b28a6b4046feb088f23050008056e446385a7d1fbd3c8c63084ad2308bf82bc304b8c5192cfd98296ee118cebfec260448c8409f9c58307d1e17cca73c1e8aa3a7cbb71ebec8c7398f3f5ceab9e4a31bc93e284b44ddcb429dcb197b488b5729b8932490afaa8412c16bcf6d0a9c00bd3c4a25b99610b546aa91beb424479a0a911c104492af288d3ea967c3dd41dafc3e1561fb774812c723ae36302914d622ae2f52abfde26e9ef33775977b7cc62f3db269b7d69b6d2592c71f52c20cf318c0c0b7252a6b948fdfc0204bef769981ce9c1616bf2fd5280462c894904baeaf5b4f6f63a1be9cdeb77d051cf19535b7db3dbc4d2e3a5d6b8f13fee6f8f2165fd13d42d5a8bae3c62482453c09a858a8c9e251396ddc815b23fc1bd3cf639d8f7d6d17c6b4c1bfdeafc0bb8d08370931e3dc23ab513a1c1aa6bd3c31698adb72f0388615c1c2ace5ba7ca8b6b5be6aa40982dfddcc496d566e59f9d37a5415e3ee66b222c9397c0d9ab3e38656c6c41ee73e7303e2260771ee0326343e91f3d90737cb03e9a02c261b73093fb9f93d0bc012703f654864e5fd8a068e36ca4acdedface33f0d6d778350506dd5c32fc467d8f1c379f8372f5e26e8b100105c434fd72e86eb1824cc9d2755798477adede491fcf76f343ace0fc2fcf37e62fd905768e9e0889262bddfbdb055ddf6aabfb5db4e9f25eba6dd331c2c480c82e3c469251e0b409c96cba63d53feee3b8d79083f99063ad553a583c3e7f9cc1e5828f7183da40adecfe61ada0734273563dc17e022e6f0bb1f04385d4108483a75ee9d94ea5291307adae7a005ee8584b6b0ef54ff24668631a72a6419496fcb82911257c893b1f9d1527f52cb22f9e77562314231b88be18756c02cf64aed32596ac441ff7bc6dae4b258889735a39e0183e52d89abb4ffb68577d2362f0a8a11606cd5cb15bc41a07c82a4825a3aa045c8e5739da30a677263ab87985f5bcb9c916af8c7af11457f50eafaf80c005123cd1e6d3794e967d9ed675fe26bd8a22c19a3e3f7b7eef483a71acb044a0ef9d04c446336a31834775ce131305f98923a29d674e48d906986037c37b10a733252fe5aff7a23120771f8776225310cbe1c304fb3f47bc75885d1643daa7053cae100073ef013cd3d5adef26fe1777a5a4097ae5745630296c42c9d07c82a14a8c1eccfc0057efbe7dbf7888d963402efe8bfc94857cf172e7085f1e21c93fa5366f46c013d267d1b50b9e198a188d8e5b042c2c54028831c26d3c0ae7dcae374a2ccd0ef3ee1364a34ad02956ebd5b33b2e6965a74a72e66794587d5fe461ea8c539a7f2ed28986712bdd9c97914ba07c07aa743eed406efea4cc9cd7fd506a665421e07ad237c42d02f2155f2155c800c21366f9d55f9e2ced7e8d2081e16d2b336c1339183043d56c4486cfece256f2ea0ba65cb621567869473eea8c93758bf5073c30b43b64c0f8c678d9979917806deacf89a87e095452cd645270ac7be7f30716a12d4acaa3075d9f0f3d3452582fd3b01d80b4091da14bf9da2616fb96b8c29c8ae34d7758edcf4ca3058bcae2274240035c0435ce866446b81255cd964bb48a8c3a0f51ff4a9d513fba8c57f187090d8a54eb1deaf281a56ace0ef362854be8e81cfcd434da08377609bb600c5b6a77c6d50e3641c987eeb478fab1ca00711c65753f10f3b72b846ff050cc59a5f8629cfcd0bc099f6f379a7a7a95d03f265be3ba803195ce4e8b5077fee4ffc86ab9b9eed744229057fa61c7e71b6c33232195d6400ae6c11c2321c01b0fc15ef6a045346dca6450f384102faddc7893b4c26055abb816ac5aab16d7eaf2f197e04df48f81562b19fd5c8637e87fd85df48127128b04697a92d4fac709a703859e1beeee48718b11426305be638ab797345603774480af24c3d2e4616ae24f217f88211a5b41a2aaf221850943b4993832cf0d8baffb860a0db93e119bc49a849ef8ed195a78a0ef2487da1e5b4ddd3acabd6860fcfdc12535a80467a513df55eec226f3aff2d13e9034f0582c333026117c8e02de3410384765384eaf715e4a356e6a30f263ae8183b812589d78160963b1554c5474617f4c2e89e768420cf976fbbda8e3dcbcedb24b82d3d2c875a71c9741798cf50b140014dd4912256cea18d1f6863af7189ef321118612dd9a136ee238f4c9a69f5edecaf4da015a333ed2ec03711de0dcc167846031dd52b4fba246072794b3feb255a5f3a1cca8fa2701ddbf9f88b44eb67b150747ca5362869410d1db2882d0fb5924a95065371a010ecf5c9f470579586dea37c888dfc153d307f8f60847b5af5317a82ec242768732295ff546e18b2519f273ae9043a7b51acebec594fb4104f7c2b1750c39d4a255ade828e410add92bdb7ff69897529dea29806d0548cc098c1187427724de3cf1af6777589953cb6197e1cbb546cfee0f3ade960c93c00bdebd89fb49c64eb512b8633d1b62e12158dd97d06ff0e90efdaefb5ffc62689ff68033c6c6b1a52d7c912a65fa451f1d4b5ad0deec7122cfcb566c1f9030afb4a2d135674a693e8c59268ad1ed26057d2bf84ea0c57eb61582e1a2ff56de22e6ff4130b648466a89cb3eec59ada3a73c654c708f7de467fe98499f07103708e2f509985e49f1db473173cb478ff2bc4fbe095da7b754e73aad71d2b78f63a3de5203900412d7a9e29f3d7760dce265c923715f8801ad68ca9d9ab519afa7badcc54b7a735c5fb121ff0dee8021fa0b441d4cdf8190e8844a86d1d065f8b116eb6b052be9cf1a8bee469b0c40f7ace60d51c81a217befd315677dd65687eb69803c5700abf2bc49a1610c7b137c117319a3602cc99b53390f426257a0eca7bab70b5b08bd7904e48735c297be2fe0df927e62ed29d9ff74d693914bb7ca821111e235d10ac50c6baad9246e99f1c3fa4f87d06b54b91300e78d4babcb97a68e7e514eb3ceef62b1c4342b7cc295bec14eef340a0ffaeb71211ac576e884153cce61aa69eca9cc7883d7d44e1e5170dde4fee9580c550d33070ef7a546588abe29c2c181cffc4096c5f7181a5fcc2fe8c28440cba49baa98402f74596387eb52235d8b701e36db1c7a3bcfa98d167bee05cf6bdb309f94b959fe0700cbafab0c42c28df00176a28e3c7bfd9332bac6fd6bfde90a14655b2e455b577b5d8f71a183538959d7d2b88de4e2ce6bef72decec76efb6626f54b4095ce86c9b475338a1238d8a9f25abd55166c11ec34d63fd203e7df43b9e1af0915d06f2845c5634d861d274ed868034516b9a65147376b6d2c7afc4270c229bb58979841581a2809db4757bd54a952a77df2257250c6d9896bbc101498e447f77a41a069c2f6021f0ddcaddf78c20f1f63aa2417955c79b03b3c7237ae1577033eee19f0408e272e7f684cec932bb014a1fb4c99130be5524e340b8eea875d8e66321ec8bf2b4630f1ccdd675208b3ced00126876bcdf53bf279e1ae75127ccda63fd8fe4ef2b9d1ffb4efa638ac07b363d40f65c10e84ea507b2cca899f4e0ec61b7fa6648230ee5a112eb8ab55de67cf02d456df174ff1865c3f7c7c585796458aa37146bc622ad287737378322c02301f5d15072f84335bbaf60623a33fb5ed45b7e948f8cc38cacfeb8836e3a48484e5fc31b9e93fd61b6953f1dd70222670ed506de6c4edafcd883c1babc36d2e6d9b2c74e207d5e67b8f18c8517e0b05df1d0b2739e02696613acf25bdfdbc03aa580b9286ae6641299cc1711a06d87738acb0f94f2f5722539481e156ac096e52a18b1e0abf2ef14069b031db1b493bfc592bd5fd8c4531c111e8a933faa8419de7082fe5157b89f1f0425fd36c52dcca5fd4335dd0fe45751aa58f0c19740707ebebe7b85bc4cae816ebcb3475c4a4b71734dcfd62a34b1304aba536165745053d24c5f7d5c33308ee4c3198c4bde9ce247764569a02e73cc4a5b7c3543cdfc17da5ef09284d9a46e0443802c17daf47cd04f42ec91a982c9d4cff9023d933f835ffcca7501bf7a17c96e56f74a088b9d938bc102e28df59026dd87534ccdc47e7b5e7afc3493ef19ded453e7f704f16df1fe2056c600cd7e35a5455544d33188815fda3b4bd559ac27db21ae2ebe8ebf18f95fa270a5138abeb4f7c28811a37dc0cba123963306923439d831497d561a7c48ee6c48374f9251691e3f1c7ad21fd37224d733ac027d975f744b70cf6d970ed86f8c0996039274b3b598531084cb783ccb32588129622cedd4a77d7ce5b928decea2e13356eec89e0b111cb0a849576326e0a32dd201474354200b3bb4fbeb23a9bcfac7dd923c79380305de8fc4f164cde141ac4dca37704509e0b83d4298e7bc6fc7e7f8a67203edc9a9913c44f46bb1948dea4122c0766ecc4a6e3dc554c5014088fe9ea37f313a2e2449d43ae3f9c44216cbf36872d2568895ce38e8a541d7574df24fbe1a9bced81ca102a8940318b426722539df5570dee0648a3c98625da1eac38751945c7a3b790523aa64b5603f712dfa406b8559adc416cf4786d0c9fe236db3fc5510dcdecf280ed854c226c74666093f25eebdaf76ccf969665a8805d5da1427f4cd2958420f4f77cbe9b030890b43eac00072aa9003d74a6d90037ebbfe4190e30c98a6cbfee53f6af433b8dc9122113103b1ed64db023b94ce3b86bf3168376be6bf04c54c2fcd8c9899a8028e0e322e4093496ebb01a8382edb1f6b666b06fa8ea93199cece8408b1c41be53b2c4e7a36db80fd09fe31fe8838721d8b173fddc44cf7de5fe9cdb219b09ede1f18b29894d6dd593e26a5e4dda846dad0c0d7fa28daf00280d7241dc03a8b8c6b54eb69f82b438b079ad14b5e484cb9f64fc2900410152798cf48fd213ce250da179e9e271c25f50b644acdc5fd10dd64e1500a92858341ec802db0ce7b297bc39cfb03760ed0944e2677e2c2e12e1d9e2d1400500162a85cfe28adaf0354c50c5664513d1e6de18f63c86bdead711836a4d5ca5cecd49c3549988538b66dd3a7d320d3b6681b03835b9cee6851bbdf81a86014b615855b2ed969ae971bacd3f9048cb45b8e4d1f56f18025cbede03ef29f2b2d35ca17a62f096146e78a2241b52990f598344ffd12faca4daf505362f2b2108b33d4dcf078fefb2dc541dbaa97a9b5835af29be7bcc550e7ac5375fa053f61505362fc149159dd03b9a4a3901cecc9313c5a6aa17f3d6c23090ef3bb498760d8ffca388c3ab8d8a559dc68da315acfcbc7185f5fdff00dfc81cd6fda54e847b9160d322ebf70804ce5de1be37971a801422419150355ce5569d602213bffb1635c4b7d1e9871c110317ae4d2ec8646a8e04f417ca3dcb4030d1ebfef95de5bdbe846733af4f4b300952041747f57e41e67e1e6568a6a42642436eee37147cebbea2628adbde44622642f1f8ec8ac60c3551c0104d8e15a02fbcad4140f55252951ba8f1f95396669577eacdc7439fcf8c388655a68ad0ca4cd301aed7f50c45b7f8af689ee646aae4975d226675e67358d5233bd721c9326ce671c2132f7c87e37ed2b80469ad711150ca6f9616a733791a2cb8c49356cfe5735ba9d8a852b2c0e49e3525876cf8ed556612974602b4b12694d378fa583a06bb46b57b7259a56ac0abedc2598eb97ffe7d89c493ae34fb286ee40f8f69291bc1c79de5d4a1b17b343435facf915aa96030ade1f3957c149fccc6b488e1ef0c25adc2efacea68d7f51b9b863f5811e197d4bfe93835f42d2e9455d631bc6131568f21e02e48119ed14e1758d39fc5f9753b18cf478fc34e1768eeb3c30e248a1ddfc5db7bdd328230d3cea6d27b80a7700da55a55de33bd01c4a47570dd99b1a36490cd04835dd447b50d6f20480509b92c2478b41e31c87eb06767890bf67f196c7f8c580153a39a7b8b2521e9f9cb33cd20aee6d38e5ddf2066b4372d4ee9fc565039392549bce0a36109dc1e6cc30fd84d71739ad22a74bbc795b4fa2204b75956ce2995f9d8d89840843e94f29e8411b36d75d41384e5292f588803c4cf04712df977646beabd2af77e9af1b8f73bdfa41bd61b410ea4defccf8471e5d75eec9138f0866572065784ee6f2d36453917374c5cbaee70611c5dbfc62ca9fd6e09f2e4d7ef6d19e70c0e9d7f6046cde5c92b771a083d2a8c23f287f8b2cb405e8015e6786c8385ff15ffba3bb5ed95f0830e94881bfea19f75484feddc574e6bd04048debdf40cf0aa4999346498267f1a41f6f39d8ef3b41d6df7302287e976df4db9633c98d65a85bf6f9197d206298bca55ade29f6d8fe654a817ec1861a2fe00868954c2bdef578cc9cdda7426428db18c8436613250488d1183e15447e4d987773b1942d778bbeece50004ebe99175e3d6f7533f623548ae3d208948a99445b8a08bc038075ee25c188e441011c6301fff3ca4e5ff939bd344bed9f29b8e811f81387d9da1c8f214382ec2c63b4fc4a82d31f26d1ceef94d1fc54b2e71644c9c5b72178b723ba8cef92cd42855b900bc3557a3549c02316bde3bdd7a2eaada26a2b17d4c80631f45b8e9d663054dd8f7cff1d3822e0c31caf0453039f05d40cf91dbe14fe61c5ad92e32713ca8d4d8225dd1e2d3729ceb0a604bd27e7b0a97f0c52cb4a57a7c21b8f7fbb67f265667bab774d897b96154d0efbc34e11a6f393ccedf35326c68485829cfb7a471035c864e574b80cf635faa45a12ca7b5bdbb1ad14ff90b484a303fee271176ab9c3fc248f0e267e99304b76270bbb3b520d0ea95477df162697cb5f5d57264e6b5bed92d449435fe1e48ceb2b2b57fca45e74b418b3865249f1c6418c7934146d76b4f1ad974046f7eda77102f47463804ff54847efe7995dfbcb3ff2c5eeba5e5077247b0204772cb35c35057752e4394e7d84c3233f2517661cae49886a5838086c53b13e895fe9d8c93474ef4a5dd8738f1075956a73e672475fdfc7edc79dd95283fb028de27c701c7cdbd33bd13693f5e654baba3b7900c7b0bc6e0cc91d01899014973f65b31215cfd6eb06847e709116ef4771cef710781e21154cbc80eaaf3c603de443dd32773732727705ff1124c59f9014ca461a6f64f1cbe06c35f34e0bec43aa425d9bd3e90d41e3dafac8030d913323069b392dc49ddc74bce83fb9f9a6174fa311bea85d1536139f752337a4254ddc53b000735dd9ef01c57c0604639cc08d4643940c75318fb6f0482daf7acc6c6e91d437950e6bc0ff5fb4f3ef57591964cd55c02e41520493cfcc6a2e4460987ea6f3f7f6a6f6cdb67bf556188a890958c4a7762fa7af112f85454f4e2dafae3308422050c4545da73115edde54b803a4feff8cce0d75db40ca2f838e7e1abff35bd26315f06fe496154b43d3e213f2ef5511ce3f5603a2f0aa1ea991587af892d7d91ccda642f5dd471f2a8bc2d6f74d0ea5f0f688716eaf8dc0c9457c22afce8e97b9a57ccaef9850e5f80c6e5c2cbe3d14d5d0deaeb26d3b429af187e2763c373f537b136d5cc76415e0deb750f6bb5998fc9fdf081acb742cff6d4b2d7db2f41a5be3c32bf6ddf0cb98ea56329553bcfbeaa0a5462c84690c8265a2fe7c6cbae82185997e469fed793b705b088efa3248e86f4dd7a80205605e6255e38a732e5fce3f0bbe20468037ce3247c29461d97691a4529e9277a7cf622805dcc37f3a4fd737c46d589e0e0414bcb37e938c77e42f19df494f99218ddecd41f639fbfa9963515ab6f69e0479cb839389572c6aa32016efe0c5fa06eed7001d543e886acc30a82c6594e5e5e8d2c57e01c9f9f7a3983edf56cd50cc48d9b753845f8d876ca8aa505f038ddba869040d2e05e0a0e8cbf5e7bf9e97ea6369ebfa2c617bcac9a54def4795cca5ea96f71532c030980ee6700f9955b17402a42877634c1ba43d7db10131fa3691b6aa9f72bfa8a98435e2fa1a0bc7e11eef62d7eafd7e5c1d8c8c8d22dccf46cceb038cdebdba9c28dcfe94392fa3b4e5b7a48c51468507814dd7b4445548c7cf5e84ad5d823a8bf6197da47037ce0fd937cf7294c6127c5746448926f6312de853cb4b7db92da238dd2f8263b39eb1c96a7567679874e14297062417cd53ac0597f78b8f87b10022b693355d01a62d742f15343391cee36bbd7e1c697cc5b29c2d56987331dfcbfc08d2dcfcb390754f31e815658f2f9a24aa1eff72641fe6990858922a0c567d0f2385cc0dcca051a68688347a10250fe4afc37e8836ee95cc679f03d8573750c93a2ef63fc5d6d20f07615d9a7bc1fb38f537c98a34635dfe362bd491626fa20ecd0c8ef7ba8db175118973319fd283a7c21b0c83ae24406b724e623dfe89fcdb1382f57576949bccbbd4a98e4b42571a1d3a3cdeafac85d785fdeadfdc5016fd0d37ad281723cff7f679efc0b95f5138b287bb3df229141ee7ff9a9b6f428bbc506502da20ed7dc5a4fe563991c6b4419ef5a1aee7b4e6938f7451425b8209cfcb8ed147ec6dd5ecda1977568e40650960d9b8dcfabab5f31761c2edc20863201c01d910337857003a9f939f647b9697e3c16e19f966aa5073250c51e1303a80e65a2d20450f0b9caef7069fbd3ca4d1c6ff0f5b646b97781a285d4a53e5ee89c679d1fb650a84670891dcb6e826fc7675290efa0cd9f867444a97387fa15fc8e839d9ea4ef59c31080ef49e7e555b9979ef6e999cb51c4b4459bc9609517aa4a83e7f112c0a5bd8dfea6a6043ea4b9f6ad8c42f29da5b4eeb7fe20a686e4a2ac10e0b038cb65bc4e75f1d0fecad3ac770c01689537b3a4242b737a6d65c870502517b591b8e44368e5414ea31161b2fd4227af4b81ad19d9fde298f7c744ac6c704959e0e61c4ba5bb2434eb2771905e5043a8614282301e071bc35b2867b01abfaf099d65d8802c54ebb3ac256bd7ef1c0025ad23af1079e9c8864042cbd25922e2099c0ff37288d115659d91b8db53f77037003c7565ebd1a516b76be6b4d0c5a39226b6e67bb750a0161934a3213ad1cd7abc8ec6a15b2d96b10a7ad68bba8711a9170b49213e651c67da7fda1291e4d474fcfba16d14562e9fe600aea774ac739d0acce15b72438f14aea4523eedf76277523162441802445df34472843a26983df929a760f5ff0514abc4360a250209da8d6860226e437f5eac6c9977577a15a1cfa26d4fd00e6b34ed4d0db0d9aec12b691585c3e18e60067e30f1e7cbbc277ab2a2e7985b56f105d8b233d54dcd7fe235858562b8f1f200bfba815af0cae5e2de17b88523224203688b288cfac7b60ac416c86eb716c819d95436e5f959d98e6bb4b2375a0d21adfe4183b85df6b3c274495972e86d1e61b0ada63e0d9051db0849d19912be238a216a23b38906426fb534bb1267419111bd3d0f48a49526888ca5f779b078e1319c3733964e55e651befb163dff2a9e6afe90322fae8f3688f9dcd2a4c476bc4c28c03fcb7851e29a14139db73ee105e44e42f9e61d908aa8d7ecca99019a4535e399b9a9138c4f9d537b07c93698b5e22deee2272ceb9e4cff01ac16491d4ee704d07da3a28c6aedb028e5a6770e9a3be7b95349378502ac4ea06b824aa516095d00906622f6ee23af2d408549338e5756571f3ec5ece4f3173a507ce8e02b32ac5c89f736d805cec48508d03fd21a275cd7e4fdda33694a0fe6537cbb5f1ef43d4d10572cafa5a8d0a4396e1a69271a69807363d5bcf0d7a8c1f3db0b752ea82eed5215f4e901c0768ec78685f1428cb5551843b654de74874b2405dbb7cb3c10e28540079cdb093218fa328f1b8cdc21a7c87c9d1e7373124b3489e99b0e760957ff8484e30eccd532ed66eb98e2442faca3cfa1da6beaf1f317061246b7356fb2f81843a05be2f9af5a06b906bb58d4fe5f48fe0118c54d9580a0d271c4c3b08cae6d79a8e545b6546820e0107af83af70bc888a129e57698cbf2fbdd98d977375df239848e4462e6b66100e457e787326608979160506dcf0ed4afb297aab6fb62ef4210ac1273cd14337f497447280ac58c1a6f1246dce17846c8824c01e46c97985165cf0f614045f1ac352514a62267c24250d895c0b7ccf7e3a81cdc543f583783e07ae89e1447f561cd70bf7faa40a68e2416af44a9afa41d75d887e5681c06620f92130b1d43296ae3d991539e9961452978ace26e67fcc3ca374045ee577607cd2711be067328ca578082487848265c79024ae5b5e0bac9c78a9c445da9fdc987c498f3e036f43099837a70200c20e406c1decde843e323fe8641e66ea39869623a3a4b2bd71785709fb43eadb602f6003415c9094c32391f245966d46e4944ac53225c25c0741df7520790ea651e0e7cf67e061c477f90957359995453e058062f5bcb049623a49fe455de1893f7df6b8673ce50214dfa8044faa296d152a894396805c9df2138e30122f256dcc158d3bd15f1d9b7db816295068b75771ee13c7110b37c3b8e0579815f8c519342b33801c6bd94e7ad112bd3a1c7b8d8c73a508ccf6710c207e3be20d252a68336a98c14d8e5ce09a178ec992e3e70d1daf1c6c747f0c938eaeacb6abb4d292b70c53db2b8856957d416f8229cf8879f9cd9a5053df0d17919d11d289d4cea9981ae42a81b0287e4c6862964988a9983af02be004bb52df22fe7997d9a03897e57411289632fd8ba7413b0298c4b161b0b8c235d2ba7fe3caadb90bda1962c44e90e93b460d6ad3daa8ebe47366745e015f5e228cb92b1d0e1d6ee5247dec30d3241507e0a122efb89e6aba7ffb53a2b77a58c1526757dc935f3029e2e7133d421e294c34603c4ea4a2885e1db6424783d3e0413bb41a3a95554e40c2c155d45bdda256e2d61f4af97a9792ca5755a154bf6a4e5bf9ad8b9367bbb0679e17fae0804573a7712cdd47edee6bc855969eed6a9f4d744c7ee9e3f81fb1bf2f4fd36be604d2f7c48be8e94b422b75b305f2b2782aaa2e25f58bbba70e2b5dfc1756680707c464221c4fcbdb3b7fdf4cfef753aff8db93abc24f0f2369698c928a9288b0a61020efb8655a6d75e141a360c1a7b67d319f4de53b04c7d0ddb718a47fe36bd01ca4d808f19f1f292dc0ac5ced81306d3768ab3a91c901343c5cca35648b2fd915ee2a4ddbd7bf8267770b1141c7f7b52f2bff447d6fcd54bf617685cb35efe7999b26c189a687c6f01cc22702a16bae5c6a862f016b7bf33ebb68f0d6e6a57f8980b94ddd7e06532b085d9dbb57881e939ba8b6ae91ab8ed2cec1060132354ede0648529b68c930c6aecb4c002e0d21828890f64f4b52a979c7fd5d0c62e59b42dbbfd46d28300980e14abe292770d7b47384c69e2bc2d82912be1cfa24fc02909871c5407e49e8ad8030995383d2b2fdfbe267613662a0973554a14b4f0b096a7481d3262019850b647167f8c89215dbe1bb25009edc29190249fa6e9a4ab25cb732890160b8306b88cc1ea98bb796e9a3566bd8231900414a107790737c5eb8cb682371bf2e3609da0c47f355620aff0c7a811b813856b9e66ea78db49799c85b2c9444044d903a0661d8fd9952197caf3def24dcb3e076ba0181f8e10b41acefad3b76aa7d84db46f7f70ef452799715aa4337159919e0634647f3655cdf37757cc69182f29a6687449e363df67ed80aa2f374ccf6c6f7a7d5ae80ee39931eeb1b381ee6bc4ebb7ac006246912d61e5e304adedca2c0886c35e82099edce8f68b9195c666e9913b30a50d65775b5b215975b7e901fd8d27d6240cb9d4ed0e365ded46114fad6cbd694a0046df239777bd3d6b61a29ce223f23f8277e642945d8266d72f51371214ef32d31a9c1e395ca9ef71c9755a5a65f072d53a6d49e8fe440cc410f040034be14dafddc2ff6419bec6ac647634dbb8bbfdcf97997f71cfb20a6d7e41f22b81418294bc58c9bb4dd45dc26db31f68b36367f132257fb889ae6ea7b83a4fa4cbe079f8e7076cd9c44fb21154c286b992cac013abb0027671f50509d1a01d1ba20c29c4853f7c8d978be6866acb4b40df8f90d4c538442d109ec90efd9d162e71e3d82cfd8b9afae7d97da034b84c7b27c0ee59ff6c4975714ff5c982241b3036f7719d83dd111ce3039e7dd748306078d6507930881c6361f14364b39baf125f700d68934fa8339e16837785f0b1a7907e38f6c7951d8d703b340228963b90213dd6c472f689fc913bd138f03853f483ce9bdfcdf5574d7491bb8c25b467d1287ad275d7cfb3e96a8c5d2fcdaf0702c1361f3e07474a1fcd494698f3de83387d29d3988b588a20aa9237ac0146558db8a8aa78c6319dff9c21713b6a41ff9e40cde65db5b13e792748dd2e73c4f8887998d999d88245b2307ab0c5e551bc40b8a9f28f7e4012ba05599cc84cead168afd1f05ad78fd1e37d8746453bccef630d103d4e514ab0a35e8c9595d2cb815bca51dac3d3e411ad662d976241b30fe07b91ffcf50fdabb2712d92f136459c4223ff85e1d26da34868257a2fadc43aabc39300f54ef829b4c5337279ee143d5e8be675ed2d996ffa1ace354083eb6ab3250e2399a02c161ce6b3850c3a7acf13865f1c1acfa7b829575da5e1d008440f7ca915e4ccf0dcae836e4ba1b02e4281f43b13744323e80ef6efa3a0fdf4050b026e55410f456ab37604496cedbc0d97ca1e84dc2ad2e66d18c6efd3963288a92bbb525d87c9abde5de41d777da86a1d47332bb52f528cb9243a137f58a19cb20c8dba324bb2546c0e0c4124afc2d12565b4c857e29c19a8e065b36ab3005a72137c5da2aaf379b4bab5e155f2acee315697433d4380e7372dddd1f2a9dcc81c16c59dff5dbf36fd9619d745fce857de7badd7d92b1dfc8b69273a38eaa227454a3000dfccaf9b4bb279550d8851759dec3de231d1517f5227b0687731c3d4504923d6db9c746e6b3525e48c8a4823852d372b6250744f9ab285264ee3f97dea5c7b643a6a52862a70c8a8d45253eccecd2065188968aea1dd4f5378db4118635ccecc48905e249ee78447b16b6a9f456f07d535e8a274d2729890bb2e319f62353fd3cf2cb32a090727e249ee7b225c39d9915127f5d9c967f074c70a65964b4ca991e41f12a5bacf3e1f2e60f13a714a725f6775970d13353285580239bd3d2b4d19f28271840b14fd448d490a34b2cf8bbd9575b4f5fcc69ea1b9bd63bb44b6a5d45c519f287bb9d6e6977779b35e9d8d78cf6453fb7f377b8f49a35c2f94dc2b9bbb09c1450a0ef2983b9a001c4437facff8d553dcae180518c2e8f68ba1c7d52962846ab1c288dce82d5b633edb257973a3d987fc7dada3cf5d5876857210eda589046ed20ecaae170360ef89bfc528a0b95bebf68d97017dcf91dc5541b450a27001e5e1b04bc05c0d6d646deb564f01bb78dd0e84989b3d59c6f6078f7398ec8201e0ee675c1e3bc270ff74130083f0175f9231dc29f3d9b42f601c8c0b13877a638c4d748ff6e2d9081d1020e6323fb47d16d8bb454e959004265a0e8caf10d20c5d80fb233806ede5fdbecdf19268d80f701ebf2d00ea7e289a614cdaf193384b4090315af0e0f0734873de13f894aeccf10715464c03f03a47c19f520df3f7943ce08aec15302227d06097a15737283ae33ca0bc6452c0629b5751081db907ad096eeba4b8d3440f5b0071c2c4621e584267e3c864888b72cea016a32a51e4f36c5bad681c32e018ddb3e79805899b3e759bfebdf835cb5437d4907639fe5a83525f43327e62ff1a9037804a08f5ea197f3e7e38b9cba58e67f3a94a182c3782beff8b8f159928fc114557700eadb346876b7247b4ffc8ee9d42716938885df0d2bab371f5a830920a4206ab949a1d423c4c8a1b8494bb1c83e766777edcdb2c39b4db6e4c1aaf6214f217eec2679803b6143aa15cf4f66f69565c7d5155badd519e775ab0023c3941eb5095f7ae13c562bc1a486d3588b470387e78ead01b856837c219ba9def6a32ff67bf449d8aa56c791f5648711e545250823acf93e175797751030262f6186bb7a4402932b4e83bc225e5bd49979a0ffffdf9af8e1e87a68570b707cc7d1fb9fb8d7eda801474d92efc21d269af6d9e2880414ec28973f4c23d6ac5ba7c922ef743a9aea6c7fa60cbb5794504d6b67f94b55c47a7e2b33fd83498eedcafd16da5cea63b2965c3208bec6a76ec2a48db181e146dd22f4a22871a12e32f66e85c5cf4a3c522fa65c9c5e6e8e760b7fa9d9a90bf804c529cc7d0379042a3cb35afb814b89716883159683d74a94be09546c39c2b5505f10d35f45ed8d9879f9742facdb28a6b0248d682797b87ec13b9e0b4e8a906efdb9312d706d9d7f97e14ba3fabd86585dc9dcb00d07306615c403d990dca0b65f76e97e69793971bb12c745d177d463146da114bac8b4bd8978c18211f0fc97749f794161e39fa5df2108c94996c4dcad8a3c943d7751614c171d42b1c503befa33c3146d34620926b261479a590a610ef31b398532e20726c0e71dfc50d1ce6c975a77e4200c8ba0f5ed267dc844c22816e58097184cc6501dae05f2e08844fb6717669f80c761c7d2f150572f088ebb62047a6e2d71d8c9e145d032e529770b9119931b4cc5a5585c37b8244b86ea828a15070848ac4399962b4cea6d23feadd67f5652df11d97788b8c59020b11d41da27b4a6685caa73ae4aacde3add1fceda0aa1ca6047d19232d766361b0dc463d1066c3823ec51bcb4421ed99ed1a39c7fb5d336f9c2f88165314e7b9112b99ff3b2b9d068a34a2b3779cd97f7e96f6de5d721df4650f9946383cd44c0720044b989e32c2b043c907edd5b62af5fee2026fa0f1f50aab226ab60f481eec90a7c3b438f47a3b728793d18284a2996940c4025c9644cd463605f10213f8d63f0a69c356399d914c7606eb173724bc3be1083963958835a6d4ba99c6c6c684ad2ac0b315a7de4c69c9262cf4407202633f2bfd768568defbad53fefa54badcd4f8bd9b7f8c3b0bf811d7abfc60c19e08b42862e8cee50ecc50717b67e6c8d016c1b4cf8bc009c2f1f597a71865492ca849b08f8169f28b79eb1f26a5abb68fc257229ee4981c18028290b962a09bc3191adb6053ec7f27d7f53652879fe964bd0360daca9d7f29b2b72178824c5db6eec4aa874b1ed122a0ecbb8de2fbcfc666afb93fc8a7151108d11eddc2da4c741461f64d1918670a3a05c0cc281acdbe7fba869e29fd43e3eec467a7c449ac2d63aa48e1e49a77e89baecf0c99879e52ce4285c086ebcdbef2abb340b4ac38d86c2d164c9e55af9540b715b7a9512108861b66d7504906b68c2738d0fb92ca45e288f84402eb307346dfe6b5079809a6a6870fd98ef746e91488dbab3e83ba25c87140f01d5593fdda7bbbb5d0476dd6483e0a7b66da32c6ab05ff4d18442fb6ba2c2137c6a0611c65a0b5abe08d7ca447fe344d356ae0354fb019ca79f92d350dd58b5faa5fb796d1e651c91af27791985d4c79e1c3203c070fd3714036ea7920a14b43bdf46f20db7050cc285be70b3302af46d7793b31abf703b530d50b7bc5c4e6d43b774d0108ffa5fc76d7856edb6727a6da37a803ed84ee4ef38047093caccd5b70f2ef4b7537eefbfde733065c43fc5adc6f74355c3d128778146133661c9965fb257259583755bca3282acb1efbac4f16da6234912085575659d52cd358c9f0b6bd0cf3b0b423401935bab7fb3abaf32ae5d746c5df15f7503c4ad178a275d89d1a652993463e6ec1427a82fd6f671711938bc7a0ac8492e0d6cc5200b95fd200f6ea64cbd6a394294f5699e385259307f7ed6102bf984a788daccc5b5c9da342ef857541f9f6c26b9993ca20927b4a8e66aa14f8940e5a261230613f77426421187d121ac845ae4fa16806f92fd86671dcee2e4e711c17879365c7c25bccb1377db58643328132f5718ca6e6a2bf063a0bc72e9fc4fcd0be549cbe830a0b0878b872abe2e1e2fb2db0813e8288e38ed9654c2ee89758448ed8df34296f18ec44cb4a80e5eb222aff1b8a4219f14cb316fee9df81bfc8870a87da40f92cf5e735f5efc3717b736fd39da6104f53d37d7e8bcde0cee5c61b89bbe5ae6c293d272cf03872cbb9c62d7f5facd661b3e98e998c8442df4232c1a622d947ccb8e9db859fa6dcd09ac8de9322dcd1428fe578baebedf069b797e7827e733d169739c9be8be91cee100280e7773506a226e3c98157a2e48f266f3776c919c4110544509dbdb75903af1fdf80a9a79c9c1514d93e48db19e6bde4586fc66b3111941bbf8d45a4da1251f5446760e7c8023ca75e51ae48f6999e57649b797bfb18b9df8b2981f5b13d83d5a554ee14ccc556254af95e8f146423670c4a2b29c466d83f7d68af49367640e2055a20de8621eb7c10aed007398ca62a4266c8097990947f889f586e5ee6320d823de989fecb4fb8ba5992a30f68debd500fd6444040827fa0d8de46e86bdf0916537ded84879884d65c0000c390dc879df3d7d59007f36f338f0939bbaf4727cea46b88beebe032163d41b65d57ef4fc4a8be2cc449695a48fea0211f05cfa827a5688f87bdcf08f4e4ce5e1ddbef9876c6dd3c31d17e38aada140516e8b558f59143501a5db5f0218e88a54c56d55f85e0d6f544689a372d5ec8c88b0b2a98b5012166260a6e02f7e7669d47218cf920fd80188b1e871dddd526ef64dcafbbe48ba48b7ae68f5b0f57a82208a101ee83ad0bb5ac14f205f7da6194bb1511f1d4cc53869bcb9c141155fa0f5bce57e4491d833d09acac63ac21f294b50c713273f9355725fc6614477079fa87a8d941d2cc43347f8a96e1d4dfc5ae6f608ab77a7dce110e66c305551212954e87921c5fcb9d129708adfccfe4f50d280cfde6d530b78308796da1cc9fcfd2c609e4568b1f9b2ec759c1f8b91f345b774a40838806a295886297ee1d76476ea680f858528c901fad178a1e80808b3b533f3d02520d7997ece8df5bfb48bb60e32e01a59a01d9d8910c41488980ea80fd12696fbeb742bd0a0082bdb2a9b55667acc3083a433fce84c25f499f5716d7e93dce4dbecc863f0c8487aeb377a56b409b1ffa8efd37b5886c15a2fc4622ac14c917c761cf2bd2490b1d2d66daacb8ce2b620071bdad3811c8ba92276e3f31f3c5052ff8f878b6632ec84b279a134767f15c3958dfbddc39e1d53e193de05cabde650fc115cd145315b0ff539e81422033a04808e39dbb3f23555be6c00d453d8b6e20ec78e08a82e588de80cc77f81d89586ac1c2968389f155af0a1025d51fa6da7d667219bd99e2edd413179911fbc521b16aa0afab035d3df5d56a08d60a314e7c1b2ccb73472ec8dcdd279cc752955bbdbd3bbe8e620d38f77d1cb2d6abd4cd54e828291e29ef20ff624a8042dd3709d464cb2180b1ff7a619149f0d86484fa8f435eb5cb353cb47f57b94f9a24e8cef25f1529958b30478f5aad90c5491d8aa56f5fcdf8c95a29118935a40c24f6ca08194eec9d58dbac9471e7b9903f167d2dd387d6b16b46bb6d50b0fb70154c5185de81dfbd75ef79d71e9b7a4317ca42a7ed603b8494a928414498663f9db14d6af2077749fbfcd6672c68f8b4f56145b87ef8556f176b069e9ec2e18073143e1b75552536100e5ab75644c48a0b1256e1297f18ab6c1083efb3f42ef93b3f12900b411b3925a3c3ebde516c62f8bd74fd9c1b31f64459ff669098128fdc52a856a8e6f224c7604f4e59e0083cf22c149081c6aa1774fd74def6dc1029969ae67dc97f5ccf263390b8a99b30d381a9399edfad546968e1cc6df23bcf5c4ced15e726d9d86caf96c354a32362bb46631c319a4abd9ef71eaa5cc9de575179ffd013edfa1d5501d3d72c7d0f6da655e984dee9a90b40276fee04e05a57e0fcceeff3be1524ddc9c8133d61c9e87a643928ceb4dd6f95005e519f8ab399b0c4fd00c25fc7e393f692e5ed0f4eab1b83d271bc57f5859a08715f5f608701d62ad2fbd4dbe9073b52a85d841d508ddd50fa895cc2b2f16e689869ee2d140e0f0e0b2dfdbde448106d72ab3bc29bfcc41b93b50e7ed2fa296126c1fdb5a4439cc660e534d696ce9f3d71a353be95ffcfcf50a98c34ace9b6b9d4c5987c1227d88df434057875460e36cc7ca69b9f4ecf4681f745fa067353556daa8e06d315be93f8708fcc82b4c1e42e539bc1dd406350c4ab656484821f22ad7b05c4f56ca0316dc31bbe63a91060ead10f209181a25059dacd6677b10c1a5497d87acdba786dbbdd740b5e28384c35c58135eed044e51d06beedb1081c997e64796ba56c0eef39c8d8c5a2dd44970f43c7225e9db0ffb3a876df289e7bcc2abd2027a3148c86ad8cc0c58d66f19be8df92a85e821f3d83f0b97320748d11b1577434149729289946c1b037fc402ba0516ca63ca61b5bafbb037192132f101c0c121f6f843e9e3ba95bb7a20e439db8b4835e99413c9806f8d276da098dd4e0920d0c47391d2f3d17995f89630ff6b669c30eee1a3ab9ceb7c503b074db6c00577e863dde7f2977144c0b7ddd4de02cb95648f4be939243ece44286de438677efff51fba65d4997dd8a95ebecb98bbe56635c46668c9aa57bb1486596964e1ffd3b4590c1217737594fb9846a50ea7aaf34a01740c0de0a09258b0e734bdd05db7cc2d404e33f3ab43124b88dc606dbc12df7de4eaf21e614ecb0bb35c208644ec900c8cae77db943f713d124d61f654a153559f88e48d9b8946557d8723e3c3331e6a79f0a75ede9a6b0b9edcc9f50171ea62445a76e043d82c98128bec41f7055b5bb830815d8c7e6d2dc2e3ef5515d89bda1cc0abae8bb77024b643d65e2221d6496c8179eef60a608939deeecfee8bf889e83ba6b4ad77b2292843a5f1189912e83548af8b0a4a725e013cb0b7292b9fb84e1e5fc65d26d4d8526d8c72d457e71d1d1cd535fe3af4ea91b868932114f391d814998af11caed48b1d6ac630c45adba66ab0a4d85d4e5bef754d54c16af564c01f00c8f9f1b5dd1f72127294bdc0909e10b798aedd728d573a8f52f84f4d360b057db7386f90192271803dffd0acb137c4d22f7d59ddbf6adb068cd5f2dd088d898381f867d43a3acee55356009e3c1b8d1f7ea6de6c1af045195ae822ff58505994166b7b7fe78d6745b4888347b153fe143c63e5ed37766bf0a714b89604f4ff35ff972e364b462adc0335085de69cfc8033d23e2d950b174251af88d2c65c745e603621fd621793a604f8045ccedac31d8b8c945e2f9611fece860ca506e1e813acd8f68e57d37d35cecef3fe1e8b752e2cafd017cfce0bcb3be18eb8be845fb0f36d0284259e5e17ee43efdb3b68c868c4e2b8b710817e658f88257ff75ec44331fa0170c0102d5e827f2d5100c4cf84d771e0f210bcf567f5b77b8aebeec69b449aefa6133dcd4cf2394ce227e77f368a36015cc9e22414bcd54da5a615689ca08b8685289f590c6899a098fb26c6a7fd6fd188a6f2e6b2e40ebc3cdb72a23ea71e5389bb12b2a326ab06018b647fd833a112d8b4e2b6b87eff0eddd42f4a7dfff71e22d6c58e6d36c6db5bd6c9c9c4ba95785749e1b2d2999d4484878accc6e062fd5431db2e13ba7c564d12dd72870fbf679e36207da8bdd66d8c240071ffda7d6fff5400f6b4474e5ba4ecd806ac9bf24d5f1904529944bf80a15c5a5c3063ecdd722480fd9501490f35ebc742391f1fe14079743b1cf5e71c4242b8c3715652c28241763cd43c7eebcd8bc0d7da22fef60541a21ff55d7646b20597dded3ac03229637c86f19deea18970df22fa0dcb0017b3963bcea10f377c2f8a71eb1d0fb5bdabf3d8e7a4facf41e5f7f477954693ae6080a2f5c721244835557851139efca7fa5a88dba050673536caa59fbc8a1d5bf2d2fd5b4863c47f37a369352dd4a330b88d0683df6da382f2084545a1c60cb62d5df50c881451cdd36fe524aa6db8e537ab2628488f51c04eef490eed19cf8fff71e9a1941a3b4a7fb97edae34be9e1927c65f644b1a42260a7c03a45a938748e45907293d27aae8c04bcd11e569785695f440fb1bc8a1f70c89720b4bc8ef90c013c553a2eaa54f3f597c29f8a3101b5a6d157e8382b8c64eadf438f9d4a2263e8a04ce02792ca7ac591e7c69a94cd7586fabf2841747148e34e6a0dd166bcfc3713f982bde932fe4e13c79ec5757e80a9bf9764359c659ed2f20482ebfaca08924196d746236dd0d0fd1a09468cba3be795c4377518909df2d1881e4da736f9a382e10fbb03c131f96fefb6f39e17d548f3c72df232c09089e505c25bb3d044d7580e0baedf85416dfe4a8aef94944abea0d4d687c4e1761373b02abe61fc8db19e2edf560c5922f8e3cd418f2f582d3025356b2d9f48686fc260b5a725e1fb4ad838ba5865503a0de03c962eace7679f7b777d6af270a5a89e2a3b7cd78308d805d0c1ef39cfbe093498c99c8df0ee09176d9ff33fbf4f046771fe89bdd25a828d58f8de87cd09242c4889b42b1e9c9a5eef72e2612f382e44e79090209bea52b382fa0e591aede64b51af3916a9dce1140d0853b30b6bded30d2760bd73de886539514e48a5e68f3f1dc15dd46edbb1c773812a7c5da61a66ec47ea42e7383abdf7ac4d41115b57bb92f82afac3e4972848792b11bf3d1964337e77613e78d5c5d79ea9e036968b7cf7f7ac4594c603289761b24da6a3bdb088abb2f68059a1477bdbd51adeba223332673c72e81618e15f025a025e9b4847352d0313819ed037ad011b81d649c13a413e86b693760f9f7830cddcbcbd7dd13c9d7adf603229aac37270b4c3f4cb4b48de4ec9a23a2819480dfe988f8a33822155ed0c6838367f2670e3715f79d63075ca571a3431e15f0bdce3db1247f5eefa49e3cd63b140fe20ce7f9c895e346a392adbf44c9aa43dc7ee361816a4fe380980f88a70d3ae4cccd2a86f0129e83c9b478acd6631bb0494db5a5f4b760db96f0c958451209083132043683bb15119a98f6736d0b3fd3099906bc90fe3be9915345112ed959463cfda39c96236b412ca8c960f49b2075b0886234cd23ae1d62f22a142afae8f0d56c72b6d679b1a9105a6a09843f0ceed93f2f18686d766b92799e4bddedbea236e611d84e29c6d39e9cf0b2836157cd0469b9f681578631d921aaf0bee8bd42a514ea1e1041bbfd8cf5f4bd5c216b93dd2b8bb074cf3bad3ea9b148f6f5505d0445d63e5759f00ada6622830ccf5726c7201fd324bb7fc74c1cbacf32a3527aa98188098d53d868854fca7257a923506cf7f6e377c3c635d9d119f0cbe1262585e97a2e63ce1f05e4c13f98d191ff3a35f6eccaa2da57f3645560fabfa379adaba9898035ab97218d16d848368155def703e6c9d16d797ff9c43fed1141e765cc58c5af3c34d9fb35e010a626d7fa28e728f8dbad57e1f13e566771570e88dca7e39b623c21649d54be331c45d754eeaefc5614d648f16c2d0797bca3d42d4d172e237731f72a1e27c00796382f295fa23196dd581f149d0ccfdcefa7807f3cfc57d070a9181fc2257ef3c64ec3ef8be52a8b4845d3a1420c6ecb3193f78e08e9ec50a8681a3ee99bad9966c8f1fddf092970ade85cea3e3c341ddb0a0f840b3b9408db4a59ed7c65704f7c73108afa4e8bbf3724e68f5256525201618c0e7d4ca4d2c2d336c04266894dee5dce37e992a0bc4ea2656aca856b29715ac98c5d750d8ee3e3daef9bc609a6207a7376beebf201a49a615d3a58bd07d49bd3735eb1328567bc5835dcec3cf2df659efb0f6abf9c8b67d482d5f9b89451693e605071ee9efb213f61b200b38ef74cec40f78b3f4ce45985121ccc6449ba435d9e3c292072a7f97b43c5d6d4fe182181193353c92d912bedb0a880303101fdf5094dd537a93a1298ee68791025f4c245c65097654ae69852749bfbe9be72eb388816935de9dd16870868e01ba3d8b42ace35782debd8b79f6dc45b721216e69712f987a73f22538363b8b887b3e3c877415382fedfd8e2cdb4a5f5615b797267e12471d72dd63bf6f113dfff88b6ab28ad13886a22e326dd83d543843d238295b5d3be035805d25dde617db840aaa2cc04d864bd2932ce3dde308e044acbba80d491eca1c872fad0012173f78441a510fd25f99f55a70b77af9513c5adf0a06b554abcad61eed906b01f8961a81d7283e5f1fc7fe7c98557e03f642e06993bc97441a7a2e1836670e5dbb4cb162debc627569f6241169517cb9624e3f638969f8c5343e874f2de8d2cbf2deaa5b24544ba6d7ff32cbf396a3c3c6194dfbed1a56231f9f6c0a771e7514a6edeae8bf1d542a5626c4c6b865a1db068c54fd793f7653ed91386470def506a2d1ccf1648436020d8abf7bcda1649fb1199f01683c8fb227e28647d279a40b95e17dc9df5958ca4ff20c6b937338a94eec9655e2d6bdb6cd7a958ea01e3f914ea7c260619ecd5f93b9ef4061ada58bbdb3c9d862bccb652d941285c2219a2bffcca57ec29f85865aab1795987f75be1531c9a0579b7d4dfe6652d1fcd5d3ec57d6ad9b634d4496b017d5f4aa320382ec66071bbea69ef1e80a618b888f4729e1e1b1a13742715f3213e94cebd1bbcbdfae780d49eeda6e10d5deef17b595cb33178af0394afcd0a622227cce466d29f21f04220838658dbb8df35f941d97d6728d7179e0252f9c19370a5dec1018ec92ba36ede5574f0b1980da49187057c7a686265f3c8cd2dd2f19ac2deb5b391a1f7bc3ffa2ff07255cc62c21924dfd301e5b92859074a98f80ee71096fc2626b6e8e2f0c9cfc1391b05a69d0cf27ccbc447313c27325578cbb57427134b9742783d08f6c955371c794390470745e624752b83d760d5d95c7c8ac19089311199bb375b86a5bfd5847dbf1724e4406f735b29f8aded4bdf1569a2b3dfe90d8f48340897ccf149211fbbc8aaf4cd20f20e0fa92893bc1ebef7ff92a5920eafc517dcd003ac54af78179d2a8811a7189a41d071f1783b10cb3084567050c3ea2676a192d86ece19a837c13b41e2a3cb336238165d3c59e1e2c5d7fb666724c2371f2e52aac2268fa1cfaf076f23e1bd6a719dbf9fb2e93089be55159d5e8346a5026c806fedb95577430fbb165c54ccd2e26934a5be25a32261ccf36054de57307ea53bd5b8438cd1877d3be80ed0812a91e86c991c718d06f8144d76706bceda1ae69899d185f8b4fbc7e10e3acc63250ffd7a86a78289270ae53d58dda611f0e646d444e7845645effe82736e67551d434b57028180de191b33ca8c9f88cd68c152c2a040b06e44d626410d734985430e9d6d511b079842966c89f81cce686ef7e3d87bd3283550576239914f05f64bbc839d10069bb65fe202aafb2ba343d42a2d675672eb4d77b556afd973c5b0b35705d69adab4fd8b01f2c26caf0bdb05c44d576928c6eb44ae3d5d29ee0e2f7132f0a37c48934f5ac3aa5bcdf8176920630e146f3a8e04f4a9e9c1672da60ee7953c47f1748337da85bffe5823aaca62dd53ce71f67dcc15ad26136f70fc8d24249c9097e678fe859d3c9f39e72302cd2c44f8a199486fd56dc13be5490ab9d8875ae77c7e4ce864b663609e11f5c64372fd29967fb26df05c4ab1cfedd765da17c40fa4a56bb25c2e898c739af4957afd501d549b3756b821c99385fb20f520a931832546ac1f30eae5ca8725bac408626e4f8473d25a9264ece8409476163fd5d86cfed9142c711c80854c29f2133d62cb63dbf1bb93e6a31183d00307284c60e3ddfade9f0bf058380d0a3a1dad848b1c4c04773261aa4b0763b1f8c5e7b102461f06f9c3aa5cc5372b7addbc4c7c5947fd72c07770a9558150272604e8da44483465f3e220fede6ef935fdff2f29b670e9dad6eaf7a1a7bc27acdea77a70b68dd5736592b81a5f3e50866502696890ba98a0db0b0d45c6f6ce84fde8d3bfd67d1d9f9a795097467012348bb7143e2bf9a8202c71986e3c9d9aee80be13428eee66e29d249e91147a9d5f465545132108cfce89c59cd64ec55eb6e59920bb556760221f384649ff364db26d8313924c271e2b35b095cb30764d391cf496259f0ecffdf296612d9938a8e78175f140aa8e372526d60be3f968d1e4e6fd810131a7069cfeed048705a527c75ed22c5936241c10cefe41f0613bc41a3b4edfedafdc6fddfacca3d8be585c7f7d7e588a1afe88ecb0f8b01158b665dd5bb70752613fb852edc68c47935adc637e0dcc77f5165a1d25321fdcd51f96c0cb12b900856af7a66c96261278160c1d562fea1c7ac9aba7163cc53f1a98cbaaa7a9e3cb414942d91f42090f398f0b75d126b4da3e84dd2e35505da5ff180f2e0167d3ab8f76d807798ca163df5b1890d5d0ea841aa0eb62738da8167d963e6fc3648fc44a786a1e4f45c70fa8660b5a328ee516e55469d0eff06355066f43b1ef71d5db7c2842faa5b809caa0b574cdf1e1c8d77c2938fc07869b5902b1845d19c13fbf5052a5e851bb86dd50702935cdb2cffb08557e325e4c58a9cdfc73505db53e583e3e39e134af66ca9f7fa375551d856735ab5d21b8901ccd580e783aa3449371b883f599ea7a693ab4e42ee10ec3a5c7f7ef7be311c909eebc927ab4bc96f0595d49db173bb9e592be71bb6fa54c94c5111afa80d4c9c6facce5b85ea61e5af692349df7c42f6fc7b275e4e521fad0329bd7082fca239c0bcb92414b05541c5ce2c5f5a4d533800d104acb4d4895fa3048f4551cc61569d9a7211cf1209fa72d008a3af9332e92d4cc4e8b558fe3137678bf0265081702ed7ed30d69fa76d97b98234632f93a6ed5c25cc1d9f9bd855908cb27cccf2d53e245e6300e0d611bc5e4c679aeee1a6560e8dd67fb279b077a5f32c8cfb9ad6e76fca18ed184bd418e6a5c6d7e5e92c8c88061e35d5796a2a85436ff8f976cf0d7fa24c231ad7dc9c8ec1b83e0d6a54a421e5b5e9e9f8b8871487ef500a08ea2f7d1fae5ccbb44117020b5d0b547b63b4b433c8ce33bf38f022a903e89bac2f90cfb8c56f1065e34953aabb6b69b19a644c3e7e126113d1b4301253c401c0e2109f7a15492d43d6892c3a056f972f34b2fda626bd23590870306272886b73c29bd277d814e56c32566009faac198747e151f9fef48d959038b4b22e394f3c757c23d2b52a606cf5a6baca4bf4d677f511a5093a60ac70169a148ecbd468967e66747c2158f77c0b0fcd045f9e8a4a79bdc91602bc8459d078d9a070d7def334dd109f5c1a7a474b5f5724494fa38bb7df82461a98a914f4686e6bad5d44247850dc7a28d0d3dcdb06bef0556ef68354fd750ccd8a125ad059dc5b52ed91f10526c4fa410bb359837b9aedd8f275e75b55736b948455ba665d914e3d0e6e2f26e1cdce925bfa5847f66895a35ca12178912f6aeca48d60429342910b94dfdff3a7e2e283203929097945113d82a4094a0bbd6a765d1ae849e3a09b94e4bcc6a76c2f27d95e9c4e23c4960b8f9f11ea440ddabc0175c01bb8575ad8416b86d89b57364dde43ce90d954fa596f81735f248ebd1fa29912bb7354d673a355ed42019e755f1fb4b772ca49f545052ec5ba934ad4f2ad4e83ff2311eccbd2705afd3c80206b4dfb0f5c37369541f82ed9e6646caa322847d86cb930efbeaa866f9ac5f5223ea61db49b7af8d793c407fd271d5631fd923c53406bd9c2be3e657b7e7eab1096b5ce5fa664384c7b4099df390a3df267443953f8d6f24dce5e730ee5170542f44e9e8dc3b4d0eed904ea022335165bf287b41435ec998208682dea710f8071f0ee43312208c8994150e8192c3a0d2fadadc561ecce4868954a915eaf19f5df487de6fb2981a0d28b8632d03df0bb3b17968b88f6c8a5cc999396e9b3b95b527d3de25f1c681cf06cfdfadafd945bf59896ef4d5e9662aadc9bffe981b82b155c3fb9caf909203b03f3ecfd6b3f8a73f1f3fd76a3eae7515511ece17313c384bcbe509e571affb8218bb038506f92f0418cb83d87a8c50ccaf0c3581d3dce64cf4915c11be1aec525555e1955aa35160dce0aad07fa1eb8ee0e8d7d1fcc04de79cc7d08486b79485c93e9543ccbbda9106c47508ec268ccf50db78ac85666b129455576f3e36c601fd0592b9eba50cc4588a13a575ad5de88b9d532a6c3c5e0d89ff60a5df5e75c8f89a40b7242404d7ee17a0d0c3ad9fa2307c254a2515f121c68542786b438cd5ca7f8d5d5fa0b3e1f5590da989cf0fd75ddc12e9e2c51a1017c434336f6a958fa9253169b494108ea8ce6db16dbb114cc65228acee356306cc5603a7caf49ffbb662fec375e34b46eb6ea6531b0f7c512cb708a674fe2318370dea1e191e0b81a5a6766e068a6046c082608269ccd31f301d5e69e49d9d3134cfc1e9f83766bb7680bbf1fd7c835c4e7cd8ac62da517bcc65c1d4c2d09c82ee1a1f15073f0846194ed037f4b5d7e903622ad69992d3e625ee54c815506784b3ca757f3b49f152e15f976fc84a1f6efed818651be7fdab6a20168f354ecef2d716d48b4447dc58dac8793df50f26f526747940ba4881bb9b902d351fb39b49f7ccbde811dede1a32090b268b36e1ae883b90ee55345ce163a5930366b4bc142ea1d6941131220ba7dcd992d87e1737da585f12632d7f9a3c09f62a324e13aff0671ce774485b6edc28a810e3108490558fed236fc6621f3c6a675c4fe6a9abb008f13d3e8c38327837866f1fee5bc08848472baf1b49a21c22564ba77a3f75de1a28dfc1ba7eb443788b9dee65d869ec795e757beeb101df109bddc12dbe20b5ab38893b59f49180e23bb74eb342299040fc8b5fac85f7b8f223bc3077648553a88cd15dfb4a88ca37a49116e15633935711a8212894e6241f3f20435e396cb4882fde47d059bf70eb44af4e068be6473f6183c067dfe40f30ceeb612126e3981eb17a69ca69dc17ed6f946eaf77359e45e2bfd60f9dc7862cddae45488bebdae7db689a371196effb7244f34806bbd647418a9a23958f9166a9e6ac44a2627a7874cddde84719ebb99aa0519acff9a2630ab5bf69e94c90ec64a1415d1066b276d77d2c7d9ebc5e140a63969c52ed7412702c66fb0c504d647dec6a42f83665abc8f4e5adcd958910d3108e99f82b7e7b1a90f92354bfe97b3f9f921cefd765153bc3092904065d6bed1a1d2523e8879952a1ad2720d7494a8ba5f95721d4e56022eb4d24295dea9cc7dd81d3a63ed512d05b7d1eb8dc619c999fa61ddbdd529b9ef3545a255a8068fde6c09349748d711aa14caedd974fddc3ac91a9d5da6b059dc1d109daeece6dfa0ca28fe5952782aa5e2643749f116935495862b42b1fcc49ac5b56c6abf21c06fce33bd8b0aa17bd41d3ae6fe55f84c1319de776ec1efa0bbf36d2797d4b5e033c2f639e251d9f5a16e0ac546a8ebe78e35f7eadb9b5deafda0ff237ed703e43cf0543a41d866a0d3204e67207b112854e418ca268fad81ddca626f0696807d85049a47a98b847461c53a62e6312110a7eb615c8e806ef41d9e9a2319c212ca59c47af362ae896aa5f5b23579bf56e66f2b9195e55aa17b41b611e980f34939f0b388b65c7a926ca29d6957cb734c3381fb41b1da50159fc5a4c0f79726e153e19c1fef61179d70ec6c8a947aa4eacea83062616f7ec19a43cec14eb9da757596bf4195ccd740f27fd41c44ec0eaa758fb1e7167f4d729f978c37f4492bc4e3f0d331ea82ee41948df4042c7f5c15bd5df3762a25ba0b6e812917427f3801f51f4279231c80fff70e92150a2d51a60e74ff8e997e91161adb8226aa4014891e997d601bc2e9def81247447a811afccc6f1def990b29b178d7953ec623730d8c281c26ad334a548c786d8a428a1367c2163c167255daaa90b150e6969ea03297523aed482ee71337f888fab1b680e9f0ac1c66473767d413b1bd9416484d61038ec79374152de68e370a190aacd82c99ae9f485a7117f7836ade1ce279ac2af32ecdf17e183d46642c9eae54a40f0a503d57a1c0020bfc1008524b8f7d06248525ff079b3792b6056698061cbdb9670883a90001b0e5ff6004cc9e25ff2471e1d111d43c4aa14edebd1a26b40430fd53d236ea16cf3e673d593d1bc96cf82be7ac9485ffb757f249e2a4c764cbb2e374f24a03ddeb7b9649f74e5a0782967cbb83b362ec87d320744faf9487a8c04a6a732a1314ee7e0dfa3af671a7eb0c3ec6e2be35ed467798c5e29f3b68b1b77061b540b6159eb15992f39901339977984dcd92c1e7a39bcb349c97984f2a95f0b088156fcdc51832639cfbb106564bb579e8797e227f1ea803867b919c80edcffd0c7da29a718d80fd865d5d4d35d18254baf2a82f9d00ea4cc97127aca9e9e26b5df973e14f4e2462307613df6055d914640942243b153926500e199ee8f0d83af18659a4bc092092c04e616f5ef07ca26ee14a6f7fcc409f7330072dbb6c6130ffd405880d1d58d91b7eab8f6677c711fea0b63c1fc6b3ef0fddd1e8b6ed9f5b507e2e68f69bf092cf1a68088e9cc42488c1bb676adb6ecdfcde460ceab348050a264fa0fd02fd933e0bf28190cf4407228d01beccefc3d00c118317e50d6dd31909db140ac10acf9ff36a52a57ca24177b8dbf05e3b1587278ca9aac18b5336ec31be59e06d4d6fed4c9f45cf8c8f6fd006a2d68f774320449ded48e9a75e0127d5558c8de7ce316a2c20564d48fdfbe1faf96bb8d011088a83035bdb28086d3a911d217e72cee20fed62d9c622ebd7804d552aacc97176f9cb607ff46e6a326ef54ad6b717dbb764fff1e960302b25bd5f16cddb3bce0c37b82adb63750ccb694502b9de9950484cfddca3f64433ca5de5ef93b7ba962edae3fa48d78ca37ec21fee0dbc38f8563d44e28669166e7a1e4b80bbe278fc40f5a9acc5f07072045c64e1c1dc5b148b93231115e3414654024cf91bbfb7936100a98e579ebb8ada9e40e563c8012b58accc6e543ef15b7eb47a35f63f26c1fbbd55ca5af1f65f7423865de1fe15b188ded8bf738684fe8216b84db30b6e4a5c29f127095e45ef9d2d307f08bdbbe79ad4942dad7c00e97fcfa00321733955c5a2df3df8fb1ee29f2b3f3697c97a552433fc3b39ee04fbfdb43ff264b36d60f67e2a1758f69fe054e400480a26acdbb84467752f3eb8edbd8cfa49aef31b0c327b0c69211c8f7e1ea43eca60aa921152da5c6f25432920e429d6885168eb022a051240889314928d6ac22d5ffdcca2a9e51afc5f9ba11371deb1b0931e82a3ebe017f7d9d9fdc35ebd9dc23c22608e7ad5deb6f3c17ba1729c54ecaaff8bdd30c7dcab7ca59b07367b267ea4bd5d225c8c7cedd26d4322f48dba0cfd166469f2eb57c0c421b5d7b7fc060132408a94d74ce7c5a29e6a7d7b213264e5277b2199c7b773abb4909d4f0125ac364d8ddcac99d2182bb2c3c3acc48c3f63df3421328df8a5bb1c8e41ce714918825f574b5c4da79f495486f7bd35900f0839b762f27936d35374d293d75c22f602f04d1b663a70bb344631d8fe66b64794152abfd18f522346dee0e998d4a5c1bae985a388b739a0e493821bf68276097c2c3288ab2fb3d9ef667643451e076b98bc30088005cf6863dca69d9b67f61c3c9dac2c2cfca9d2834cde9a76b5bfae7471485bffcc5d5b26c4db27e95e4db292d2f1f5fe0c502d723850a5232e40a5496eb8e6a9f35fc97932246c517053c8c0d2294c5ddfe70f566f323122c76478d3bee92ee740274d47e034df8884cdcad44e2ee5d7b12977aac79cf285d523d59736e47f0cdbf7a235b6fb3ddb8a0388476572dcfe92ed565efe91aced6fd7b05132a1e44767082b0b1cea08e91a075988bf6a44da0b381e4bcc6a956dcefc31e218f964078640fcfa63d57853afc3f07808fc436d3b81b5a384ed0e793e0da87e0a3dc5eace6b4f4aa04dba110470bf50007f7c359e6d1d46662d62dae2f845fa7e14c1f84b637673155679b91839578af4e476a1b6c23f3babdcda25c6cf77c7e94aa8b6a714270a9cc39b2447a259930b1b8daaf0b7c6a84f119e21ff57968610754541296c1b60c04a52c7f42a310a9aa447079666680e07a7106001b6b0c8423d0486aa70a580426a390d6c8407d5057576980d8ccb6c12dfa2adeb8ce8104870d3dbf766acb3a856ce30aea51b86b080e136d4336800cdd09fc1fa89386d0be30bf5cd87bf8b943417fe98ba8e858485d4eefaa3dcf51fd4c0c8750bf8ef5d41d1ac266713d8ed1cfc0606c0f5acf7506279589e6645ec88344cb5a7273fd019bb150d3fd80ea50c72663ba1918f77d2247d55dfdea070079a5eee19d498d61942265f9173302f9b7e9e2e635565f3e4d979e79ea6257573222c97220273b326141191b38ccf194ee3cd7930a3d3ad89ee93694105e1cab50597b248ed81f960cb2b7590b06ca30efe0ccee520b248a69efe5d00d1efd76ba5f0ba951a863e089002237f8c3c124961e7a16ae87e8ac98728ec50f872251ea73527030c1d8502719f94fac0730b10b323ef91e161709ead22d6fa0f8a5f6debf696a199695d793483e5cefec37cb2886477a66d990b2e696a7da16d0728328306766094a791c6b36c9e19128f1fd119885cc91476051746e204d86c5f33ccd66c0433668049d28f709fcc4918138ca67e984c72cf377734dc742447e5a05fa225e7c290509c928e0615fd632e90ca48a71fae51706979eee0e73f16bf70f7eb3b6559d07120f3400fd7f5f256a3c89277f3aae9836bc3bd0386e8ec462a7ab50a65751cbedd96af08a9e1e0dec3fb899977ccfea7df3e769e6a640ab24ac1da095ea2364c76e7e330c6788eafa8804dbdd030f18f80195b918cb8c7923770978f8850823c967f01fdbcc7cbaede491942b6ef0e0f05f47046c4186629cde0e2ff1e24215cb7f59f85a6aedd7a89813560de13c71126f49cf492f163c64953bab8f6ad927f4b8d54eef7d537a6e734735996b59309c5c4f190523dc1021f0a07db5978923cc9b4a8ba18810b6fedb737b360852f87d8d1bbdc3c33c22b15e88869ea7932f866c6a62807e1bed9d8f2ad090204aff97e17c958d381b7b743da1b75fa684965d69a278b924571a68f6760c6006c7529b059aa0b9da72b69f0666b41a8ef2e9df46998e6b5c23bd6a4e3929e7f89dbb3d63e985371dcbed949449a3966a0ecfec915980af345752ad9e280cec84f5ad7b03089f398601bbbddbad14dc9ff710d7c631521399d52d42aea9bd1a17b36e787a6c227f07103aab8ad6f80e7f5bccad0960be1f8f8136c851190f4c9814fd50af54bdab51f533354d23d3ab5ac0854146bc63d36584463656eb2c0eace73aa663adc026576d9a494690bca78fb2169122a5f437e5aa3936b661456cec40988d793ffc6c2e2419dd8b37ffed919c2434db1c1e0047035febb53d33ba19692574e74d07439aa60e36a886cd93adbcb81b83ca7f91ffb602a224165bf00abe14459522667b0380b4e260ff1ac4f9a2bffbc3c7f4d7fef4fffbd7ac3d84f2bdc5a97cbab877d167d0309223ce917a1d4acb5c68c83e4d7a9ef90d9da85dc173bc8d8b0acb6dbc7fad76f1f3db03d04b1c74b09fbf294fe2b04b94b346864f040f6d10b47c583167ecfdf343dd54e3369d6faa41b6c6c0a5cacc75be9b05ca85e9b6a44b840037fe279a0654f781c9cf2ce969a03e359445a188dec90b545d082fa53c32657e9c246ddde059b38ade9b821df6a11410247fc9d47ad0530cb84ba45d275e9ade0bd47a4d84ff5812cc4f60c93a28dc400f85e80d340be55f655e2fbb21edaecbbc45dfbb485b6d55bca6f8db55721aea2471689136d32bf57526971b52265a3cf30c20a0a606a357c28cc8b69f3df930faa2589d1d771b065431b45deaabe17ee30b5901ea4f8fedb9cb088b562bcc9f5b4151e5dc2b0a09a7ff53d4f82d01757adee2c04f0b44492010ac854325d672aa1b8cc32b2e3e20106bb26291e021c5a8d14fe64e020fe8e0abf9bd85d071dec6ba2fc52b9fd041eb4982ce985d89f1418859e5d1a943094fb7ffa2f3dd76758611150d12610ffd8f2c78f6b8e69a7e8dbfed5584f3d0f04d419757ddf0ebea762c1b365caa2ec35f24a2aab6973fe2359fce66b3912d11f31fd08d8dd58af298416762f06286ade4614ec72fc30ad47247bf9dd0d259d5c2ad907b3986832fadb4e77bc12c583ae1f04f3f97d972b05bf6ce03ba92d376e43c77b3386c18c22bec69ae800214a4ebb32ae134374c402fb564dac74c5a48c2e1e692bfdf7b6538bb2847d28ca515039993079db667a3b006cac64713f633c12f4708a372c695b3882dd86a65aa9d5e3ec67538ab4b90fddc37253937b0f99f2a6ec4fc10355f6af5847de9f546aadc2a8f468c0128b9313889bd510140cdc89135db108476ad9172040c3f8b7858925abc36e63b22c13e5a689a555c5b47f7abdad8699e3f1c47dacc088b5dc7887545968473911c3614dd4867410fe1f8c47e31630a4e144b4f82fe3e4764e8ee745749c805a7b33cb811c3d0add925ca6288807c858949cc925251f067d565a77b4d78adf51a473ffe7824898d519555a51d402b32ef4c8e301ed6651ada8d72835c7f68c728db268cf609fc17ba7faaa9d7f383534248e6134ee4b255f9a69a1451ece054567ec3a16bcb7b03d8bd2dce1f5ea5a30553d55522effb037beeb62885fa5adcfdb61fa1653773a215214c0f8bcca786b77f9deaf0e39a525e7d8df52a11c280f44d8f030af666c6799dc316d8f16d82549e87e05dd896bede7d1da654f5bc08ad18558d0d75ebed71def2d43a597f375efbd183e3de120e600123cc3024713d881b3cf70a9948b996556559eff2c5d02304b1afd547de7adda218b7330bfd7eaa84e55746d5e3550caea313240d35f07d972e2ef0a1c2ab33d5ec98cf5d0be1e705d73d4a4e0b835e4d26bd4cb6fb9cab07b3766c65b1bcdf5ee9a2b8f747d1fdfa01d250d454250fae2cdd65be152abfdd10b988bc30b79685b1ab7f5a96b8b9e15f357a3740103ef809305d07072874d3c44521e097c918c7ef9aeed5fa1abe9252e949e4c6db562e3d8d126a2d6a1e0e765e7cda645d5ab48ab33c9989efb4ab3dfaac67f11800c4ac3d9d840d19101653d8b3cf31536b8021cf3413edb70ae1e93f3c1ecc47fbc4e04cf1b49170edaa04c23fb970f7ab21abebb9fe93220a76f5a8a4d0d758a635afe0fdce73d01b82e7337bfd182e317f99cfbcf2e624b6c02ff3a25dc45f957563b49a92461adbf50e46a4985f7cbe864e13b13d5138f6a1df59d5fa6275dd3b4ec037d93e18644644c2b5f5f9cd3580a7e81bc473355ac386744fe2fd886b958054a1cebc833cf199862fbd69039c329c566b8be2d42fddf2d9681180aa1bc0e44e49408244e9cc16e8df51ad70ff41b76d85d6cd4bdf560ebbb02f2f6026c07e787bbfeca2ae20b6d63d41bb33064b7d2b80e332448f1ebeab260618e31dcd78373f06766f46578b486ad58f17585a9045c327b86201df4ad1b6945e10c6d22838dd18cc1ee411077be7e10670fe53cd7857cbf547b3d6e0421add6216f39b5348f5050264087d57c4493a6f3537ef1bf58951199c9b8d3c1619289f71df60583d79a233f9d59e3b204c7d61c029b6e0cae316aaaf9a26a4b5ff83cd6784a3e3d5f00d24ee56445831fb7ed95f114d9c38066c47d596fd7923dcb23740e26798db15b180af88762f3951247d88001ba69bc2b5e6803ff7add583dc6c24968111df2993bac6d3baf919f1ecc6703162723f6394fddf7460ecd37ca1c555b3f9f55c3a5ce409aeb52451b685b59a68342ebc774fc7ea714ed6ab1d3efc527f49b7fdc958c4e60b380b2740e339d3b4f10fa085d9bc03057280523501194e4be595570ce66abcc654228be8f79783b8436245ecbdd7218697c89d9c829b28a8ffab850e26c9a29eea079c1bfa0d1b34a7a5a6803694f6263e99f3e95c3bcf769b7606b6219d8136295959f510448f3ee38b9d5628ca56ddc3d901462d8cced0e9cb880770135fa07df2883a24486f48731b2e3e3d248ccc17dc24aedb0c265a5f9e47aceef061a8233cf495e4addea5e5afb3db6ab623acde7609fa5dbff1935cb8f9ab017324d421ca8b805959607add926d1e80734147d958887b209655ae07a0d7de7c1a522def47317df75cdd5b9a29b00ea23f7042478f906993578f91b90d1eb3ea89b12fef9f30edba592d42bb8681ed230a409fc44c8eb2618ee659ab28cbdf4eb72257805f93e3d64b7c73b4973c8e8a8db21f1865895d14ebe794f235208f13a49cd79e51899e4e6be57953128b03203af79a7377f77ab698f543671ebbefb3e31086cf5605c0983f004fb62d9ab97ffc1cc48ac772825c2467e0a6fb8399eb6fd4aa36250e44d053dc0dbac20ebd5a4dd5d81b62cc0a41e900967ecc1ec4abcb484c79bb0d81c327248bac1e34cefb0cb7cc88eba5eea7ad787b3a444a4f347bfae68fb51e266ad8218c5381e15d539d73e7f51ca09c10086638e175669cc342a07e32df91e08a564378aa7da832df43161fedabf7797ea931ba0d0327b3c34a69ad445b2f64ea3ad2233babbc01c8d464ce1ae919ae706bef467817fee3c48d094ac63c9471d67bb599750bcca0203fcbf34fcb7e11f8e291522c436d5e520ccf45bb7a9dcca3f87bc11a45bc890c92489b7d6b6fbf123499f5bce0ffbbc062bc4afe25215df312729da1525ff7a7d7735360c1dd875cec3257a86af7257c79872b598281dc1774adc89833e57f17830c1b0418ad926447eb97671a5b40eb3d632adff74be9a83843ccf0b7d5b829d1f718ab68178b0449e0691dc8a9745664dc609409dbc31caeecb540b73f5414c25e260d2dfc4baa95e58843febd94361b3d67f0f9cce308180dfeb3283c8fff235a8c2d1e98ce9b615666a0f2cc6a28b89205a689373483c5930d803aacb04a0a63d8d3c9c4435c1965e6ea33a1d32547843a41d6c855eef7da188fe879844e8c8016a61cbb7aa905816544af9fd62ecc00f44155c77a0e30a8e8687d682103fd56df6bd6539605845534d99c03c93cc3b377cd44558aa966e1db159eb76494b5c2292a5ab0c49b556edada69e9e0693cd714be59e96883f3643549eb8eb60b0c84d52db285ed945aa42f6ac910ddc0e673a38384eaed738c766a5e5d8b65286b542d8f7a89a9e209ab0bb3b7bcbe67619dacac7b1f954ea384fa1540d5d5b003f06019445e9f26c6dbd082fc8347b037fb27fe76420f3cc1864549a728ef86fc483283bf54c03d8ef81a75ee7002e2fbda5ba9b14da07d931bae214d96da4b52ca0e059ea49d7304fdfbc8f5e69b3c2ea9067235ab5824b5eb99506d367b0bf65e431fed7cd83a051b83e5cc2a32a7634fc15752b8939dda3948f37eb0a7142132d6d542ec40742ec8e408dd786b6016b80a3a18116898f2c65459a5e0416cdd76b1d369cfe11da4d0e24b1a8eb6e0b621fab5076565a0103ed9bb272ba20198fb523b95355d258e10fa0d3467ac805736a3b82b88182170d5fa543dfa5ba00a6c111c32e9134006e1f16c0f43d620bac56fcdb804fd7de950ab7e8d1fb43e930caf7934c972ff12c2659debf7e51b3a0f2d09dd8f3e805cd34c8fa31d5649fdd13bb5ae7fbf2f566ab8c09fd0f3bec555279f795c197ff38434d58d8aca2d34e9ff74861024b978c55328d85399b7c6890dedbcb008d06d71563a108f6ba40ee5ba9dadbfd4c533d2be1341c85976db2c51d2c81fa4687269100c479fe64af58ec8db207662a8920da7a411c77258cb3a84899b8d86e84fcb16f2760e949735c2e1d080498ff6236b9682e506e82e684cc8f37c3f85bcf467a81ba78b1103f70315e8c89863e18e217d6ed674224779873d6d4a9d8ce3b28b98d7f6ab90a1e4de2f3caace3bcdca26139854e541e073ad047f01e2f24aa8ae34a3f726cb4fc6c39ca357e6902fff3087221cf4ba8524c4a51ad18107ed500f45f36f3c52f8107790a15918c914ca528a0d6e404d81b5eb4378bc1fb0b90d6c0800904046f4ca284db5119d1e3578f10d3c5f8634884eac7b217058ccae56541a3325c89a552514bd50c3f48f0d924982fa0e200066eaf2f727497d5abf38ad6844979512ff975c70110b3c7d553aea087a7d6ddcf4529c8d061859bafdb9bf1b8b5f1f157d8ede2763e2eca65a304ba9781a09f9876a8370311d37e73c99321eb014d3d692cc1de4e5a19ca6850ea72c507c605bcd9ee3117125f80eba2b3b20a6d7f918f3a45312c6f44f79d8d59740e0138fe647bf8173dcfc0cf2dc5905142d6e133fec6e72eab73d30ab3841186b3bc71f142a12c5734e8bb055c335566fcc0f5fb321a457d9668988edba15b9b84bcd831162fc69297102a8b096129a394f47d06af28a1e637b093e40e453e6a39df35fda4ad1ca9419107ec576fc274b80ccf6d73f552b1bcadc797fc5a4eb6d8602c4ae151d3e1ae5961c01c9e514f8f624f502faf51a3db65b5af43beb2ee5d117706bf89a213cc9651b2314a6eae3a5f98783e97f79b9ff63b1adde7d13f22e257158796ae8eecbc6e597effde1f280f5ef3bbbcdbdc37ea19d5c512ef49b452e695ca61d73e2aa7ffa5661331aa85418f0155b647c9a532cf0b79be2a851921c8e3c72d679fe9f44628df42d04d815bf3661212f1d498c623a548b2bbbc622dec98e4e8ef436de33efbf4bf9353f950cd2cfed80d10c00ae706b58caa45baf7e31ad28acc6a3b0d3ceeb2a92fba122d991c19dfefb156447867243b906b2d64e7ae654cd90d160b55e7c542244621eab7b6a01095875683126aaa674bb1d426bccdfd2a7bdaa3f7e1a0b7a8909de972afa89b8fedd28325a37bc1283df60e7d2fd2c3f3166236f1c5e9775e10beb802d873ee944802f2dae0bb9dfa127451cd9a5c189f0ec147ccbfe43c83d63703b9c3a93662780a5b3958b26dd835b62eb23830e779bafafaa3f23e1eb1baf6c95c1817fac2ae64c229843e34804a93610fd306e3559eddc5aa781528ea4abf94669158f3fc141a3151b7d1ca641241cc1535c63bc611a21316be0fe95eaa147aa5645707e12a8493811460607f03deaee70569130cc9b732537750022b0bd268a1f0f6424e1428bf2ddb5e7f90337115e034ad733004084af486a145131cbc69340ec4e933eef10356ee17340da281dce013df72bbc718b7aad8ba2265773e9fac54597382a1a255e5288a4d1fd5f39f027cee1467e3f9a29a48eff395259f9dec779d74915e35e32ebdeaa3e13b59beeca66f71862da21bebc6c3ac0646995ad180699d20a90565ebf47ce6a509c771ed251b9f37929ab4551f91cbd95358d3bca89bc81fb0440699b0cf82bcb36156af122faf24469be578ce8f4f73ef288cbe90f58674317fcf13578a5b022418502d8f76f07aad1b0702bfa302078b90a8f19987034678033f848111f4d5d5a2257f795d89562486d61f2f450d0123c0f26637db625f3c0135e46d5cfee961816b24ef1c740e949671f1818643931c9eb0941c6ab76e37a9437779a6b581088d3958256573ff8ecd28beb3a79127ad8b15f8c2a9bf7625f2123ad31e1f41871a770aef09f34f65f66c033a5c9ccac53c6ad9a0b6119d92ac8f78de02312edb2fc94728bbcd7c2a7cb74d49f6c625dcb8218678f22e562d260defad745e8e349da0e406ccfe1fde6f05f952835099613e09902bc6c512f45e1befc81c59ce3de8c7af202075118b79b519381856f4b448d8c4922c43582b0ca3dee58e7a42eea98c900df9c0fb1ad09613cb352aa74e537306ed2848f748ad99e927754a11954deded06142d66a21f9ca5464b3478d35bb851606c45cd5a1f8504cb9f1894d190d6cc870ddd00015277456e4802f74314315e81df7026aac7afd1c7b8fe0fa37390c822cd45dfe457ee17930a2cb96f1ca47adbb9c8fda10cdf9ff6ae56809845b6224999bd6538ddba35880e86eb5c9a2da0b835d38f33937a0e98e6c9fdec10c1009a2dd8d02a7aef30d85003b9de9c38ebb28d2a38c7b7aa1352e760a9a2c21e2697ccef6c975598d086aaca951de6728d168cdec8a2289a77cdeae00a85b1041af1b4f1ec6bf3f7ee8d1a4b2a06fe2bcb2561018896924c57fb6a0ba1a4b7e3a097c4dfbed62c687f75f69d97936c7fd0c14fc5050914ee9f3789f6f45d0dfc9615c44047c26f80ffc9b0a5eb5d96af062137150c893025c85753ef64d6b3e03523e744daae295fc73fafeb7b97ce67a40d4ec6ae306f5df435c819c1e1e8a7f4ef042698d76ca617d20db76f8d91ec522ad6d830f7b61ec1228dc240737d9a698a40caf78ed969871b5cc7649aa5e5ea46841a44d18d0f5ecb527083059a3c9d823c8a97625078ff79a10b4a4d037016fb14e31d47b352d6c0bdcd83f67e91e1aa44785c1127f1d5b9f54257d8eee4dcd0e4d270b381ace5172b8c706fb0369f1928d153f281ca00411f50187abca01bab1940d63637eb2b00c8751526977b48ce0d91e487e988cb754d7e23497395c39de04d27618f15cb662d20736e3c54605bd2a50f5c0fbb727779a2eb266984ef7b616c7c369bced28d50235eabe8956c749c5c11a2422c950c42ac8e098b8bbeb7b2c6b1811ef17f5ddeb98a021c13778f73237bf0ad9ed9e6c0ac213edf1b55e66f275b933ab5198b92b093b6bebbd0122364f493e5277158a1c4c409b788316570d44625870ebe55338a7198f8e39214196df858ae78095b55645de3e8d52e4396e9a5c4a949197e2ba03491f94243dfb12714d56577939bc09a76927ff08327fe2ace2bff5eca8cd62dd87b376de92a10a9e664923957e767423b764f284b8e39fef9f13e14973a983ada0808cf32992ca00e9bf44f2e02a53cc3f4126b39d879d208c852816e1db8fee8fb419a86dd2bbadd8ce71ecbad310e13fc018445774582a94db15e8ce66eeb93b5504ca52e2566aa73f96ae6fb71a474748a8542377aa80ad11d390adebc64fa6dc0b7fed9e82ca610f9e3e7c6f96bb390bd98935b7d4ec28e9caa7b9b3a5835a98348a0e5dc48b3b04a9313ae00b463fc6f288401f5a0f324ef6b328963e0d9d3fe280b488e679eb9623d64c4cb15d7da438965e5e566ab499370ed7dc4fc5c2c4622e293655bb9c287e070c1758bc8405d1402930d98fb25d50607379c831f7b1b7dfd9e00ca4de8a7b89adeb1caea15c0a99842d8fb336716914531b89343548a46a3b0ada7a18ef74dd5c1368ec190de41dea7edc30ebc65b84d6e54662dfb33f2f98f6958702a94a8f33948f8e277709568dc9649088e87af1130f5793cb0948d875f3f3ea7c4bb42000b8993b072498d7ef8c51c986fb1de0bb493554da4f274c6c33ce88eb127f0dad5a108c2deb5d123762fe0cb26c050ad264373b93527a0face6eab037de0960626803287d9f6f0e8ef8d633d1808d7e9d4aa22453dd1a224ff9d0a63c857344ef4859a289c0af3ece82a8ebc32a7de084e83b04b5c6ada0713a45de7ece70ad5deb3a92c703f6fd23aad77a8512f7ef1a7b27b925e78bd56fc2ec360a48fb87b49d12f187b65fc5326bfd73ab5c9bd082f14b04196b990fdecd49c2a9358db8e5a29d68ddf867008582cfff4a379819d3f0b46d7364722deac4e12ceef8f1bc3aefb8902aa7c2c971511eac7eba440716716f8a929204978e3f8185995b071b03f839d163c12f2599237a5a488a1df356deb31278633ecdb7b994f5b98952fd8d34043d18aebb8d65954195218243f0651fac3e889dec4a9cdae49d9c91237684e2116effa5bd83028828495f10b3ecd37c0f73896e74a1b55e1cc9fe2a7599993aeac14b44daa5016e0c1ccc8874e9451ffdea4ae44ddc576da1e4c67f08f4f4d06cb8a99285c493ca4944247643e85d57413ba9c7d28e833a344a8290d566a8ed150d0d6ed2606d5dfd7944447fd12c065848b4992bdedb8d8eef77349606ed6513eb637cb6392b5b5711fdfe53a7ae96e7e091086a6cee510fc7d46a809e98edb93b75e5949ea1f5fc146f2d1e4dae283075a37458e381cc14a6865b98a6f498794562cc89dd579a942f8296e2352e54264018abb48d06dfa5399acc7f6b5b0df1cd51bd51e56311414f9bca4ceacafd3278b838979f00ac7cef7ed42a97e2f79412c5805a6fdf051c5728e7b1403a39b6ec50933b4309b53c3dc722d67d33008cde06586fca17d53470e31f6ef516487e108d0486e2f8bfc6bf5e16ebae08b2323c2ccc5ddbdd6bf6482b64199254806621da2ede2498308a7ae4ba4688f43030ff891285779d29c8073b31ac6ba7bb403385f2ac80940c1a8e971585c9809f5ad264bdc3c12f04701f9ef0e11cee3b424b8e502109e59af22612fa1631ba1da9ab6b4ecb672067d747d5e7f5ae6c2f7cdf9dc5c23d2b618192ee76584f113678d2dc4c55bf08153f7c4dc14def87b1b9fc707c92cd785342785f0d868549ea4c3a471ac4e9a3ff35e1fdd5424381ce68dba197917f2d9d6a4eac6b17ae580b4be5200dfc55fda1450ed8fcc5892eb0863d93fbe5cc0a5b0e1c094d0ea78be1367c341d0c5b323de1bbda19f6a4b5ab39e5032141d950f7724d581207cbe22909e0afaeb16172f6c69c4cc5d0550506bbc644acb7c8a147aea3be877ae0b8f970767adb4cd4d5e93c367d33bfb0ab61de415965a75b63d1bc474d32529517c5e9203e259c077aeaed38b6296d3d878a613e1f4becf3d3c1b009e5aa3f6c533b3d807bcce5bad991792d97f32df5bb9d714f51933cf050be7f6cf2c1b891b10941f2092f3de320a8c50c5750fc515633500579cdd5d13061c7e1ea5ea2823499c537e08f46605195f13071f3bda72c37bf18316065a7388cedc8102ed4a6021827c152c292e9d7613582b5e28a0208ae993ea3b4581bbb4096b1c22e22bae5b39f4275d57accd3440b275ab8d80d5c95fc6fa4004a4744ea0d7e96cf01063a12a5d5fe474cb99422de9d254e6dfda0cd6c5c91dea59d0c2e290c2e9786d637b2ef7ba0dc8d44d047b9a74911b1cf8b01c745e475c7e05c0357b0f9f5c071b0432c1318f5d4b61816d7a2a63a24d5962680193fa0e89ad130ef0c60efacf7ffe6da835b06d89781f8d2c78071b76c0ed30ae85211fed62d5982ce68ac6c3160f98722700d89bf7f8da98bb8e5090bb34dccfa796739b62fd9246624eb2b07082ecb5491b337182686adb9506abc20be9116c07c88edd2a7321d7459829f056b2de08a420a97bfca16183401d5567099d1328b59d93bcac8820aa6dbc74265d0c59a34e9f7e014f4c9ea264a114649aa467682ebba50527433f16bd081affddf9cee3114915300c2b3888fa5e161b75946b6151abe393d60fcf5dd99684ea97bf32be266eb745b9bf911b7e4a0c28951c7c0b5b997c3830d2413e227a72160ede453c7e84268d2deaa845139ff6a3a6b209cc52dda25b266cbfa79aca476cf11f3b42dffb249638986de16ba9e7f13e77dd837aac04ea542db69329d3a143e4466bf84fa1d32253d4da3d14f7df112e2ebcea6ff3231cac9ab052c8a06cc0af098616b9756ae72c9be7737ec8a6e64bee5b4cf7546fb5c9dc146f658cd0e06f36e16e34447666a8f14cf52afaf5d6c78ac8853d33369651fd46773043a4e00a3027fcb600f1c3059657132582ecd341212927a08072b54c1f824808eac25cf5a91216937e25ffe8e1f80b89369039ae48027075f24fb1d8fd0b8e88acad85e97ed525122065fe94be43fb6cfa02cd0b09e5e84c0da492356e984d256a8af680276c5626aabd84999373cc72dba9403c45544e567bc51dcb654405f8d9d95585cf2fc4176f56b647e8f4f36d708f95323d95416732ba35148738d84381fea00e18296156dabab5ea7bceba492648bd62889e13cf01cb3992a47aec215f9fa7cb501e1aced81ebffb6d9924340cbaad1277894b0bc1a76f1eb4786ea073c6b215612dc7a2fd9417f62ea43b96fc70680a6d75dc0ffc948709b4ba9fa8ad49afa58c91c5301311256d2842160ac2b7eab6d7674d17be2704e4faec4b5dc2cdcfc64a36d1282686ff87ed649ae108ee26cf1d480447996f9a531928bbe11b310481b148d04c169bbf861e812b282697b46cc35846be8806ed10c8d5eec3393d1a5da1e2e5477b5c5ab8e5ef333f480dce2de08ef563ef678f83f83cfb1ce0bb1b1cd855e49ddb9f61fe9bbbd55e2dfa141c37ce06cff653e03742586ceab349905cca6227367d4f683c681264da1f01a279ae74e23e8bb015ebf240916abd8903c4c257a7b9c9a39892db53c06effdb1910e2aeced5288534b9e8a025b73175bb2ea7ec88372f589c13edb6524166fb61769384816c6f26a4ea976d4abe8629cfed5feccc8421538794c05f3370dd2f7aed0419c6f50a06e4636f02992e7950d5032246e9b830d40a370e2d7070547ec135e727b8dd6ad89c7d8a010515dd02ca0a5c39dc0a9fee5d2f4bc03bd6d3821a548930faf9ff35f54498299788b6cf98c4fc7855fceda270e2645ed6861df8d1cda79db5ab4dae3774d78cda21b0dde4f336d7c7063f0b39eb0e8818786ace5a94818e623fadfd591f6b07fc634f33e5deef8a06fed00847e90703ac5c5540c73db6bf808b347531ad794dfd93cefdbe1bc3492976064a4e8b5dff3d7fd699db2460373ea92b52a7953dae3449a78f88bee8407952406edec614ea850f410a51a7ed2d43aa6c4a645ac161c6eac80fbd50fc9e3e4a14bcf4947ccf1b26b4124ec16ad95eeaac82f0bc209f9d6c2f8dc595914e48905ea1d6e3e1287410b955761154a2d3b7d3c3b8a7ae24dd27994acc8cce4e93666223476033c8207fe41c3be8ded7f58a77442271640548b7ed5eef57461f76c4d944f1f1df8afd1bd39aa2e7a62fca7bcb8499c137bbb3b0130223365cc6c56bef59e0395f5b2fc20b3d57a179aab2dfc0794d43b8ef2b6643bf1b4ad677d27b72411d590ca72f7736040238a55d1545c4b5e5e358214ae93fd443667a871adbc5b185f1f3f32b40f4e99beae1aa5d30b65dee0df704cffdf067c1f542ca9486f6ce539a1df201b6c8dc7cf2453b4310bb734b31228e658cd5770661fdf00eaf7b4a18dfdb85b2c58bfe3a2062f5562cf616f9cee5121c6548fa8c10f15df658fa5e3e4003cd94be620d7af34f02d6ceb4eb2fc29d100f06ac170b6d2ee5f95169bb9afb9e5eaa8b915244b25fa05621a22754b776f69876bf29961b91dcb3657febe202483e3159ae1951cca90303f07d76b399067b50510e71eea2474c8d20bbaacbf6017d70ed40ed3dcae6bef178bc06483e6f0753a52374a026137af21373bb9a8a0c59badeba003a3ed35bc7d8fabf765d7ae50a791f6c074b73adb9da55d00883893dd7727997466d96881b6c7945279a091d774eee26c6e2b6f773bacbd103f8b740f42aea36fe4c3d09e87d67c0faca542bf9655b9c0418de7cda3aa17a6656a8a6d46dffbc9016d87e27e7bd156bc491c8c48ad87296311c94aa1cef1fd1db0331d4763247c6bf7bde2d6d2e77b97b61d73c5f195bc28ac3293a0c3b2911af2910d5bedd33474fb6d670a904264622f9d0b820952d918402ad64227c9d44c307bfdb3d126f56b7c63f3455ebc59a03e3af6c1446cbbb2436188130782c37c1f52a789aab146000fd8c495686988ce8b74e2a51b6f7dd374f4cdef4fdb68a1c400c80bb6f42fd6edcc879b70bc2e6ef7259f50b7a89b78b00f69184db5fa7b6878313aa441259f0223870a89a11c214c878953b797bb21ad1a6b189cc49c276a85ec740cd35a10b6002fa4eb58b234d2c87ea1e4ceaa67ea29afa7f63ce924f7144ed35949f06b9378ab261772197e90df652af628d166a27af36ad71025411d622975df1aa6c33555da4caa90aa30b130f482cf553ce53c6437a4b279fcb2e076c0ec9ad6829c0d224a19162c7a96dfdd9c14b727cbda7d738c5cfac92b0eb85c13f67ccb564b4058ad64782f2cd2cc540b1ed31eb1f1237151c5054c46e24c62cfdd8c90f2c5f1d1385a41b601715c2a49681680d0042364233d63c998e0abbc82ff1347de322479aa55a753ed4e2d20f8d26d588b6a7f35c955ef1c1300d1a6eba37356c6e3b9d26a8240eb8cb89430232d993f355c2898e0439838d1aa91e414c93aafabd952256199fdf9efb979f543236c41bb0abd77201d3505ada8befe4eaa9712ea4b747225e22e5b31011d71a0a9f0f45e99cb8dba4108ea037c273e04faf8ff368f222d96e581ef04de776428fa5450018849725e0f8ed30eef26e99a2b13c32b32316ca04f6756c85e0399d870861b81801293fc9e6ff905c1c2d25da6cce72c30ff9df53289ff88a61c7378a6c0ac54602bdc48574485e47f3894847e80049e33e2710ec9bda96d51c5dc1e2e131cc6e2acafe6111bb21cb98eeab252e18d1bac88e89827e85b24293c218eb2a105f61bab090d64fc3c8d7e4dd5aa6d97807c28c9533dc38fcf6f48f4f87527a649111513b48f72a505e72c8964c4685f0d1d6381f1c465db2df2f988b466bab1dc4e1318f53766b919a89b10847597c9d2b4de6ffe6561e9d21a0e24c56ea772769e89c7286b0d79fc001765dce2e382d43cc9f2c70b3df340902eb8b42bfd20b3991006973fe3d209ce6243a989b2733d2ecc7cfa9d141604c96295aa23fbb005c6501e2aa2ca6183e5471e0eaaa9a4e9eb359c551c785b1cae84cdbbd2c1420e7d8bb66fffe10e06b3887199053ea3f941e93bce9733c00864af4ed3e3c2232bfc2188bb9ff0f4066ac97f9271978c5a00d25f99a86b6eca4fa02688b8f114344b19f8b7c84f767d283182c3c50e97625e77101c92e0b0095f6c3e7dc8cdcd1026216ec285f1d60171d004c6a8469b7f75012634c9cf26dc7bf0cced47722a3271c41302e34890d96480e5509aa4646763245566728caa2a5db25a1fc8336ad982257e900a5b3481e7f690b343604aa540f89a43669a016371d67cb854b221cbfb3dc35a9c8175ff88c18fa1abc5ffde4f2c76c57d25e407c5711aaa318ea208e2bef534e4bb967fd22860b2fc14e6e6dd8fca1ea76a715d9df3b6a94dcdf99c8d383855e4673c627b523f6374505c60a50a3207917fee5768fae3dfb10518d4b5118ac8a6fcf12ad362ef637317005a4660831e3e421ba44d0d7579f90c6e6adbf4c7d9873502b363898c82913d29ad428e1c932963b2427ba267b75825b013637110d21b6247933eb786ae602944ad77e10eee6c41155201423aeb35be644910369bdcdc1c8fc4f9814c31981cb929e83793e4c3fa5467aed21ffc66f32b2cc16c3755fdc37b2e9c37bb9dca8c3f4a2d8dfa7621e8f51248962b21acec3b56558d2a276dcb1d47c8eacf7283c4e7172a6e08b85947f09f55f22d8a55fddebfb3aadedfdd582e3bda3b1bc70bb796bde7f4b3f9e371887036365a3b59aac827145c807dce07de4a93d5916a34b9b1f7cfd1f6b40387659a08fc468394ef4db1416898dbfaa3244e9b67386a03380dc66ba9bc5d4792d87431e28b4f5859193571434c3f95f008d19292d4fa2e61cfccdaed774a3221096ba8648f4daf83e142c05fc8a00aca70432685d3e506e254cc511038ff1120bb1dd5177744b5380d0d019f718390312de9ba37d5ef791173e7b5f869c4382fc3e8c18726a18f3b6bc0f925b39ddbad505907016324f3c27069076459f6e4167045eab3879ee85292ba37919197ef2a3af154252769b9751d368275fcbdd13464c227ba0ec8a522a29483775c2d68009d7a302bdeeea43976dce24995d262be204cdb647130f5a13585ba06735bca5399a583050b221617dd8ce7ada88bd185e6c5d278b893b46eaafe5e779469598da596b54aef825cafca15fc6df3f2cddc4888b977bb0e0471d6accb13a8c8b8c34cd1c2de845474522c4ad6ed0bf7da4c06c57a801a7ba041ebf27b11880713bbf4777b53363e9410440add422d3b4d5437d8bdf86761a0a66f61006d478ac79a951d476eadb58e88f1df9d1dcaf31b8a794e22592b61e78f7a88d2ebd299510ee2bcee961d7aad3bcd3f1e0813c21aa8389d753eb9aade9da2077cd3c620cb78fb0ee70ba552cd87d92da9a50f211556b1d40fdad7b2d9d0f5fd949b07e8f6adc9e5d8f02746c9020f1484624af7712ba9274774e26513e1df34865d3a1ea2a1eb4f374821b64a8fb5dcafe11e239d56ca7248f9b69a5b70ee81ecd485fc85f44110dcf7543a04d93aa36e0534aa708b441fa561794d56a4afc0a2cc1476882306781b0510e3a112448c0f0c81a334c68e511ad5ca0e0f032f2532b9805e36ab686f4512b3240d5aff90ea2e3df0f4f3436fe7e839b9b2f612d19b5bd61b15550a11a85566240284ded204f7a92a54dfb4a1f7c3686e46600135dbe1da30e2c8682bdeeb94886c75cb8e8510654901189a9e2c75ae082027297850c6502b3d83bb79f4d5acd07b4f50803fbcad8d2a35f9094bd274e5930e4465463c8164ef0bd2b4982bd1ac1b644a5127816b83d307894fc28fbcfcf26d59eeb335c6e327ff271576a2dbdc772786787da91f7a622f32e94b7dcfc076078c5b082344f21b5c7df0a044e6627f6f6d6b8b600c7e33429977a1d0e3462b89311a38bb86207a81cae1e341a5872b66efc84e793ab193ff6a19d4d78e49f8638554349595c5b587bca40a668ac8070863228ecbe8400dc416b38e51e976e99d34ade2385a4e143fcbad6662d7534246964eeecfe1f217119f2633c2c6e8db5ce466c02ddb14f36482136d5ea43b8b1456dc852150b8d6c8f042885e390e86fe6597889a804d4a45d4e2d078cb5056fbf9086d7ae575e0f731fa5cedf550bd2a3abdb8127c14841398320d594aa0510ce424ab107af78b03cb5d9db7c99db40ec81a6ea121fe1b555aa70665f925b4501f1aa2355f89c8ee0988846cfa152bb53329bd9457324f41595f1fa5b349572ac5c45c9a6fd0cb7476b72fa20b18dbe4ad2e3ebd2424d9af6897714dc6af2c4ea118b7491f369af8e4097894099f7b537e2c30322b0f7333bb79b3d4d1f0e612df42e90e042cc6e33aac8c9f248a5248b3890fdffed18e3f0fb9ccfbee77227e310495f010b07e90756a4afea36d319a754429c49956dd1fbaff1da085c1b91b54b71b083483ca83f4da2db7e61efcec03da08bb51076ca82057aa8ae1a66bb777979eb7c5ee5d86446c4d4101c99a4a6ea8d0d8cc3178dd529f3a241798a161972e23271bf78ff13fd0013b3a4dc835e9ea63f54cd475c0a1523abfc7a601fa98780117d1723cf9a97e62b6077cedb349632b4051cc826fc1c19cfdbf72ffb450037be67763f08090f5c9cbbbea9e97d15d31105509aa783c0220da1187798543f8ab7f529f1403e1dcfd1d82d1d111b925fa2cd2a88c658ef23d7a45c8c593c119d54237559e663d24be6229cfbd933996d8cc1051c06caf88733e1584ed5d8bd01a56f71b6bd705f8b2191a202596e7113615081204bc25b898850097319678275ab9abceb4cb048aaaec95fe75b926147e267c3fa7e4447d5a283b6fd142ef3391ae14f9a1b4495ab971cc8ef6e3a8b373158a0a168f2db4ca31d2c08b2d082aa0d71a37614e182699898307b2b55f11208924be8672429142321da4d29a9bb260c0c29fead7755c732b40778a87e7b58a4a44836365cdf6c358ec365af295655f5800a88e073d044c7b648ea30f98a79d929f4c1e010eb0d3d829c25c1248a72f21b687d3b8de0d24a500e95c0c4c660b3f0427c3851486fed8040677b05ccf150d4052fd3621b9bf638d4563daafca3997cc84e709a9ba6c31e0310cc4c551041431cf88289769356ad6439654a070ed8e342f55676637cc0661c35a3ea84788bbccfcdceb42685400fc381266c5e36d8cc9c63963b92018f421a269df244d781cce7eff1f39bb0fc8de2f5cf71232e16fee19a4187071b4929aa2cabcb8ace92a4acc091c1c8aad981aab323d1f99193fc30177d11e2cab94f08d42c4bc39ef7a164613862181628a5814210176aac8705594a60be155f3f7c35c47d3d8db63165354bdd0f40f4c40394e4966679a6282aa60938f74efa547db07a382037aef383b99a0b365a55986c410b40511fff5d0bd2838258155d2e802c314762559c152da4cec97eb2e39d56b621dae55a4416479b7141071e878e1cdfab14b18f1c5f62a1e37feb1ab66f404cd61c2dfcd4b3509df6a230527dc37ad536eb38dfe31765ab2ae2165fba38fedd665d4d44d1fcaa1ffb199108067bbb34cce5cf547ce42eb13a54b0b5300fac74112a77d4fa983b1defc80edf78103d31379bbc6c006544d807f45a67c812f86ec17a3e7901dcd9c1a37818049dea03252288ed9db37df28dad64190b8261495aad54d4f2375d9543a11d939c5895780943629ecd02a56987d53c1fcace2d2fc8aefcc9f6327e17ec977234adeec86092871ba33134743ff5a879183f630f43206794d2a2fab5de3aadd89d919510c4b7cb1d1801e1aa2c2c6d3b6063284a423a5aa7256905f6d887ebb5b86d16bad3cc982a9e671d387829737c7bf85eec0e60aca21763cb86337d3df0d8346b0c52f297bf1f732375968014a0a23211c1af77c680da4beefe9c30bdc4e2b40d6628b9a4736345d010fa7079dbfabdf14893fb76803b5bfb6c574e4441be7c6db4bd06cb54e4bd08999ba3d487b29a3b0cb47a9ed18613cce353ca8568f97c1883142f1e449d023ce47da33afd9378bd2f7842abca93b4e0b224fec187235d0edb788b524a7795b8267c5f3e35f11e130afc2395d43a2ea156db6fc639402b9ed18c9113872e19e4970e12958d00b0c66d3f694a63a71b9c3fa50d2122edff88e99342e1e4c60730621031cc4c918a23679cdb861784d7cf3b38c468adcc9b7f1f64be8c00f62d17302d2fe493b40bbeb43bd0810eec61f1b8155dfa4e7f770720e09a8c3ce44eb9fb5a8d6936f5f19f56a3ee1459cff8f2be7a64372bf76ad03a3afa2e2b7aa77f2e778a8e8266c062c80164153887d5ebff91f877d3fb5ffb6301d6086f76a9ce89354a876173bb1ba7c00f60e22f7b01e575c8f309332428ee5d23fc677821836fd4f2a95de9e2e135fbb281dac41a60bc8d8810ef32402e89cbcb114aa9dc5660b8d0ff4054a5b769f10156bad8ea04c3a0afc2a1052b9863d667fb034bb81d7f58493f02d06018c4c1363887beabd3a5b1e2be8747099d1180ec60cd2468defba5c967719fba80ca480dd6b104a882381f642e52bcab399387ebf7771cb60b0cf4cbc661af64956082dbc29971d36c1662615778f5f4756373c16f54ff1bb9812e9059cff594a95966e5638aed76cfcb9994f89ce415e29efa7f96239bf1f619a476e0c2658e3ae36be6c7c1fc80c49952c27371a7b620376d91907287b2b34ae02c16aa27e9496183d080b2074e409e90885226a4d83fe19c6d4aef9022391f1741ae709f323a25d62aa6f5aed03333ea04378a1e4a6ed45246c1e67aaaed52737697e5b62696844dbb05b841e0571d8c42c6f2eca73530ddfad25b37369af4b8d3ff864f6ac80e0f5be43bb3a968866d824c07867c7dabb7db55d7d3c69a4e4b5db81e1f58400e1f477a276ccd39ca4a8a3d15243fe9fa75efbb52e4eb89002da7a2a59dff5403a9f0335baca760728d648d9cb89ad86afae53cb2d68300f8bbc5a5e4db57fcf04464eecb779eca821aa5f326d91d2c38736a306e62bcc3238f7ecf65c568391069a69605420b6d5dbaee275df3101f7f0414a4801840450081bae77c7d531aff81edff61201f0f0d2fda8331c7a813d97384afaa92d3d0e97834c6c9d79699406b4a482296af375d6cb81aa6639f4acd25a1f78a1d1f03b9e585406381dacadbdeca752b3720baf911a0a376095dffe15998ed7204a57a817eac6ef7cc9ad75a5694293aae0fd225afb5c09906bf1a647e8433cf4eb1a30e21bfb0fdd94f76a0f5ad880df7194790b6a67f6f91394414aa42a39d66ddf3756445074d370ea78f58c053b3afb9f3d875318ac2852e086f9520bb93765a60d62b312cd56f1ce74cf043cdae87298dd1a8fc062467a2bbe8c8d9e7cf34e8cf1fd27d1bef9a0bce6a8a5b4d75a0915eb85384088628566ab488c76999d099ffbfaf89444637cdf6ff6396b34fbdef086d09135aaa40557366f4ae635e97062137dc1df3acd4a9018e13548a58084ea61bfafda34a4d3245cd4ac3b18067f6081384fcac46b96f881a1d155653043bd80194fd2fe6417d736a0d942d092f8629970dd58039ee12f5d9803aa5123215a9fd90a97c0608d8a0a0bcfb5bd029d37235b5859cc1175c3cf91f6fbf3b1e84613db745fae5466d20141695ea9c42c60f092158bcbecda586e9619f4fc8ed11d60aec585ad2f7c61276301cd55fbf2dbb080609950bcbf3bbd30652250a8ac72ce250fdfe59f05c86a9634ce4bdf245afa06adea196426abe4ed824f1a5183dcf8ea5d8e1879a986d75dc9a36c1b7c9f0c5be257009e2d7404b4cf8966817953cafcb9a342b2f4a4406cc8359aeeaff1fa479b69e8827eb21bbf65f14a191d53184a89f2c585b14e8644940a44b1dfae6a703ddaa102c654b67c00ca28513005359e194d443510c2cf94a27a963e2b9ca1c7fb5ed440b8cfa1894ca280e908549c4b17c4425cbfd014c22feec666daef4ef6230fdea009ea429bb7174b48ad43b801d674011bf05a728598eaab4b915e0af9c2312d29a4dcbaf2e038e7853b34d60b9b4886fc8907aa750d4e07a8bc23bffd11d865f92b73b3c1cad4736cbbb8de96c9216c80f01a43a0e1edd071a79cbe73262c4049c08ba835f5fa4c10189a7a876ca8cd880aa7dc28b0e41f487b2b7ee530f19a317e6f373add45864ca75026ba85d30ee7ce4abb728601ff4b2f3f6b8f4f4e13ab52b0ed0aa64744e87262a8ae44937527020ab874715d996d7c91c6e3ae42ec402b46299188dd84f1cb74764343f8aff397d67523723a322782d64f2a6a495e48c0584570a17592f145818a2ff35306a0eb01957af78c9f7c8abcf4347e23d44305829cbec486feda0d2322f92e99232870d11ebeb674b266a77706b1f27e615ec8c700392b6f19e2ee7aaee2ae118d8a2023590eb97da2d5c57f2b39cdb4b4114271c77cce042bd0d9bbcfc3ff3f13c0f51360819f48dc3f6c0bf50e2755ebe52eeab6e5e1d5cdf0b0170d6216a1c9b5f532a84e26df94c613b013673aa424d5c5be4e603ea7d5dd9812cb10b03bb414eb5280f66a0e89728cdbe7aa57a48bbe3189bfee9e44d2c80cec416a8406788216c3aa38cc21bd3dff4163ce355256250fbb0eb0abcf82bc9a71a891c02bd5b8fc1b7c86287921d115e46e28319dace442971be8055fc24726eef7ebe0c3e1a4d29958eb427f2fe725afe48905b429401f493eca6b02dcb038fa44a9531edce185b26cd9c193536dbf4993ef1ef04974cf2d8e98d657129951db9e30a0d35ca6c8cc242c2fb016a239ac83e2f1e5646d0acaf243540b38965e826aafca68fc8dfe09d6190e5a6b526ff70f35ae0409f0c864125a0b7fcc98c3085b5dfa88530edda8c65e7fe5a10a680880e45d82ce8725b8cd44a6a7a57cba5283dad21c18f962322d38e6853ce8fa2a1c5ab77da693e8b9cb1e1a44d66d2bb0f8cd9c1815f601d5a91c07d6824487f617f8d7eab48c45ebf732b72053a130841ab82f0f216022ca0a04cece4d5efadececa2ba2ac588a456171849c34de47fb71134d04a6cf7a1679319656861ff1113a324e1c70059556bda9e34dd547b719cdba5594301dda8feb95c6ce6e7f1c919328b8e7575ffe35923aa27d3d28d3c9165e0f7f4acc547de8c4c0cb9fc36a8c2e3909a85a0e9d698676e08280de2182b01661f63d1fde0648615ad82cff1a34132f4368761005bba184f2c2dec226b8c7cbef973a4dae7e14c6304bfdfb3820b036553726fec4cf52111c18dce16a516edc388080a60a0418a046e46c702a7f08cef53450b346f231df094630b8171d2065651555b8861ade3b278fe53fabcb8fac5085bf0413e06cf21b999c6ba4a93d8cd64727061d310b8bb92fb73dca9d34027c6941ec1aa42e403da3f1800b43004782c00672793d7ba983d7493deca72eb4577b8d8e6a7d778b24cdeece0cad4174f18308e0e22afb1789da94ec8ffdfcf22dccc1c9724b2a3704da4023821d8fee379c2cd1190f88ee8bac6bf277218e9fce19bc37a844d692b95e5a3e0926ea5e7a705974d68e5cc2dcf568716b59b05dc75633830508ffec467e83bba2f0d7b9ce51528e5513a6c414afe3b2d5482c089c9bfd796852fa6033b2b22758f617a00f04e957f3481bb28c0153fdedf0133a833dd2a68bf87cc71359ed59e51f47525c24d300eee793d9fc2555cc81270e00475d76ef95fb4646779c722b9dc7fe27f196cd9798faff8ba365d877dd78698d8dc9a1773c48c70647430dd0db2bc693f04ffb71f9e4150e9eeb796086ded176fd3cc6577aa2132d4cd7170312217d13ca661ebb3653fbcc27a00bffa0405dce92a8ae3c1961039f3bcb480869576c3f96a38f9114d3fd897326367bc7f56a10e00e818c4c24e86f405dfa61bfc98db3021b1931b99bc75fb8dbed643d8d33a03162263c7e247fba8c2a874abd622da9e197f2f81bab04d49ed08d4eb44827148bfb5e8c96b91ad301e2e3ee063cbb66a0ca2dece92bc47ffa30abc01e21b709e0ee576debb5cb04c1ba1463679491c7cc8acdeb14aafbb78e7063b542a88e9fec8f17eb5b496113f095213fbe6a39904a2e8afab82c646dc2307a5da616a714d1aa4ea398647523cd6ecc11e6bccc181c7f9d2085b339549a9b5f0f62f6f266648fb75710754c8ca66d50443fd87d71169c264e6e705ddeadc2f63890654ac3980d8887c8451d491cdeb5dbbc2652bcd134064256c4feb67773a115a98ff482b0927e3ce7034b3a37ac125113cb99292ef57fa47be1a1f8d07cc3a28e2972d9ab642564eda96bcb67fef1d8b30ea1202c7607b1c040af9cf42323a56c6a5c1fd32e9900672ad705534f115deab3ef56a4d1e1ea5c0585edcdea2877b99a54dd1804c1ddc532e60db270d8dc6009982bfb9b0faa9b1b41102a843221b2a7ab335bfe3877b1510ebcb59be45087c35419d9d65d202f37a5845932dab6eae1cb77db149fef8213382ccc454ba25c12ae8f9d22ee457752a23752ef114c9dd505244baecdd184abef9b69167a27c3f00a8ee1d861c8e3fcf788ea45b8a296860d8d5bd8543c32487236aa4d32c89f270a487c6bad7e82919304b788661b80d89b4d5ae2d53d984632b6c5acbfbdab8ae06117120d93dfa0164d08b1bec71730654145ea25f18d93a2740bda0677bab6684333b0e59ec554d73db7d57d87eb46aaec848638dd12f2ca9778456e49f8ca5b1ec4194ed25e3eb01b8b7336677b3dcbd6fabc1e25d5450a808775f5a881be3065093a86ee54ad841ef91769c2c23b12311bfa9e6e2f014153b4200c0bc9d7102621c9537f898c9c67f0c8d49025cda4cc91e1cc4b87857e0524bb6f96395ae6d68c5764ed1935ec288d38818a255f0d6783b7161c71377e5eb1b4fe7abb5af99d82dfc790463ef2950a54ebac0f40700e411cf46c23d19c2c5a7d7086ed2eb989383790da6c322160faedfa8e9be6c6577f5711190cf12540e722ec2056691f053adb43b72623295aaa87ba6d4d88863568237880e1aa738ce54126aeb3eada1bc7b682834f418c33291dae751fc8c1d8dbcb9926778ed25da35b3edf18450da896da2058d6ff81cef4f23f673115388de55cfe8956aaa3c28837637e438eaa232afb75c3b4d13724d9b80d6d7a1b89d7825976ca21be2b3d2e51d680c681b27e9561bebf66a6a75d483426b0ff40b3826e252d9264400ca713b6718e5fac6a3226a3ade618b00e62ebc38607de2dc4f6498a0596fcb5ac98b4531557752261883f1aec6110010f794199a271a8d91099df5dad4b9ba851afdbb5a852c6a2e146e91287a9d3bd2be968c504f30d0d90841d571bf6f81ce0af082ba0beda3b03ff1a87cfbffcbf2f64627fc0df216733f8037974c5acae7ce57be167d959d69027b8f56710c321d6fac2e08bd48145c14f699a6a923eba97ad18d296d8e0a873d5dae1f235b3a6c59dbc5f903099dda1c9a19b0ceaab92a90b33e983ef54ea7eef6ba922162550a2435697b283bf7bec71b8ee0c41e3598b93e433026d5742ecaaf0058f0ec93dce2c4aaaeb35f23270320d70e413917f06c83e2eb51a950ad09f176a7e9b5b0db07073a6884bd753af1f71c8caabc3fb7e23dc68dcd6d5ccc9a5ca13b20a5205443900934dded378ae94dd47c32db0ead99e3cdde07cf2a4db22d92c7839168b543a290615b581d338cdddb495155ae3238003c6596492611e9b2bead4490e77a43ab389de68d17fef3e164d8b9855f3296ab60974b2d40f50cb6c7f995f770554e0037d0e5b357266a04a1d3559a8b8a4249c15387b893bb116a64c3f99dc662247ad1b6a5c112251ee7cfa7821d8fab1263eacbded2c520695a350001b29225dd1c79bd22c902d9ce021ce2ae79d075aa8add81b843a1ee888ebe214b1699c8cd495c1e7d22a290151cf3b2b9816702b51a38e720038ef11ebacfc2113cdf85e609e40af958e3410863c444a834cab5a58bb018ea173698d7dab894be9c3238c8338926fb9db8b30bf8cc2992d0bbf6678872f1203ed6bae41cc6e8a7ac7597d66d4501c29c32ada910c92f2e269a9fd1f07d85f781b4f5dc5c927f1004cd8130c3e2f90a666f080abe50e37406bfac00c2cc9e97366420f3d55fed8780c396cbe8b2aa7d4c15487249a1b5c72308ef3fda74025e9632c34c3ce8151691eba0c849e0e9820392bb2870fe4777173856c7b4c08b095d376efb7bc91c89b4da70c466703ea6aa8790343a2de9613a26d9989c64bf10290fe72ae8dc101d7d13b8f31a06ad1b15fcce9c0f28009ed1a46f6dce413f84fa01f9986ad98aef47c07edd66bf2f8b5c34de6c23c4e60d2c7323beb34ad052eda1c6137139ec9e38dc1b263ec410987a3810355d7c0549c2194392fc2dc78fd278496b4b7eca42cb512975f57e143a77ff22165005a4e6b687ce6dc6c67491f9cbd4dca2872082240ab8d3afc9f2261761ddc59d0d99348df871eed121ed87818a7f5297f087392c8591809a3f57e4739f5432417b4f16343c5c5bea8b981fcc8ff89c829e8b956d1b33d83d2c7ac035470d4f6a1d40209f88120f5800abd89441f239df9659b8ac440e43a6e5e96c48f7b3b3df741bf173ed728cb5c3f84d12d3696eac6f54917d57049f52ca126022a53f689768db644b24784aca12ba4323fccf9e8e674dc0666993dc80902d0479a73de12061e5e2c57d0d50beb89a2a9494845fdea402dc7b8c5a0e6b164de9fb243e39f74cca3f845538a6906bc3ed1e6dbd1994593cea54e7ed9654fd1fd40109368f5db698db058b76b4b9507a2ea9cfc6fc76ed7406713db763f7ca9e2ce77733db99ba369c7af7cc495492132f80c9b99d06b36af63b9f37f16a028ce15115e666b66c79e53c85e2d8bdf60064ea9011f688f862bc04faaeb5da108940742ba77c563827dd7458408f47f162efe3e6a527f72101532fc346b20c1fb26ead6d5bfe9411bd242328e073a5ff395d6c494f3eddec192c90b8b84fba169409948c7a1eeb82d38b332e83925128b76c2b9c9bc3d942bb80cb015d149d3170908e57edd44ad78286580f41b4e6c075d79a7193a72bfee6ab64859b431bce904b92123c623dcfddd925bb79f4d18bc17eee66447787d5ac238501c53b5742107d918dbe198073e1860a57a00e6b2b7617266115d155968a494aefdec55b3606ee1228c74f7d2d4ee82913d0d1bc893dd3fcba5cf63aaba8340b80130fd926494086386a0f5b42d02fb81b09d1d0e15c55c9a8dcf912947b017e3b5ea08d0de0015bd106f6481ef567c9d335e0f1ca91be50db2a47050928ac3329fc2a6a3a18062c622411899aff91f4199ff1e9064e9581f26b0ce2b5728c0582d77cbff0ae34311702bbac9e78353a5a94b15e7216640f938cd74cda7629d1700c66ea4fb76db0f8d7c63b6dd96377f1c15ab2ee91c0d6c861043d33bd568e515c78f12296ffd1be9ce674e449867c8475d527d07e7267f761ee9b4418db1d860608ba9eb27043b41628b7ab58cad8f6c2c8e67d74138938107c8f1dad1f64871e9a6606951826181874de815b5001d396390f798782311d62e96fc63b4824cae8f2c1a602783d28501f46d02f16272eda896df99baa2de1e01caf888fb037c9a6dd8f0c843f434f0e30cdd6d79ca55c7e2cc53ca0549fac6be589aaa499249c0c4fc4a06ba4332b3934f26869c5d068ec3cee6469ec25d5cfb41d9d38d8ec6e9da379db31984bdf7c5092d7107f6fa88ad2ad62812e45f4c23ebdbff5a1bb643c359c6200b34d716a56f1b0bd1c200f56e21e24567a4a64b7a41682f3c9bddb45c10dc75c0cc1f00aaf384b942cab7b80b87c8b7edc72f63317707ca843376f0637c336cf6bc84374262800d60d442ea67076ae457af53d861c4bc1e32ae3249a57d153bff59546e3454ae7f9c9c86cefde42de7d306cba2c51ca21387eb14dab6e4f48089084d67dbfca163c9aefe8301321782b038f64d24acfac45eb1024082157789a1a57281fcb997bef8621af4c1e15cc7eea4b46cf50d334f5c055fbe1301558e3850a6a8e8ed9b99d509799fcaff8368c5ce08289d580298683d0652cd936c26c940718d42b2ecc5b4e43a43f40e34856939b36266d31faed6ba19be7494550cbc015bd15306b1caa2de6369371574aae6f4aa64acde396c071cb7b5c021879a75637c7b32621b6b7bcb68a9dba9b6a0bbf329a447ea9d12cbbed14894d69db672b27d1bcbc0cdece7f0680b84d36baaf4db331ab55e60bb8fe9d957da5de53d6e996de754409a904833c85a413a72c9b2fe822e774db206db8c86b14386a33bd443eb5a0178c66b47d32317b4242c28eb90586f83e04f684dd20992b22383010402f04a8e6602f44937e12b35839318863d6559e0a2c0830162ceb62beefd586050225d77ddf2bc063933f57018b636756c19e7e5adf32357bd75e795663a5dd7182394b5f0a6996f30e971a7886ca7e6a703cd0bd025921ff1c9ea1d9342f7670169c88e37623889533f59960beaaa860361159667df25b2b7996d8b19229e9449cc69aefd82f57531d2a5bb3b0d6995036449cfabac4539daec59e2e403af9e405334ac4b6af851327e2e4af5a1299af4a500390c130d9276d654b31ff719d471eff1a41c745ad3a5904622b240408b6b43b6293ad184231809ab9aa2893d17cf2dcc02a147b0490bdbea9d072f7a4020ed2160a11e2e08875c95f8fcacdc7d7fd2e488b2c875452706671658bea7c55886d033bf74ab3e34c60a822bafc306fef097f13fd2ca6bbbc50add539bf070a6e387af1560bcb0a1207aa3c5560214977c2d51272de0dae265bfdf72dab560dc7d44d82326579f91282a85fa9aca2fa4fe89dea92072c05b30fdb4cc94c2cb4ebfd4e35fee5c71f8341090376fbd278ea178c5af8b04c44a5cd275e6e0ffa17c2089b612dabb37ba2d4a36c331ce3a2b50f760e3b53c8f6f375a44cd23b11e26e7800cd975acb325ee01662c2ca4203d42930eb1c22d7848487d9b17f6a05101d0617ccd88da3af6565c6d43f01f1d9dd11f004374adb64e2d4911de00e604787b8b24748fd8c4aea24c3792bc2917319eb3677b648e41670b3f31d8a9a3a046e796b506bb09820829620a50293e24b2ea7f1d8ac576d2646d793e5d2ce2d0be2b26a039866cccfef7f711379fb13387e3fea9db76aa74191d62f7ac771557e2fbfe4ac89ee4eaaca814ac2c70e9b75fe4e5f477b5f5833f88f95721008dba5163ad261ab1048067b84f96849a30a4e0cca7edfb15d16126c4b931722bc66a168af9c01c2b3b043c806d79c9e7149682406a0e6812fd9c1c5bc8d88995c378d34946edee1f22872e8ec44dc1f5e0f8977fab4539da5f5125045b4423c795b8f7a30e52e94c02c3085b581162060c2fe80d89bdade0f792862882a3da0762549bd02fa6b65054ae7b287b32407499bc73b0c0c253499567379f39d9a275f6f822f29eb56964d949cad2406287590945f9a64a0f88a13b5edd6b7ead7b9c18a95383b07a034984f03405f9a8ef4d587394dc80c2b49c88e9c335b80e14ded808c3e8ca2ad8808b913ea1446129d723b82c9d61762c1ff6e7796ffa8cb523dad63fb10d9bdd97b50d596b0dd856323cbe6bbfc2655d8e19f169ff95f056422f159d860f00bc55ca5b157f8ad562cf7e0e1cd90624d66c0cf4be84919b600ec2643c88dcc75c6a79019f04d2e37b0145633761334910de7a34267979182f8e8d2b0adde22d39a2dba22fae9fe2edc4f16251c7a5033808a2c041cdb246601c3fad965dd60ded258cde413388a3bf88f0d9d74d267c78d9a78e42424cffe05faa4bf25fb5f7849601814754d7f47ffa39d42d3ffe6b458f2ae5d0bd3f2a819c4dab0c26cfdcc90258b6e82f2108ed2fe9e44b013ffcef875ab4c88641c6419a23ba13842a3b2f029fe608ca6aded9040ed30e64433b85f3cb68cf1cb9a61f4e2dca6a909377a1b424e5dd4e476b816ae2bc35ae9811fa5a634f1c51450e5f9a96050b60e562c771a3338d72758ce67463fd64ac469e58333a321fbfdca8ce4115159cb1f4dc5ee57868a2641d0d21da0d5f5b8e75b54afb42bc5bde731969b81323f4bff5b79617d5ed15ad8067ecf23e4e8604b37f5e6b3bc97eb65c1b9352e6416ec32cdc478edebab0ce62799545bfbabc01f137f64861132d886a8271805d57589958078370a6735d08a09c0ba5cc487937829b24aac268a53c5e7f962fc76d72d1c4ccd6e1a30e89d33b66a2c11b7d88868b3c7d51261ebcda56a902ce2d4cb021a7143727faae7b6a77022e170175f1938cfd84446c9eac19bb3bf372d33742396f772f280fe3eabea942c1414c09b0de73a0b6ebf83fa6deebc93b1e6876cbdea5b6f661b6499e185db54053018b5f891b1c6c0755bc56522aee59a7140b32ab6a53432177fd0e628dead472477058351f1f77f3a93267d78489ebb6d4574c6a63221dddb0a211746202d9942ea0f4f6b50184008f9acefaa015350ed69d9cf36a8ef292ad10a1b6f1f710e46f6b1c6a05d3e8e3b926041fc76f37708ba65d3ee647a1ce4a5141ca5e5ec53973d6e690d99707ddae01729da6bb3853e6017c4d919e24c147d6f0ba518c215f515d3dab52c9efd032b209b09b5488fc70d5f9380fb1420851ad29dc879b5f0d0a5df4c4a5535fe1a78bd4e91b4e555a1285a420bb77edaab5c2858139ead60ce605cd21ee6e0dfaa5e1ff1ae637cae87b78b75607b11369b64c49381d48c6c2b09173a6e39c98e4a8a67639bf038ec56910966f2ac497863e054698c0e11d375bcd4b9650db107ad024de696e4f4600d3a1f8c294d682f00ca8df0c205b7ebf9b65043f3469431b62f612972a1e57afe9a5e02eaf934c25f2b9cf6c77846ed80d03a5a7de383863814c5031ab34acdd0a8171bff467a3152da8941f0bcf9cc9f44e971424f8586cdbc993808eabcba30f8b00ce2782fdbd666236ced116d67d7f8669ce3eb5b58d1f8bf3d8f17ec00d60c9557cb0d32612922430bf852e3056b96f8256bb06f7dbeceb7db6e1f4231be08e90d041950af2119d5b90ed0b37083e5294892cfca6e4ec094594c064c1831cf961d5d465037ea003da0697b8ad1fd96e4f5cd432347a14fda1f5e2e41eb3619a9c4bdb370d7f42beb368703e3c22de5ecc8e1a43c1c35f867718d659660322c8ab6f1d28522515b815e8d51803379be608566c911908940830d8126d1cb029d0e0cd1319d3c4c9550438333a2dc6fa5c43b90e0e7d277a78f64df4e0167788c8f7c186017b8362f33136c3c3c3aeefe43da7d7103c9d35b24634e094fa9f7df0bfdba14122bc30c89dab08528f600cc32fdf0d30de681c41f25b034890775435d69a8ec76e95c10b608255c01ea9f08b34c823e39c1da9d17c92155cecf207ca2076262f3b3771172faa89d711008f6109907a6e60035f3238dc90e013edab1af3dee9e75e48a66a664256ca23d317d9612284522829ca866e55e4b23f400a73c72abc243261e6ce03c7beb80920a4746bd9defa87deaf8ab520984f656d9147d0dd1a1fd5f4f312fb769a41844bc29e7bf2284b6b956892936a6274744d7cc2ad5a74d100e29589b026bb47935d8516cc9eaee4bc54a45a7fe8acf9dff7a0e72582ef77f0b13bbea1d855faf1bbd03b5f9f2497bd291b413984ec99395db7b1a314973e1364bb448359cd310358969f84e4c3a07abcd1b008ec8e03effa39815e7e0400042e0b143af8c8d37a4d86a6389547e6756c83636f84aa92cee361a2b44992e82a04e4063988b682663b67aa1534372590c8ba481cdc263f67058c1ca8105faa150a6825636a50f48ff9ded797e163d3b1efec57084a5256f1946c2960d46e09a50c8fc9e116a7363e4bf630c9c661abb857987939fa39f2021b5ca5f8f4c0e1601048d0944b6d8ea482584c50d0558cdfe122d5de7e4186de6ec9405e806549cf1bcdd8c72c34f45dc5cf302b40c5ea4954b3c6fa4bff0237d01a26357f61ad01b18c215344f99d31594b2bfb0b892b524000896580cce7aa0d9f848f5870703409b9b0f0fc3cef92bddf2f057c88ffca0354e5ae9f2c6696547d18087c2be9db6e9f0591299066c50343c768d6f88bcfa29034181283fd72607591514f3ba7ea050e301194a8fd791bdedc5785e3a232dc31f65e2215132c6334790c3897bcf13289104df19c8ee4214ee2499d655b5e4ceb32758b5dd4ff48c29ebbf9d63ebc1ca50b2f93e18f13b80aefaefd87c9b0a69c3f12dbfce1be1b37195e3e09e6d938c46426a0601607d9f5c55cf2edddfade80f0b9f598b1d2d16decc181d89e60614dca6661c9b24adb05703aa20a5506a348ac221b0984d01d4c821bada55553d95184af24e53c264c9822685966de51dd655a64c1dcdefe3418a9ace7b946b5e6b57fc749a2358fce9361a0ca5914e3eb7213bc794adc416f6bea1a3b305ee22130d66e698e491622fb1e0c56275459217df8b456238dc4b229d9dbb3bff5447e345c402a22a89356915db88c5015ec0d9ca25af8f913f96f89cea6eee80202f71bdea0cfcc85572d8971c55e7ff7e483466e8e6cdd09775a05fe0b607a18376b4878c28ed0a2189ea1c9c42cd7f04c8ca54cda869cae8cc14fcea320765c4064dfe251e7dbe42b8fed363d31f274ef80bf0ed7878f8c89999a0c46f01a47131bde101ed2b69374dfc821994a83423557a25131dd01ba54e88cc8de622720e861e07800038f8a2ecabd7a8e84904f5baad1b2b192cad27d0a7d277f4d540b127f3d8c04ba71bf04aa3bc6bed3644a52edb794f180cbf1a4e3365144026a6b28dc22725ea106a6313e9fea00b37ba7bc99fa9157950b58bbf15437e8b008f3bc2a9bca17ddc1f98454be780e8c05e9c79e857448d750286f42d898b11ab983e23ed1184488e474f79782667caac797f7de8b3d80218a7d29f5335d656379746704501d3fddfd5151551a36c86e2097a9b77148f84cbc62fc8e1c52d7538b6b87105e1fc210e696ae0e9e341455cf88effa297413c60b75002c48d687036770f21b70e2e366ccfe8ca97c9b0fb10b19147c93d8b2b36c2688e4a6241b9bb842bca0f5b5be7cd4aaab926c23c815f544ca1ce39d95ee08d5727ef29aa2d5c59070b129d4812cd1f41b3078586eb4ecf521bd3ed2326ffac7e2dfa2200de58cf77f0285c58efe561ddd41047f9624f02e49362c0f31ad413bc41044bec6bd091a60136d93e60264dde043c4fe1eb0b371d7f0b383f125f895a36d55f797ba7c36d31533e8d153b700771a9375cf79d8c2f333d05a6ef5031a0a05754e6ad8029c9c5e27e1aa5e81029a8c5c8299b72944b341c872ef799d501f87d9261c2931d16d70824493348e76528492ad6acbc205c5a2d981cdb08824cc0fcff8e5b5d85376cfbda5450563b8f927347c8c5d81f05c4e0fe83ac58fe73a679efb9679c1c41a30f2802f071b49c1e2ce92c56420459b174a22d6d2b93f3bc939594edc63dc6ea7e7d291df8327df2dbeefbc63f75d63a83a9609e3ffafcaf057faaffcd207089cdaaf9913d143bd6270130fc7d62546a8ca441fd788d82878b00e455085b6696bcbda2ef1ce8c6e4ac90bdc2a887804679d1ecf79ffc5e814739104b2df2dce29bf76fb53f42490a7a17394a68f812994254ca295b7ca55832dbd6ca6af1486dbf14fbbd10f1960a5ac183f3cffa3262eb753736ad2d8b3babd2a9933a6311326df2e5b828347f519d91b6cb7e4d28fed9be008cb50d690c4df67b2cbbe4e8e5bc8a560454d5e16525e1089e49e3ff0426d68ebf1e23b7ebc41e315d8b3868613b4c019ea59e597b328e1978f373210dc187d22f1b1c539c3602f280eb6edd2fe9c6cfd1dee58b8fbb5230f99b7fcda08feb0bb39b0c3391362e27e6e19efcca3880e8b234044e4bc5f2961977b308a209b44e7864c569432517c8f0772eafc5ac81dc75f6bdb284c0e2a5dcd38df37fe52d44d22fdf1f269fe6f29df863d1c7b610c04443fca630dcb8540a01d83a7fd339ab8c723cee39cf0a37c4e055d2671a4ef2c44c0695a81825cf17edf27e821157c1fc252125d6fb8264533f99fe2e2dd3b3495f8e82ce51c852f6e9a625c24b14ef9ce8849750d4c3dfca57ee37d583764f74f2e1033d51fa95f4200d87449a73a06cd3c4fd78ce46a3477efba44687db1eeb909f9e96e3f7778c076203e514eb86bd84ea0404ffe43a38cf0144dddde3bfc662807e448636fc7b16653fc6959595a7829d19384f0371661d37c599e4853be8ae2deb3bb812d758fc833e0d915b9ac33d19a7db4fceedeaaed3a13165e9a61af33cc136f61848d89b0ea34ed9c3ecb6765c9bb26a2e2c967b0483beb6c7546fe0b21ceba572b0167c1b72c09d4feb83d7581bb242f96f5f39d7a2aeae56810a2bb66109ea01edeca5d38efbf260188169d9fc2defbd7eef8cc1aa437e6801a42a41e60769b616bec40b104b52a77d79467639dd1a65a0fe940a4c7c9a9c58e7a6805ee302afe36698f437ded35480951185a6e15bc6466c7e42b6158bff6a1fb0776eda745d1535f785f7f9fd1ebdca8442904ba8db4842dc3b4fb6c4bfaae9cce74126b36144f550a42d2825ffa8c540f7c0639c9453a390d64d4d00b50333a43bfe06cac2807c35b3887e06a47c3646dd04c121fe801c4c95767dc1c9d06238751a81da09792c277751d0563846554fa4c25ea0938d3b7849a539ff5a44a31de18835c3ae0fd275992ac41b090cb92ac481ca84f9babc0a4b6193090ddf4dc7a573ed5ffd523a0196c8152ce1dc9822ffc7b4cf33cfa43c877da3310377748aade8c91f0a9db3ff5fb13e626fea755679eea908b4d1470424bbff6ae15c60f8e763115b8e4cf1311cd719f0b22871f5fec86b1b276ec7f81fd5ab2fa800a3d46610e1b7a5c97b5e18f00a27c7312360efca4cad3e39b6be6d94bbe50d2ef7c6e70042d26d4c205bbff0814c3a22cf4a5885ad899b969355dae41263d48e41a0f0b3c14479811106d0ad9351b9e59a76ffdb4c203a5eaa29a450d6c8469c4d940574c213eae5d0baad9bd1d1eb5009b09e1b165724946f0ad1f29f8332a2d22122589da3985cc8949050b241387fa4f5551ea94e15d55b19c09ad2f9c210e227c78bcaa93ce4f03b64064b64778ebbcc9b0552fe9fcbb03b57e2b30b1747afcbaa3cfd5c190ca8c650af6529a53a2248eb24de9a9036714a8e878fe1ab7bca65a3f85abbefeb1867963ce55d9e8944bb4edd1074932fce8af99d8d8a62c2fa3afae2c7537d08969fa4b3b9f9bc77a0fc286427e62d6bcb871f7e0b6a3e602f5a1cfd46fb56c2399ee4e79d1d6ed6ceee594846a1006b38b9bd66a3d7690f725c402723bac144bb4edbb33e77eb3413ddcb437ae4b9955b0f797de0252b9932651257abcc21ae564926c2cd10c7434632faf6805f3adb565b2d897cda3f930013a728678ff2ab3d2ac48fed31f520d869441c8fabd9738465c7ec3686c4d076540302b20a1e5f3c3a2eb9ea52fb6fdeb1d31536842d49e9490200d054622ed985a5b444c2914065c2aac83f784bebe008eabf36b3701f8abb556d7f2e4cf3a53e3c9919b79fde1ae9795244710511c02006fad836e092f4389c2f40db2455011e76d25cf37c42164f63975f36c7c0c6107dd6b4191c3e4e269aa59d63681a65ccfb87cb1917921a423babc9fd23b0778d46f3de3e9520784dfdf638893a3171f0af878dfa474c72e2694dba98a974cbc849d14c94d6c228a9a473e5e28f18b3175332f73e96ed6b0a766f4bde9d730100a8dff2c602bab8cb3c7d06dbeb71982de04e3fa0965fccdfe4fdf992b000a52a2ccec650876b78f2551d5ed6fa39c0f8da3c3c888cf10017823d0666252c1f9997257678c9b7126310465017b190da25acccf2103bd1a8d9e7865b446058e83e11e8ae58f5aa1469b21c37ca4fdad094e7159f1176d4075fd81557ac004b84382d858995ee03a274c94cdbcab231133b1275dd9d7f0a7e21096e0168e9c3fe9352d22b875e5bfb947dff7c4fa31fe791994028f8d0d7bfe68c4f5c33ff5835a9a719e6ebfeb708977d0248b368991b79afb7e53a88defbd7dd6e2467f8f3dadf9d9e80d559a5d4ccf3ef309054a6a17f9cb6c9b1a2f3b4677b9a83661a362be6a31613fdc22b65e36789734dbc2181aac007a4a6867138456802c5a1b08b49c965071bb3c0b534b0b80d181ddb6763a8510acddc0ceae125eea7f5a699cf6d36338d4820470862f02ed2632a9a8f65c3fd514d2dabecd93a5ed16d48ed15ae56c399cea804bc8b205be2862f4a1aad935781d3e24a675509dbdf465128e27a05287f91ad58e9e8ce3fd9753076dc3dc5c25598a959ea9b0fdb9296f6e2413d2c930ec20d0c0df1c01dbe23f6c243a57360142ba710a305c9cf88f046aaa74cf4c03b7ad47b125333ee62774f8ac2ced695e399ce2f242535a063519b6d7b64814a08a7cb6401ef70a5ba0deb4e65c01cef012074ceed9c70ca02fab324d0293ba1b09734c0b30f46508a4f72133b3cd8e8ecaef7ae39ecb970db71db54c626d8a9c5eaf509dfd9d95fc150f5cb3754f275067b9bb6f2401696033a9400c74a57c27dcf35c49083c8f7868c5845cc43686877cbba0e6b2a248fe12f179f61b614cc19b3ea8dca8aa4dd2c1dd61df0019137ffa3c56596dcf70ccc9793b8cf96cd347c1df9b3cb3c08f4524b494c38d945e6664730d5bc9ff55a3173fa83893c8a200f9e7ef8ea0ef8970c4a082c1fcdfe5b4533714ab3888976ab53c4eef60dca384d027678d5f92441c26eef73589791a46940d79490568835c60abd1a92b575466a36e1ba3c1952aaf037a9bd1d31f0e01fac17cbcd229fc6b84ec029c80a5c7f0433f38758abb49602fea39f69e7e3d0a2382ea6657de1ee45d86d7a9be0579b7b6a94075665bc6d1c6af317e7d476981e37d87a1e2ceae602acc188b36afffdbb446e2a2b3dc8ba4c3071a7ec77dc97a9f23b2cd768d33c4527986c0768ced306751976a273d4012107557e3cf8eea583012d0e0776171d1d8da5b3fa1b3a2fdc80bf1165269d8482d80d55c28d7f8e6a41b77b91491c7fd8ce678019078028adcdebc57cd38e53c1036664226cc8505c0824108c6bed3f6a489456f64e8946fab1193fe2794f3ccf60d9ba770663469efc1c9f7ff4434c33275dcfba936af824f4ec90dbe89f337f5bfd8ee8b05f7acf4cfdd54f8d3a8011a4d2b57a32076a88c891dac30bda22fa06ebae91502a9c6876cedb91dd7d9c083582d61abefb953a3165a809150ba7aabbf865a7a3937930ad6ae729a45d394e5471890dc9d6077cd652f28ed91ffc6d3aefc96f020c1561dec33cc181898200e7fc6de1395e8337fba0e0f4962b346e3b16d7d3693d21ca0b5b3b69a9b2799df9402647cc63deccefa47ef7d5d2f4c1eb98eaf6203ffd9eb3ffd18cecea0edca95ba3a42d78dc12af7518bc11eef62ab64813f0ae50a36742cd435b4490f936e5676826e156863f70bf62795d0439c66aff1f13b9e9c047f405b2fda5781f59522b2255be2fe9f0867b6ca0665de9c9268e2462caaff3d804e14fc86f4fe0dc49aa229f2f8b0b8e696045f3f40e702c98b842bdf865653443f6e000150b58fe57faf959655fe00b7a6411a33670f93e2f07a49a5139298768410c6b76c8594d0f4508ab956d9f2c3f78b1aee8bc2c27c598611d88d5ebdaabbb6aa83504ec4fe6d0426d1f5e0d07bb0d58c2b3e82cd49a53df1cb447ab2ce984a13b83a4b80f7a6e741c36277c5bef3afb1a057392e0a244a8cea8815dc5c62e5cb0f67ef1c288982e6f6c4825815debb303f20ef005a7f81b2b0cb7ce6c99510634e95b5474c9e91b764429d5ec1c3edb2e3cf6a6c78df0bd1c1c03e64df102611396c67281789ffcac851b388da0b5b3142ed590aad6cb75e0876d87bfff62094618eca5a5603ac9030f97b74f9f9278c8bb0ee31473650c6cebfe928e1853c6ed4585c114ce525fdd540b0f8d63b9fb0e9920db8bc8b4f23c021838634a166d9fd98bce17a5a0403a47c1d46843e7ee6b78e503fdbd5001f6aa53cd07fe424f10f0cebcdea4af795930a05d9a8d253d749d7fb6f0097cce9463cf465dd55d8a42387898a6dbbb73cbed1fb0faa83e7d06a3437eb04478410246950ee8b3c0005613907bbd0a126cb15b26e449749686c6454a92fdce127c654251b0e0e561dc7cb239cd9fa7baaf8d3befccae1a5776f0c74ea1ef96da2938235ef6b7d87b5889681b3c4c5add9c1bf8d63a494ad0e6c75771d52bb1e49e0c837feba2018c1b3889cf1bd01636f727a88719314c1181984dfc7c4228bdfba77769170eb8189b3452d231542708ad70f4f22ec372fa08ffe038488599116cf6b8f8dc8f49b9cfc1ff6d3e63042c95861bf64342371a8cd8cb8acfbeb96bff467434ef1be80daff995eb3875c7a2e38bde71b50997d39c92ba0e94fb4b39e9f9213dc1dbb9ec66d54cf71c664ac5822b21109fccf0026f4c428888d91ecd8a9458287a5b740297f8190af3e5c06ab1002bf994fdb617fc618e37ef280201f65988194782f33f3f3693c150fa059af611662bc8e39dbba16c49b130e4a11fb4abe202bf6dddf251c8d90877ef73bffd9d73522759e511a9b9ca763f44f3b7f93fda105cde8d0f3da5603bdcf9fba51ac09801fd6e0b48a3a6851f62771a5f98aa587447322639d6fc2413e9ecf6f92493481010dbbb4f836a43f50057294815ec19230913301c256cee6713651a5af7683d1c893a7bbc34c015cade34e6e29b37ba1a04bdde09f2a9b92f69c6fcc42ac1f0169e09995b4ed9e472f9eea724417c7c1833f4f1e5e06de4ceb61ce6a61ddff1ec7099a44be13ae844497a018392006bf1927fdeb0cd1e94630a8d7fbee4e79beefc14b6f747b0efbe26ff9add67e3f93d5899c8a3095eef4be7f0c5d5118ce2e76e2e8c75818688a341f1350ecf67b55ac8a4ed251ec5ca26fd98140ead1c64a7ffd8a986ad176c98ef21085bb33bd73f997ac4ac4735f7acc17271fa13366c55728ebba36001b875e05c8debfad7f3966493068e3b4c9161a7ed88c17bf2fb0c8d6edb7ca9cb6a206b72b90c6bd6bf64bb4cb07f2100cbc5ecf13130701a72a2f46d7fd7aa3ce939c1573a646ba82b06978c841aa7132b4e46bc0f4964b34335317f384ca01be173937253d1614d32ae8e8f91d4d88a0651670da32100a7c284d6baf7763679ddd8d03523dccc86a8795469eff3331ea8bc2358cb91216fdd1c1c76c5c3a06ddfd36d7de36c5a019b920e40d16b05cecd3493546d906b42ca9374fd6a65b7b46b82a6b27039f522b93b327abb5eeefd302d2423c3cb578b003fd9409b0f83d7b863621a4c21fa26f8979a901a1e25c71849fdfb3c0071f817155c48d5b9996a0d42b7930a6305feaa91ce28010d71c4582c60a7e0dbf9dd28658aa4076ccd410b6c85a88066d619a0a566e386674a77e4ed5c42eef53f85d0bdd9baaf2f3504ecda97cb6eb7dac921bd1dcf826c881c1c34ebb440706e15ca38a4050b9f8a5851d291deb51f07eb2b0b73222ed20a4c1d191587e5923a6a6460f8f541516a92263321202f2dc80e2218203db9d4d89949cc57516ab9eecd72c1425ca0cbe900c6255ab7bd329cf5b659e72f86780adb09867427c683bd2616bda9bfd7cd7793116ceb40fb07c331e741dc277c56d3f9795a05b1b7d1c7601a427b009405f062ff9f3d757f0fc44495cbf2f7a2bfda2a4890b47586bc3cd427dcf1bb830a54e62bd4eaa6428937c0f4a5c437abf331a8f423a97849245607d90a7b9c4f830440d67e253abaca9702aedf03f28c2ae04ce4a3282d73a97a92928b388a5204a0a8da4c941850c9938ecfab56e2786f63e4debc8fcca9b0781ea3d562ffb6a8b7a66bba6713385f1b7913b22f45e9072ba928a3e4d8700c301e58dd5649de273cdbccd3761511d1c27d88e32482c1c5c919d62da91cb691746e731d285b7eadd9d523e69e08b48762af0d7f543a976f3b5b8b52464332f7ee600e38fceb7c679544927ae41597b935093bbd302ce064b8f3001588c161636b30b040c6175c45b9f6a4b8b6bb7ce85ae5d5e8f9b5b6d755fd3cf8a4254f1499b21dcca34dae4e0274f2e71d7bea24540fdf967eb7d23097e175a6d581fcf629b8dd6dcfc70535f2ae9e8dba9ed96b7cfa45419e869a9e8c84986e83bdb60ed30d596f288a4ee67fc079db25a50b1364c2feeeaa89bf32ad8349734c53619f7b050a3452e3f6cb835f4d72243a47faea8a0175dc246b62f0159687209e79f658fc1a6e87dbd8b17d9a3fa65fa1589fd0c7da88bbcb6311f83a881b3e0d74fda393f753e22c20b62bf428192d5cd5461632fcac105fe650f25d6b3d9c3fde296bf9d762f8cd82dc2e3ef1cd3284f9959c2cef4f8eacc145842700144b854f4b73ac47fa455a5050824cec4c42f57770e6897cc8dbe1063579d0f74bdd44c33edac72f0b507a05dd783ae9ddad94b9515cc148db187a5609004284ffb5d84c67bce21449dec23e71a6c3f27350a50d2d51becf9bf3567b65937077ec2d0e096a759587865a447c9ec45afcdd9952bb167c04159a1893867d3bf5d61770558c6d6065a3866bf8ba92f835f34e705e74c59675b8d59d29b1426921d264fa863396978a2c809d633937bffd942aaf521dda194576db916010cc3ef4d2b363890c620620e6fad727d1d34e95a46ba7e417fdce4abd58eeb263f0e6d8210cb4b193a41697970ec03f3fd8d0c12a94fd0ff8ebc2e33adfd1fe77783a6ebd38c0b8cbad1b56f6dd9b1a5498ac417da12d6e2c5cbf1dd11dec6b4d62bedccb36a083523f181574e9a0e9abd895e1e1aed690181dcd93da18bc7d778ef2e354b51270186c7c138f23a647c3f7d7db67dc4b1ffee170db10a603a9cdfc90a9dd1d0993a6c75dae96c04a11aafc98afe149dd57634c933ae2fed2809170c62784f9892fdc53dd522318750a0b4fc3801d3b2a68b00e82aeb08704fb9f2c42cea5c0ace6d4f2d4baee05f9c232883554dbb00e47b3608c74ad9038af399c463ae102f1ac7210d65d4c4d0882da5532d1ce86b7be4734619dcb1d5b88ef49de0764370337d243d30438a46cb2156399d915e7ece6754d55aaf412c53ebd2cc60276a03597e7eb773ff8014c98604e0f52668df30e7b88c8c063db2a8da8740c01f0cb4a2bbaa2d8b6af0c8209270a15941bdccf1498e10d6d8043e0ba2ca7bd31123aa1a48f1a54393fcb80de5c16930de81e466518b1e41190e6dcca73e35ca69769fec2802ce8b375a7b85623f6cfb226ffb42ef404ba0e6de45f0e945fe2f21f8fc7dee79d258fdfe0cc73e738c43c7c827d48e7a9bde766959941c86ea1649e693b0141f32763a74cb3b75c3356c0aa31d2f1a65dc95da80fb3e5ef735e2909e8b20b495a35d8b866d00e44102b836267bbbdc4e077e514caaca8c83cb6b41e2a4eef7356a839861d8e5b514d65536197de8ec2baca7e15107a3fe3e15b0bc021adb770a240aeb5c881f8570834eb73170631b4ffa1b7c6869d94ef580054fa9bb87fb14159fcad4a3d8c60abfaf4e474e33e8637aea25edb2c9bfc0ca140225444a6b16a7334fe05ecbaf4d499a6dd8f07e03d700e8b8a9426d34e79ef17d3459ca2c0001c7dbaca7c0a22befc2535019426c2412d7584d0517a69b8187f93561a89c301a7c0506c5009e3eb35299213d1df580ecb39d781bc77d85d8a04c8c0d9cd1a262cb0212be00c696f88ac2550cb00804e9c3e9a6abac799962ab426c7a7178b237a05f910b46a31f3a75c1a1d7633c3313edc478169ceb00f168a5208f61e88e8e5e3a13dad7c0bddca92477ac2fa71015aeca13b6b0c47a45cade9307cfc51b47c559141e9255b990e71f84f655d9db6c569efbc98d80f4cc7fa9d8b7fcce7095e98285ccdd1c82c387c88eccadd6b4df97804cb1a975b160815833d8b291776e3eb40c185fec51c1a8c93f665041106bae6a64eedb25a4a07e2fb16410f6d14283647a12f254733d9cc040b87d09f3b898adea01264de03f6b5458cfae0df13dfbbad3cbacc8653296dc6af7cfcf3d03892100903fd516fa84acbdd053117cc919d71b94addb603eb4b12d8285cddf1e31396f5b8532e475e9bdf4876ea2d4a5eaaa9e66c5eed977a50c469836bfcde571e03cf290c28c36469358a927ddc5c50df0aaa8d11b8cc4cfb193d681ab9fcfe79a58a1a61189c3932f23654258b49e4e848d3ce01b3c5382c741b018f18d8ae4fb4ab02822e0a273ea823cd5d3f3f46945349021936770dd1522ac26c8e181b3302277deb28cade4fc98b63c2f573e0320a3fb9e5494fc6d891ccafbd21603cb6eede9557c85a6787e155f8c7012352bff6b56e49256c757d7398189bf81954de92882145900b2dd2cd77e65d84a77e262ac9fce55157260010a7ab0f2efe6d93c3f9bb89752f93c8401ca1c4cc085b33cad2468724c997178e2ad5c4e3f99bddad1b4613710779842075de3e3ab6e9dbedb63f01a48889300d58885a06f40940f48f879f6fefedbe235d206abbce68bf0bcc413cfb67028db726b7dc5818b75d2c9e9e33db34e410c8416e293cdb25e4613c810272af5678cd930542dd45d08f6ab4d425c6cd2ab093248ddff796c12cf4a7506c7fb0972c501007bc84bb42b2f6026d64be98b30fd47bf84d60ab4d2d90c031c96ecea89dfa3421f34610b86f146e716de8f3bef40f04ddc886d4aa3574a3279404a9406bea943a2e42609d07e3bd4ed75330da5ec46d903eb1029ee4a3dab6232867790f31a2dd688b04353ed04a64a4dc64c386016f32e472c7cdd02794be4340e157d101f07115c9a800493183b7bb98047aa867658a6943b9035f961afe6e2874f44194d99d4bd10ff01b26e29e646e7d4665693304b170d6af6e171b0a87a51dcb6c120d2244db5c8f0a683a0b616bbd63f69aac79256c2b9ac852dea3117e175b8c43571e2489a25afa5ee96a1b32dd90339c7faa70077c079cd8faa65f258889f23b0d89ff04cd8905803bca8f87dfccef1f94334492aa068f5579b140f8cd45a4c5a8ba23b9b9ff1d62aef33dc151200461292e19c2b305c80860afeb7815f392d70699db9c4235804fb8b583265a47338d4fcc28034e7ec0846eb3823b47dc801d2e1e4b5ac4d157213e476a1d48b1b9794da0d386cf33137f2953b8206357068478917cc56b0d5c8bdde09a52f3c7fccb1e2d2407d06f2254663e85c6414252503f036a9f99dd2bdf61d7bde78443ed7457f799603d93c0be3db3e63eb6fbe1037db7623f52955a03157af79677d8dd586c5b4394368119209dbdc4d96a9b31a4f332cc351e1d48d3548ba20bb8c3a15c5ea7056b380793814525cfb6d0a9e91eb4373cb489307c7faf41147e299ed7e5011b44944df6efee3a84a03184cd77f29454b0524b97b1c7cbeee06218a268922539a8420b1593013624d862d8539994e2146dd8a027ea67769d998c3d9988bb1d50ce1896d8d0293a4e6862a4dceb0709da7336a2bda71f49afc3b1c2e73eb374b271419dd8f1c0ba08f8b3bea0a41b45d952b1271a07f3cdcb9350e974fc2bcf5a88044bc3f3668b0b830e924c6af12cce227f6eb31b7904f8fb0eb0820e22ede8de3b9c05068d598da1504ac6b09f91f8f54bbe761275719558de03db42493da52d5b7144e99e68b767a3e02540f93a01106d8d8408fd46238a5042d656b72a7532c080773d099a5b0da978e7a5b7cc658a209a3453f3e578fa2dc54d791d7f919d0452e64194dfc0ac629c93c541032d3e12fad4ead55d23518a423e98e0284bc5ec9ac6161a34bf376dc82dde1a5bf54a204c2138e8cc4ca9dd05bf3a64ceeb82d87ef7860b1c543a977484c151ca48ec77ce58e931eeedb9fcbc7b6cf5549301c0165c60e00a3d55456d862e1dd8e9774f10a3359f190817d41e532b508ec6e546a4d1f59494e257feb5ffd8b9c92e002d839b161a22a2ed4cc08d5269d967834b2ca524fbd1ecdb162d8034096cd37aeb421a6264503d60873d4cdbe1b3516025a93d79f1438a38963f25099a32322ff81e9a0b3f0a37e8d2223394eb04ee84111746e5b77cd48263a5f39a6056e97c0bda23801effebba98909036a4438dd1fec8d84fc388773c308b3f73e3d83b7ff241a2c1f8423574aa9b6fb0838ff5d4c3440acd45c5bd55bfb7d4b08766c9dc2a6f8919f897663f31df71b3036d3c4060810f7d6d1e7bbedf995992e69033ea0e49512808376ea3964cb7a383d37eb966078e1116304f7dc5763971ccd7f4ff878b7aa84ee1295fa7a7e3dcab437409a4c8796451f00c4cc1b32c605e789e971f6bb1b5e4cff4b32d484209e4f80d54472517317bb1ae14b7c222c43bd27a9ee70589c4078f2edf6fb79c8f73bce4067c1d0f42bc91b0fe98c2a1c085554afb2e91abe2f7f1a4a0b3efe4b3399a3ba2e595638c2095405a73a3a7a95c429cf1aee614b6329ab2791b877623cd911305789ddf21f20df6f6d9956494f7dafbe59093be7307d61bd857dfc4df56a78223709961b56b7a3f308c1f958c9352c49b4f63c63d302bb80f156f5bd6a79970cbe409ca480dbb478e3236f4a6089e16bc98445b7a294f0d1ebcd5a72f2de50a0bf9b358d4dd2522b8e594580ef5d8878798b6cc54f8f2ebec6cc514c40f7b3d1c58f32bf2610abf13616fb791b4358746857e4dac66bd4aa78fadfa26e5c1c72e5957fd7e7e4696b398e19dfa8aaab0b6cf9232d1ce5b977d1727342041312347d6c6ac6b54f20b31c489379f802b8806b5c30e558ede9ff571ea70d1efa04824fb388b72864fbfd1c7428f6f04d1a7ac46b645f515ad8d1e8152aa30a47fef49ac48ec80b27a3fde03041e85def3ec6d4fc1355e0aad7dedc74de8cf864e99cdf4dd5d273db65f48c45cb07704b7f850176731417435ce253c4a1397e219c8df73e5970a96e8d813f4160cbb58662a129ae881caad39af716557e7a462b49801d4d81516c43c2e6f1e2716bfa7551040dc8ff97db004e4faa17206939ce337849bc3c1a8776c5c89255701aefee3df2c2b1ad12b57f1faf880b453a310e27b649c1d6be8df9b8a2519f47ea7dc32396e929b4cc8bbac87bfc2e112c4878bfb20b187ab1c6321757cc4496e95982f39de326e779df97e0e2503f91436531c068b69a02146fa15d1e876cac0ef852ec37a988b8d91bd9367f87b3a48101f60ee9dc8200f3425e8748918fadc2c186bed2c1c3435a9706565006036c3ded664ca1cd1b771350c317c55915f849b4b422c1981348abf53f6971962cefd313f734dd311ecb77fe1b964d9d11939d9aefde20ff1cf79fea6008cb7618d1af222a7944395d5be802909e5429b4a7be96f5e112476bd41435b4143dd5b93d9ca5bab85d9211e58d01cfc01ea086470e148c536e48ec1c2ab4e8efe6dccf19b992d156b99e32689c53a90fefd10af9b35bed4572067fdc5a80de23395e58df003b7e764de2e063911ec50fe9f15e6bf84278bd1a39093ee891caef7a1e52ec8025301ce64d93f1509596a113018a8dc169fb351530f35b33139124385f9677770cd6a6f1e98064ce41d9da02487f8665d5b8b9e2d92c38316afd9c93a2575a00df498bc4996f340919fd30336fc2d81028af77992518137841d8abb691e1e511ba30c065cb5fef28769b9762f9daee6153d737dfdd0cfaed0293c48d19700288d895a02bdce38a70de3f5302e8eea82e984149c8f09a0bbd27a85f2a1d4ea83c2beda12a2c711fc61944d6805bfa8efab7a62aea4d161fe279c6bd8c262255ba1bf550be3efe083652375ea4a924996d8142e01e4c85abe5d55be3e0f2a3c59b393be9f6af84f79f9d16ce53fdd81e121db8b89ca6a3309e293300ee700f5227ea57b82f2e6fef01a22e4170f0d1cadcf944d2c59d8210c88d24b697c38296fe3ab0a369666c8b013bcdfa7536c85327e032b3308f9d9ddf7dc1fe0a31f4b154319839fccbbd1d19c12084f0f5aefbb3913f5ad89c6ca17877555a051e9e997bbdcbe7b91cc645f648d9c2abdc2535d488f2f34fafd3c3d3a6fc379cd149e3ae9759728a38487d87b63f5fc0769c4dd9832fafa647ec07cc0bd8ba747144fb4a63562589016c21128788d88686bd5b5428c64962cea870325b2b8f4a83dc424eb16df2b0f7bc887aaa711f145d24b60247429434e65330d26a7c59eb2aecaa7df835dbbee142759728a3a7f75d2d0683fa58a3dd7aef97f72148ddceaa7e857db278adf33ccc560463024fe5801db5525e29ce004f782c8aae874f3dea83b374380484a7f61d4b727f80e632fcd80bd1e1ca383b3a625545620c237e7c821c17723dc11a1ad76816b596effed1af377ed01270223bf21ef0c49df4582f410be91cb86b77a89b3acbc5a6d49baaf0387c9373c8bdbc737414882cafa13809bf52cb6e8c770de60e264f582f5b0bdce58dfb93a85cda75dff498cb2aeb5eb79f7dffced4172efb5d60f0a253f6c4e56c04826fa9d6164830bba1e1bd73fe2df2825027f2fa7b73a10a762d690a837be0e5b25e5bb2f48d650e4bc838c709a1bd98b832213b236985792e74878c2d49d6cc4e16dffe516ac6f71797180f39df22213337aff5fc378444febd4978926234ef2790fb88662764ade706d1dcf33e7cf0241fe8570de24dc5b5ca9ae9b86f63d199b0de47b8819fe38211abb9ad9b9c49b8e50c23c9d4809b658ce7c6ef3137d9b404b53399ebd72232ee0c06871c8ea06ca66ee5a4d62e9fa3f201ffbcff76d21f233ead52936c8e206b80f3b1ec13c9a419c2fcd44d6676c6d3489692a06b139fb6f4d1f79aba2ddee1a3000399d966aacefabb89e466dd26c8f0051224cb1f7a0806fdb1e56e0a7e33cc310a3be50843737a202490f714554abbd272db4d8b3b379fa1aea39b27dcd6e11e0b74695da8f3a95088f97d85ef1e3b747229c7f70119334f920ebe496d6f9a949f50a6dbfbfe44283c6f063e12b352bf340e3cc900484ec42a2b1ed7d3ca62d154c2f8dd554808be4612e01ceced538645109e9fc64187f4087983526e5cc9e116acbb3e347f18a5cdaf498e48494af00c65c2ff709804fdc984edaf19d2e836a1318a3084c8e6a0371c1aaf4572b9ff9bbe302c8ec5d165431346238a8e1f881cd90733c5a69bde820de96f2d300c8e4392e4472348c6d02b2f39bf0d1f601e5bed512a7c748bf700f73508afeeab87f04446063ad4cf7aa3094ab593bb137bc7375edfbbfce10ede55bafa4184139432c3475a2717f3ee01b1ae9100bd8cb63d4bab528069d60224fd55da0ee1cde7718bf95566938cfdf0c6593fa25b86f86b1713e956f1883b0bffb16193a1308d0093adc3086ebb102e3a2c7ac803e51377ce611f2d8a9918878fa352c17ccfb605c5e25029fe607f9cb9bf3743155a18c423d0a3f148ab0f55f08f06818a55bd11d00787b7bde002d7f65448c98e2f71493bb961268a5a80905a6e3659ed2d3f9ed4ff7d5d4e0b4be22c535bcb2548118ffa3d3f58942c8b3e55f2abdf5099bed454fbbc82c9359dd63ed5aa2aa41af04ba385113414d443197d4ea82a5898ee0a60cbcc05135852874e7dbbb85813503bd89d6034744496dbab72f232a03d7b1fbd892a325ea9921530cf5b65713d895202a568d8269f37252d1b62879d5fa521ec78c6ab00be8a12ef968e57c42f349ebb445bfccf7e1ec93b89e010dbfc1795f9e03db84063d86a8f0590dd96333a5dbf10bcfcdc0ba197dc1940b23fee91d6e242aa2f36d1410837fe9eebe092b7d0661cab72093a9f19521c9a0e760271dc254fdff448e97b21d0b3b8f344786da752a6af34ff873b3eaf522aa279d4818fcc9cd6b9c0906b04080c269778c96a472adb6bd5ab4913268c43fb282c86e34f0c9d58645a6db01d90dbcc9775f854505c3616e32587a4d646056b42505d147fc6ded3cebc3fca094ad0c6df86c9ad0c2f12a1a035571911cfb7c902f6c6c3e61e3089fb9ab0d2bbe201e60bc9dec87ee30bedcbd0f125a93e7bf7ba2c2e34b08c4b464ffda4d2e8ec1440962ca5b1ae15aca3c96f120b64baf910a43df8eddf257fc4d78852d2ac21d92c92bf48f6a7461484b5a8db31b230c122f11cdb017a39178608bdb1b7db9c0b2ca5d1387e2095c2cede1e122380b791cd4e5ff9673b7e676b0530167ecd0bb25b00aa88111dec7dd67b93ae16d721c4b1d5358becababa865ba693a0c018ed4acf6d26b7d1019cf9d94e8eb1ac77a6de58f524ec2e36b43cdb998757678a2bedd47820a197678e1361f898aa1cdd51567a796ced4b97c942232a47ab2763f995eb172e09f436ace8c494ed6dc04d0495db44303a72df53a9d45c9a1607a69cadf0b066e164ca6b370d0e6519a6520029c73388bac87322f405405e36742a19aeecb3d2a1a97389193dce80a272256324f24f4936eb58781597235b2e22a7d4217f8a5619b6ea956813ecf2b8d2a897a652127c76632ac85dcf3897dd82e8095feee9dff9e11d26efcccdeea2c81e16d418fdadb793457a2871e6caa36fec9679eb4796543261c05301e2cba34a7d9a903de006344131320c5c1ba9b0d124a95cffa00caaa3d971e805c4d77a4b55d3f17c235edb4300aa45a34180f6dd312544a6cb75b4f66ce437cb637327514e6862f326fafebca13660fe39177e829a33430c659ba7dfede1db9129fa5f55cc524045f85eb57f41418d3054d36def7edaf20b308c8e5b5b7dc50354cd91a65ac53e5d574b904c8e62411aaaa6915f4df857bd7af0609e94ae800932e2fcd9727752ed855812f5a5d4f3b3730c1a2a907aef9ec4e456c8be5807c57b7890e567bdf81251347a4b47659958e04d590d37be16f08e94b5e4879973e5e8ff4f1c2431e6f2c4f3943436162aa2dc927c67e39cfcf6f19c25f29544efb29f81d48cf0944a8a10f2f1dab6616c78678a4fcd7c815f2781e4f5d3ea324451d92b8989821386f353a7b6a13446c524e95e62199fb3f8764d4e09c212d585cf1eea9cc1acf21b66c1000d09bea848174195d7b354d0e71bc36a720650c0c85f56ee0e7d80e9ddf9d20a1088140c3fd14d3763553c31c19ba76ff1c96b0f507710d1503f41af02c0d025fd09e0923d3569eeec83e3de22aa4a064071e958feb62c553ea544a44d39ff9aa0cdc6191bebce388b82bedb63c59af0286b20a0421effcbf97bee0c4a7cab8d7b33d64aa45c7dea01480f4e313621f6dff3b87714775f27ad69b2ba75b75dbe242a19194c4603079685c7dd36dc73c80f64b9e9db94ae9914dc877c638f7a10a71b4baca23dc413d02f74e2c771600164246c49b51d026e9018753fe01075cb45cf9576906e794e13529fa03373081f94a7a9f91a4d2ba44ee41eb12e2189417c794190acaf20a7c12e0e7db64492bfd5628765d1ad5c04eca28adfd6aaa98b911924c14331d9f15bc3c014555d4ce36d32b55999e8389ffe4ab1c982aee85ebc9da6a4f1f7791df3e2061c4995e8ae6ceaafe1026cadb9641240ebf752b6840f034661012c7ce66303e7601367492bfc0e9e17c3a8d60759e8daf46959024234057bb3731a79e6d5952a40511e413218aaa91a3a1a5dda014c4b755127821f090ece4fc7f98cb95269ed06d8804713b04a2c4bf777f69bfe27894615f4d20d9cb777b68b6bc4e4a4560814cbb1c2372425607eaa8de244f0098b5d8250acc35aae895fe7286154848c4c644d5b4e8846e369908fa7f8d705ee4c56f25516b1c78b3933c5a4fcaa014964144b964a8a774a9eaa22b8fd06f5e8432ad0e42e2155a188bf95c79719810b547e5cfddc1600c5297526ff1cd0218b7f96aa78b7a4a6a2a5a7367187fe01296eb04a49e19077a48d58bdecbf640690d841f53ac3caff220000038ad86b5455b3709f62ba04f75240585f5fe77b32c483402cb64e236f8e5d5ecf5d3d13df3ada2a3b62b7a64808abb2ddab7d0598a6c0ec04485b33a93a1ed3c81fbb2b2613ed23ceba6f96a7601f25bc56c98dc5c66d240914939665eabbc4bc0d408949d0dc0adcbec358c060a6c5d6e89a688dc621b866364cf4f345553d59eeb37e70b25070c546ae0d20e5900aacd2bb7c9d3ba47f6c18fbb3a52d3c48dc8dd315a87008d8ff37e11d31e78f6f4f322cb4063c0ac7dcafdfa42900b036302fbfe4632f32f59dc68adc223a7b8e9f19255885d4c91c3c4ec159a00654131f5ec761a429abccaae7328b23497ce09f51e25a8ad01bf30a22c0617d6be77d5ced6ccf77f8661c4a8be88a8ade655fc3d92507f27a2f57ef7e12ddc85f9e9b3396c9241e59153084678c5a1b650052a9bc2c300ff718e7ecfd1557b427d6984715fa7c7a88ffddea68bdd19536b0616befce66472c428a30f740c3227bc4906f38f8a7c7fb09eb8f23db281878e7122f93ef57a7f5047509263f67b7bea4ac1eb8e75b8b7e182e3b90f0d3e1fe0de9fec12a384934dac32098e283ede052893893e7f37abe8a479068eaa28c762611250b79c93af3935d897740a273ee08a7f12163ac0d0976437419ba07f6281f5485e3e0f407ca5eb4c3d821ba6557f2d30560e22216ab6cd6b03f48691429d5d2d3f2ce40f18bd8d9421996001729f50375800f2ee59962fe88a72c556cd820598fa045e912f1721b2b1eb52dfa0badb9d553c75d6906b368336de70ec9229e8a1f1da41e46bb4e8de48707b1a96731a74914450bed2d40afb3eab9134552683496290b81c76d7694c6c41d8c7ca7fcda62027dabc3f4a5a23eeb6681b21f36c9fa9de40b8a6e28625624bd96679d1e816e22b5fa1ceba20bb46f63ed1f99b0da6ef881bf8d0c1aebf0490ac3faf97c641f49dfe4423ae05a1cc7d27f4bd952343447768f605f530bcd602fc2d7902209809dbc84335e7f32c0d8a019a8ab5d96acb0a2ef01e28e36693ac7a1b6a99b38887b5779be8971bff89d9b6fbba60f420dbb03b0a0a982cd5cd6496433080ece719c234fc8169633be300bfb84865acd042f960da936d540b60d4c76003dcf899b7f46bc86c2525b7f2691e1bfd14ccbbd8ed4815e4e9ec21a5910edbe023586cdd695bbd36ccfd6aa0e3df90421612b8527fb4ed084eb3cd1f907e4240fb4cb0b0a446d7b100c9a2af8fa64bae1d60eb6a4ccdb82aef4ef503de652bf88fd6ccb76784540f5b86b3e35670ab8e0c0e4bb24b782022af2a8598d24ab66c8f589ef7d9a1a16fbba70dbf23c4968c1fcf554cb2398370e2d52e7689555b92029f3bd6f7ed14033d7c5fe74729f9ad7e491d983e6779b0380aea650d2c364d4804dc49ad0cf8192a5e01f1d2c1c7a4c13eae942e41ca68c573d7ecee18339fbbd67a5ae479342eba07ddaee04937c4d539438e305950bf0d7fd151ce2fe0a05a76e81aa5acabb8f592b120387e36bf29b01582d1e85b65baa36d875c9ba404b74da30cbbeedb68148ff3657168dbd8a5d0030a487e85b089beaa5caf5c6eb5007c626f33d80097c0653f05f93a7c405834f268779337c3448c41d7e7a81acae2822f6c96c324067762099141a138eb664c31a4c1a0cc1135115783b48b8533bdd256abdbeb4500defcc47f92ff2eb71792887b651d2ea60e2660b7a380dc7949dd984fee3750c16e8e2b417b73adaec6bb43ce133c7a79982a35953234754c412cca46cda76b6bee382f9ab5446cfbf26dda3a06b30e0ae6cb607b1575ba2cd2ef7a010865cd2f8271be552dd65fc37f610a26208dc3697bd93f2e0ede55d819e8732325338f19891b978f70acafe1a1119e6a82d60488cfc38bba745c2e33eed70718a6ac11136d4b87c5e3823c41a34edc89a4ca6bfdb92a4f80474295b3c7c7cb6aefb3bc39133fd40d1d27ab069306b2b5defacd1c4932fe2dd0b928b39a4006759f0caa10ca0bf4179ffc3d48e3a0d99b5d733b7afa283b2e3a6c57982096de43fc4c6133490351876c76bea551f41d23f94644287d8b332e1a9705b686e614858c68a329b3fb20d07a1ae53d1f3e42da877f6e1d218127dc64c817c10c3267e6ffa896ef6001d3f5f46be0d315d1314e4ed2d868430b8fe0c679faeb4fd4f08e01364ed2bec1ec4ab97ea615112ba8e66500855c7d7af5816dfd1db24c636c3631c6ac8c7a6b554ca3ca487135e6c498a4f8c6160788c94fba7ac3cbf21a7827aa89613e140140e21176d49bd76189dc21ea689f999468e5e457f3e1c2b4837e114a311759e08a83416a39502ff926d2cd9f0aa71d36adff17ed57c8cf50cc0394f46ccf4c8052db8ded827aea844ec789ba6cd03c98f7b793c8f74f10894efb62276d2e6c8400cb6c4fc62672b93a9d481e8eb94a816a4f54aa8883e3b0ec8c6bf4dbd5ada94b3001a868627265a11923d95aa1571e545d79ba9881cf05131d6ca23aa382bb2a88e74c5d8fe211b99048a8fa25d63a996185884b45f0d384fd109d4a0ee6a369c4cff4c89deeda02b22a592cc25362789422bb0efee20b6233771b296d0d6975dd014be13a47fcd8cf8e4e3e53f8e3f487664981cca222a7182c113205636d315157cab315f4a98fc0e7cfb20bb0bd7bd3603bbd1ea19944e6e1d54132cc4859b52b5209448414403d2bd533a73a95498363b89d2fb81bafdd03f40776025c7c970487077922ab04ba07130fa2da62922c5ec4297ce2e2101fef05ff61de9c08a1da3bc0a053567a3f9f96de57521d9a9f609bc618bd10156e74cc56a37c615d037738610c0ebb029962bdf1ade17caba9bdb3676d09bde246ade7c2586599a32ccffa6773756401401c849fa4890dcbf022ed5da466c2146bc858450dbe8a40375f9278d3eeb7bef5f8d07d7fe14a571f8427d35d0f08a23698a434cf620e6ea6303731e13b2a72c695440acde36dfbb04abffa8f6b6bfdbc194f5e693fca461dbc8740764eec3f12ef0c10257edaf5cfdd21ca3b889f765eac0f1b2e2c4e702de66ff500f5ac0be23ff9b419fb7af6156008832507fd539c76e03304367aab41974ae2d1f0f82a5b351cfe6ff5c872be1f95451caed3cf0cf67360a47be9870639e1ad16670b03d052d227677d4ff05e0e3544f810db1b7cc0e6deb9e242e6e309073cdc985aada6698898f0eafda3e75a031341db81516e64d0fd4a57a432d84eeb3c826e1934e64bccd96d5e0ec32ef6452449e5e7712632f792b57ece00e8f8759cb9c53da7654fa3609bab5267e54e00ad93753eb8a7ad0309da2c9dd37b9a0e3f8fe943625e3209f1145b1504b5550d2827512d90fb8289d4bb19ef12cd8b9fef137d6544e43174d8c8bb0d707309e31810229efaf45400740821db199a799a2d44e67354231dd73cf5f23a87294716f690651b97eaadfbc074632cd91b68893d9ab7dff47a996884cb53742a085adf08df01bcbde515dc14a82ec054d70bf94e5949f3b7022ba555855f7389b90c64087e55ce29024589d8b728504ab6ac02849bf7b258ca12b40468607ab7346e934d1ad810ee831c9dcec250467f5ba940c8101ef330fc75d35674461cecb18c019164d15d9dd22bfff54adffa96738491d2d155f5e88abd89ea72731160ff1c46615781253991c89154901d47c80a8dcb1da2260a03b8df0e1c6d56f862fb2b211b124cc83b1fb145b5de834ea59ac5febc832910e6c8eea628a188dee2962069e712f94d7ea47cadcd3dba07666ed236669c114523e2857549ec67e1453adcdeed62b833e760adf1ba3b575703aac0cc77e3a2c4066af2c9a93172edb99d448e7a669a8301bbb5eb4e03784765c8b1b3bfa2df522475b702dce803886e86f2a5d15a80c7ada802eb4b2b176802f0881502a44bf749d22c7e38bc320e5e62495fc17a8ce3df90277c34b4b837d2855263fa0b4f434f5dc84c1982a612143a5aa118122b8dd2d461c896bcce7f516df5ee2e4a4fadb578d9b5e69af0fa47b3e8db1b9254d51d57a88f5b7ac39f688aef91fdb7c167651e88fa63d517276097d5d6c082745a44e1ac48076c089bbf96df7e904c8e09c8cdbe6a40467bc656eae3a6b66392b3c177f283f381de4faa23181deb503313a2b72b9a25ceed4b0f072d7e1a7670ecbf80372a9c261f14a733b38df38c2cc52eaced185d3b37476d4be81c776a1e638aa864df743bb841eac31617f01b3928fc2d00627d7629f8d9297b6a6ab43cdb6444cab0b94a838bf2503fac87fa00fb4880bce82f41f44444c0e9b1e5c6593f265489594291e2fdef32675531ec6aa50aba8a0b87b2b9c6b5b2435094601a36883bd4868ffa1d6f7547f4f519292eea6e40d310604bf49c8c2011904f9c7e435e98def319b53e5faff19c017942d1849fd70f24b46e3dd0ddaeda302ab1990b1e644556c50106c1a13dfd8502df8ab51b39a82224fa96acc6fa5efdd8a84c933c1fb6195b837b42673efe8f653e2d21168388a49e2179e6329d51fd40697f9c19f21207dec5bf191789042443fca2be9dfab67a4a1be73dc2a8d76f4216ac948c10bf2c464723e4d7f70e9cd61b7737e2fadae7f439ff2077c3a3756455a04918dafa920f1071e00e4b51f11fed96faeed0ded598b79213e430fca81ec713d3dffb08d29d45b49d1bdd5f8b9df65f47cb1ae9cf54047774367f9f1d93ef69ed6f6cdc8f29e8d50da3ddbe369d16bd56665be5094ff6bcd0ed38d4c87b3f57c02a8a33ea0d5221117a7c6b3e9451d6c3097c4001dea5b7f55c31bb758cd75f0963bf795801f7d793aed76819b3a282a0a89b15fa2469b0642338f1e5e8c91f3b58449c2097015513e263a822e1ddd73f215b55245db749f9772dfae81c9e7c6e6e4b1392fbb813c8c642216be45ee42705198ebe996d1cb3f2a8b43469344440222ab45141db53a93badab7052ddf3e111a2b67c991c7f9d61ac73703a1cfbb497500022211e5f0a01dbabb7fa7c2f9392602d4832041be99d07b37c687110177f38c6989f893447f8ba90fbc83c1b0781f2b9eca6f3f5574ad4235bd3762de061c5c1e87db1bb0e3d023a352751bdcebc79a03396dee556971fc2fdbcd86e8480474d4e0b7149d492c8ca6092c64e2f6c56c9c70c25532ac50d8037325ed5aa57e98f29f9ef2f706cedf30ac1ed066dabb1e38c19a0faa1860d8a496db277a7922fbaaaa2ce0c6a62f3348757da53f3ab95370278075f61e3819ab5eb17c96caafc4eb5fdd087fc0d9c3233ca2fad2cdc5cad494c1c0302c73a46efd8f8ddab60f0a65cbd02b509964019e107f23dc0812bea0d2b23c672aad0e3bbcc2f8a049ce8f8e702f07f5301bd91ebd7f1ca128df776232d1a7c18a860225ce6ebb212b7b6b36777ad7f4eef2733e127018b7f7a7adbda8c4ddc653bf496149763c0d69d574a8c47b4e36f08bf4d6313af4ebee9b8857f1efac70c6fe0d050a89e4850ccd102faddca0468484ed1dedd8387b6ef8400a96fcaa7ad1db538baaf61bfb993119a104867d17cc5049bfdbbbf7a5561a5b0d14307b1be07b8075e87e6a5815ed7f3a55db94ed82f0228c078a4ab9053f594ac989be160057eec78f3697a193863a33de166d5cc8418d08db9ef1bb4264c5a04b3159c3f71ad3718ac4f50300438a91875b556e2f31375b5a743d9b79d7e33d52f66d9a176631f1c92b84653646e17e9150c307ceccd67e72d3bd22ec8d1542cff0a775712f537331e5dd49f4d80a4fb6db1cb6977942e12def21a544f1b0e394b93c19b90393b3f30b930e7195dba5d8ab9066a699f78ccd18fa88b4e0cf9b605f1a3ae9b3717ddee812880f8f79f406f671df0c4236e2a070e86c407e3c858e02ecf1de32c3f6313c8da5403fe6c78ebeb4eeb8d2036314543164fb546e041e1122d75c6111c0c6e4061aff40bb0c440acd25d716e16467b8ef2c61eb5b75143fd7fb79fa1f8e57bbc3f762f7c8da4377aecc351dc8d33c1b416590bb1dc456557b30f9ab629dd0b854638af10919646ca7c72014bc655ea909b00627030db7ca6bd1037aa40b55da6a667ddd4b84d27cf29264999385570dce53c4bbd8a2c11dd9dd3684ff5581aea0de1d7bdff0235e143f7111e2c26b875509bd84623c93fa16173c1a6e358683e12ba876b0735034f4136a21bcccfbbacd56691253f84c3cd95330a7a5db060469319e6f188a4d94b03bffecba14a2a3b3215a47d0458dda3c66ce92d9057b06eaca7eb181e1e81f85a1f0bc286c9fc0827818b6fee4229e084c1d2a9e0472523ca7f53706b25890697ce8e4671a649e3e891432f50abecff9ceec2fc7b53706414b6633c89820a44afb0c7c8dd9cba807232e8760d495ea9f496fb1290b72d649858a55dec12e26b33013f45d124b322c85fb848f5a1533685be117c5e3916fee70c4a1c306577434a7c2fee6e35bf9681870a8fa6747167c605052c210aa78f87d0d1135028935aa0fc900fb5c121f17b0b2801b14153f3e554b8cd61d9dbd21f132297035ee0868b48297e54ea26565ac2957d703eb04fca40417f6874f47c1baa7a4a3f7778c2b0e324190b32a6f15be90e6c276657960118dce0fe5ffb76a37b52457668d90f802ae21f66cf1143d00fda9e6a62fbb4295a0c177e6bfdea46eb8d48a996b33bde93197d3c6840b0377e69b73250daa3923a83d6176c62fc47e11bc820047fcf8fe7aaf82383d7096ef3187578e96ef7f13e364858a224c26b5382a4c6bcd0c4a7697bca6fa97282b8871c92ce0344e845c5deb9a73397e6f8549599bc587eec43fcbf7c42efe67f7f0d33792514c111c89602c9f2db399ba4c7f0e15b6292912705254a53956c62ee65f7e799fe591d86a1526c5d85d1e7aea59c2170d31f6864e83a67fe6fbce38e1c35e04b8c679880caafe586f0ca99b4b53299f7439922ecf809a735e01cbab1f7777cf92b2b37c5c7b3bd5913def83e25254844331fcc86c62a0db5d28322cb3b02415115056b3c0af05d9f4592688b70b1fa088f2f92954f8a7da00769e0831e07eb4400d91e959be9187fb1574902ceb56579cbfa614bfe99979dcc5242a0bfd165ebc00d14e2282123035173b5d0c16058e9163368cae371f7c2592ab58c4960492586ae638d3a1b1d20492c1ec1fdf2b227c7cfaccd5134589822ca09db151fcd8ceedb7347d2afe54463c6acff14572ef1305c8445e5a28ec5fb2d365097edfd0ecedba2f45546cb6d1327c2493531b9312a2eaf7c3f1b611ffa13213c34eb5e96c22749c8db110944415858c47a1acb34f376ce58bb04065372f8b2ffaa182ba72b41ab5ab197c56763d7e7c163da93cd2896ed8bb6a662525b5843f5d255613f9b4c69bdc4cc125e9820cb2fe9a22df38ff45170c2d0aee0479473375b84ce66c585b7565d36b4a60c91f929c06bc927e30479a3126a9570400cdee3fba931d73f7ad14fb1de20b4b58b12802b1eb19db1b44f43013b3a9d7fcae4e6fcab5bf5869a77aba5413ea0b622a83aaefc34ce324a62aa2daae3dbbc33810af9c074eafda2c619f4240e973748093c7bccc017c6c9b5c77c6efe385be347059feb2e6646c8a47f60abb10802e222991635908435e1c50ff030a5415102c206f6ab262168f55447e6caef3aeb673322f1664a48d38550261a1fb92df30557d15e93c7fc13f907984e346fb0fe8c5f0ad062c780062aae131031075538af0b2da679fe56eda19c3c26860a97c841e74e5cc01a54781b1d55dbbedc502a91f42a1ab25d7b8e9108dfd09e1ccdd2a153de3b4d9c3973837fe3109f0d04a52fc9e40e37b740caa53991b04c477f5d207a535a90f5ceca02f957d50e5b0ac4a6fa6fc73fd8ad291943a678c8d6a66ace7333d1457acfaee00f6314ef968d117edb3c900b7eefbb31a32fdd167758855dd33a6494834f170973711624325421e58d464866cd88e06eb47b77a388b01324d85aa9933589f3d503d14bb28c36c905b93ea01f3eb205e9079e75d43a84b01a7ddf4da668a62e8789394b412a07a9990a6bca8fc2e259cdc228b091a6caa7a11f2cf95f84c371e2c8f499279d3f93ae070ac6b752988d4055cdea1c682f902e76d2547b82c4704aed9eb78a69258249960321d124058e71005f936459327b2ac38c3f845e45e57324d876029538d198aa90b671c85e3ca9646669b5a35f9b65e77e1d0a8db2e4af883004cf054e790532a10696882b596cc6680db5ccada4964e07a981a2e81adee13dcc95176372eeb42afca082dc987f72868f96e668c76d5ba022b53075bf9782e96cce029d7c434a2f6eb22e9a20ecefe8be0f313237368c5df04dffc74f2fbf4d5b646275bd3c11b1e75f5af0e4af6c75550d71aceda8c897fd963fce489c50155fca1ff9316d6a9ecf879c0a8c6357eaa6edafc4b19dd9e0557cb8fd4b1f7dc8f73795a5118b0ef09144c273b7aa7c046ef580fc02e01bd7513a49c98bded745f8f8f0e280d289d07f67030cba3d29eed4aebac4a0209ed7846d24d8e1867e4dc9dc154bc6cde01097b0abc79f7956c094a413787e59293b1b08d9afada416fe132de4e8fff1b390c601ad4702267560de4b46071617ef2302653b8ecf924af19296af582c48fd39dd18fd949209a8b66af5d0454719eb7fd9925e8052a58f5997b1b4e7682248aff777e149953d360fa915c4752c018b6ea206a436586f7db65c06474a1dd4318819858f92679e245d78d9d3267f1356fe59749a453a23031971e7d54306cdd56115111f774d3e6dc28718eb4fc67242219890032da2a353088f78d0e554f60140d3eff7b0ea449b2e8a00bf08ce44dd1ad8f7bbc910b4b63fbd961ad7f07244e1fa75b82dffc757d99f226e459b0ece32ad72e8117725109dabcf5a2832aa0376b2ad82241a2e2c71d24d9016bad4a7a11b4562ca5de22addeddbffe25f60b27748f107048f4c80c631a1c7f74d93143e8bcafa43ab67b81ab270f9e2f751db7d981d7165c0567ba209129a14e14c33be786f1e408db3576857882d190c4ac103b23aee54d01ff8035d09233dbdab6ef74a8ca6b18001cde12695dc373885d28a7ea081cc355a30f505a5035535e034877f9f9e5e8ffe46b589bc292ba9c624a2018669d30816d7134f14a14a9cbb52e8e28b4bc4141d783a7a9db28cb2f519d36991deffbade2d28f330895d8ba366e8febb03f4656362e857fc66336ac2e52be64e098987e7cb9e618241c3e2a15f0a238ef6e17d0a6543daa21941741e81df1a6b9d26783b5b8a6066844b3e552873b1d3e26b8dd6d8a34f70368b2a1496122c372860985c85d3bde89fcc35a31bc4bff9b839fcbfa5ca05cd9585ec3fb69cec3d3b51d001f59a9409b313252369fe2421ec80b08894e2b860fca062ade13e608cf72acbaa74f2860d6b7311c9891ff51f82099f7388665df1270426f37f5ebd39809bb2611f776ba6f8bdf48ec909fb67372392b38be63c64778e9a1a9f8cfd6d6ab320afaaf8852c67aad2218e6e6031c331e982458ffa6272f739ed8eea4ff64bb64cece827498e28e63bae98d4c29b2fa625c63b0b06215e7ad3ee6f0b088fccc5da125df6fe1c0edaab6554440b5694d0b8ff5f274ccdfe0839afe53ae2e1b011efcc380a6711d41cc0e9fb97e3d5708b17b74792b12cb0f2bf7dc4250434986185b3b8aae9cd10c903a06fc05eaf02d50268dfe2c49e633bd63e10a8971a25900536efb8bb2215b7cc442338118c066d07e9fb763f3a4c4dfc806fd9fc1debd1e2820b79269b16e0fee144e0ba42e7ff9dcc11754c2a50782f3ab80d4a611a6db7ed0fb90b0c74c50724295a3de80677717ee059ee7adf9ff83918fa4870f77443c2079695f7bf0d3acec77e830b4e16b3c7c5e35ededca641e54ddd62f1ca9ecaebe6ea326471627baacc47ed7bff8ec56507ec4bbbb0e599fc27351477babbc7ea82a6dd25393619b2b45e872a57b2e6a79ca2bf9a794ac1a104541e87e9e2f55bbec3f489f9765f9f11ccbafa1edae468b5a507ff9f2a4462fe30c44c1551cdbd92e0f3b9acdf90b5e6815771b4ee67c2223d510f1177222b1100309b33ed05f0ecaa858add62e4fac0013031e5c2e435319ee4e09a425ebc597b34815ff43555149d71c3aa0171898d6b3a3263c96e698aeb44d10b35b1bacf1b929877bc5aa2290b30e86a0d739b2f926daefcaad33e1efd125ced679617d099220f23f98c333cd2f57c42ff592c77876b1320e36722c30e55a850763ffe9a454aef0fe66cdb476964af1003601045e2ebee21008817235a7b9b806ecccbe7dba8c3efa3339ca7e9c9c4f0e9d1c0c495c4068ad165234c04f10ec4a818a8c4ebcbcb7c793a8fa507bdbe1dd93e0bbd5162cd82f8e47e521e62d79055b27bc74b48b16d34f4655491f35af6b952b2f7edce209e90f6cce2ae1099c29b2e4169fa8a29e0760d63b1006c21f47fe080f593ab1adf43a36e5e8841d756ee10948a5f62370f02ff43c6f287c26e33aa1461fb88651fb8a5cd32eb296da1f4def54410816a221c8a8ab9a6b94ed3dc77e5c3bf25905984c9b60902f4a2a179d551f144631836061736ed32dca61844f29060a2cfd248531cfa52e47af73e255cf7b94fbfed7a3b514f3e018042c1be3b6a4852f486e5924a6606855e68420014b00d2c95df93653e047306e581072205dcdf009488b888b39f1fb389637e602e5802f4224afb05f5de6c04d4bfe1034cdf4cfeeb0217702698fe5f5be3301dc8f4f8dc7ce73db8c774ba7882c8cdd3192fcd160c0af7c636d29b347166bbc297ea5be9d1e0f2d4faeb15e67033dddf408f78252cff2c40fdd962a737e5b1c7bf092b85ecc5957867f2b1328b37472922cf01517e3ced3a948a7cf38d1ff6507920dcd1574b5e886742eae04197bc0c0bf7b5c5a0c8b541fe47f817538a433cafef68f5a0809f5a343849c96fa3606e1e9e6eb4298c9716598b2666fd62a8a2e02cffd91c5d638407cc5424558ce99bc5180426bd608c09e62aea8035e2c363ea46bdbd82fa973e01535e36684e3ef30389e48fb3f009aa319a6098e027fac7ad3740fd63df4dfdd2d3608ffca2979fac8a55f1126f5469fce860f791e976e07df08cdcc714c3d8a2ab71ad0064717ac3a964b21bdce846519d4479d670a0862fd4814739a77ad52b34e03021504266d87db87ff1dd182526da83ba230dfc525fec956820031d8e02edcabdf649418c603cb919f79fd98638eb19c8dcea8f2b44a99d85aef7909cce74305d6389dfe0fef63720e9f0b6f4b9d0a004d384bc55cf193bb2be1d962969862614a7d2abd95b1d8ec554639d1f3097ad110b8e7bdf5c8d3a2853f3ef1154bc2aeeed985bd3d2f55c23e2ded7486c1f2e940599ae702b59d39dd9e48d772205b0e0d4ee4036fac4c40108ea5c28d5b9f20ba4cf785295ec025a92254168c683bfcc02bd44fa6af42b4ea8827b79e094c7cfaa6fad6db105dc3e9e796d5de0a49f05c489a776cdd4cbad3984b4d558d45e73e6b7e973ae1b5034787830c55145c87a1bd210c06b4a75e13746f2a5dba4a44c88c0b6878af7a1fbfbccece0719ca5e7e2629084298de508193e2afd57a2285f3ad81d7e7ea1ffbcbea55aacfc994abf8725e574c82172d1951a73d8f4d886b64a6bc282ed9bc71a16d4093eaf38c68082072b44d142256c120c02586074f9eb191cf2f66dbaa76833a53b97e5ae9bddb331bb1a54a5b39a901d5f16ecd4e02a50a7fa78c4862c56f4245bb4a189bb97407c6d530b0ef3be960e7ffaac69af8e3bda1dbb35ebcefbfe45d20219c00943e073cfc8f6dee18384ab190d138187a7f0cf4bcfa41a8515e8926e7c07c85e9de57f8604fde408b9441851fd39c2ad16a3e7afd47cf06fbcbb8b1cd41730dafa9236094791a00c461d1df370498e7de79fda42763d45c941a1fba925ad583bcc509a638ff97661646ba9b05c64d3bea66db0e39d8efab6d062e583f26dc82731cd973cdf2faf34bed9f161cc5a21aa6626f23a8cf633f00f7371e67df65b3137ff9faa427c01190bfd0f5bb430b4421e069b4ccba9eed6f3c084b73c548478fb24c4dfed32ec50810169500de01e49f258a75d7ec89579a5bd7add11df849f50f04cd93c6d6270a9e6600071ccd7f49a6a9a26a704959e2c2f54a505b7dcbcc1a46d14839ff7088abba3650c8b8e94b68c11c9ab9c29c6b3c0088f8d09669aa16222017b5c6fb67b382233aee0b7ed8eca4f6f93b56814c6a8091b6eb2c507322f6bd5bc95d818c495c285ef485f7d643fcc6ceb1162a448576551236bf1b826825cca84195f5206dfee1ca6a9c070d24bd4f6534042309029aa6bcdff8e663270cb474e7f79da6bb55a5be4733e202052625d7d8cd47d573823b7c2d3cd82d0999cc54d1e3bd72f6bee6e94067b28c02aa5ec18b50ee5bcd68895832ae1f991cde7b9bccb37cc2551ef9c18af6f0dd1a0679ad07f6408af3cf33cfe07a210747b90714b8a84294c486bbe1b07492f1e499b8d7071aef05c93286a5d0633ed65a6e3c18274dbab1e280023800ecb8515baa7531367d3658e76c91817b51b121d6289e4a34789bfb8a79aa991a6e02a5ec667f3c0f4fe551afa88e22bc6ccdc778dcf4a79be04abc101dc6d672a48e4380d44dac7758d29ea8eb7ec3c5077219035667a694ef47cd0921bc65def419e9407877351e285f31de4a12e7ee58162d4bebee81f4c553b32c3b5bdd5024055783021045190ae5cec3ed4f60e22941027130334815ffcf0f74c94f27255419da87e3aa659dc0c4951613541a90d2c93121059a096c3e6ef2ee55ac736e136b6a5399963a544f30bec5009208abc588554e79260768187f12a2487f8308e221fdf42281ed9638f4babfa89a2cd0432bd41bd4462aa35c8afc0a1dabbd163caa15b339d552f017ef21e9006eb73726a6032b115791ceabd719967627444cbc23d32e081d07e2268ce77b25a2ea62e513fe667e973b81307d7a99230873bbfb27de9bf29173ac5952e230e87d7b77a2d05e7ae0c014d81d7d420fa4c074a5de625736cbd0c7d3b232614166f53dad156ca4c65bf4e4b774a0dd812e58667eb78b3a58514380e3a308795dea8b56398e83eeb37a10c0a6ed631baa6c1b10c4324f6f1e9a7c04380ce8853befc7747f2c4f9da2402c30820fa2aebbfbd94f14441dae2a3349f85e13912751320c5ed6bbed2e6c6b16b96d2bc498fc4a16af3cc22f0db352d4745f02c789a9346e14e7ce5e927051e671a696eea2cfdbdc7c2575f828cd23b60e9ac4a27ff812db642c28eac9d273d5bf7b1a13cbafe22a7fadc87978977c89cc201035c1dcadaf98f4edb9843528284ce17b6548bd12df614e4b4970d3d60a6866a34446c72aeef39c9483a4a44bcafc4ccf59fc336290a94bcfe6d8116022b296441336a79c5172ac6fc7565f439173d99a201caecdfaab2e3cebaeaaf05f2d9d7a540ba2032373881b0be66fe91ab340a0c6f221b428d3afd31e82bfb6ab920a2b3278ca72522e789ae44bffe55981043a96bd3ae908eaa8ae61497653b8a79f9dfe472a7decd3dff7e3539c003bfb416c4a00e59c9dc9175fbc51acef13df0dc7f4b36c13e69777267632af829d2b07d2afaf38ba429819e00c4b939e68e40f10ecde0bfa87aab2cc6e32d30f9b4359bf8f47bbf1eccbe1a4d744478efbbe98860f9654fed9b608f9ac2393f233a38221ef9d8f7b3ed6628d86e1ba6fa56773f328fa9b6df20ddd146031d4928868f38af4774e1ac89ef7ff03bf2fa3246ebf2c8fd5d2427cef151f3bd5dd271b8ef03ea41b892601dfd60e91fcbdfcc377dcd459ebeef3e05d74b24ecf2b1b5f7c5ebd56eaada90e556ba177e29e0e683cb2254fb8432b3304fa9bbca1a14bd17b8559b6cca6de0a9e3243e666b00d64be0ada1887f31d2181b3dd9da134f70f79288a7361ac88caa11aaf8c0bce9c577dd80f2d5edc46ae964341baa5a83f5c9d59a1d3f5ac727717e0bd0ef55216f0a886d9b00831b443b751babed8ba6756f222864af208ca739903a45fd71f178c6f240bc77abbc4d81ba815b43fcc7d3fbc255f735f4b8219e9a463b55788e4dc11e5d656b1a8994dbf4be0ce71cc0618023ed4569c65c4ac0420803685125031969827aabfc234126be5f0a68636308c91e2b517cd444b71a3d3a3cf2d409d9ff6ce07571365c2737d7903b0116a3bbf46fec8fa279b45250b01efe1fa4eb1d2171c8b2ab95562c23345a227823c43cd0b43d7e9f76842061ed438f485bce71a6497b19448c8b2d05ad3e7b17fe5ed8d3edbf5911849260065462430dac225517110ce07cd6375897fef8bd1c3037c6d9fc12fb3c2d8a2a70140dfa8cb1a1b3dff82087f7dbf5550aec0cf14ba05c484efc40c84aaa4b19b29afa82575e5c07e014ebaa99feac3165bb0feb7644c2d704915ef4b14c4ea43d4480840e499e5bfb4073afd5909f21899cd7008145415e161861c82c292285666d915fd56b9d4126addba722188b109e24ddfe5d693f586523eea058610a356d42e2813bb3d07bf628200dd55dab85bba18aadaabcae2434614fb3dc854f89753cfaf1777ab820519aedd8c3972bb4aa3d5d8a428c3a91164a441556492acddf279a44f13531ada4ef6f6b060efb32f0c9e2987d4a79c0b7ff65c419cf27f9d06de29d2fc1b19df472367a358b3d8a3f18b7fbecc62bf31c02388dc4fe3977c213a668046fd66e61ca8638b1181feed50cf641ade63de5b8822e1cd9b5e4c77d577e7087b0c95f5af6176de08b8339f585b31de5ff9070ccdbfefcd5b33457b4c4e0c0d8634558ede0a8959ae30bdafcca6a1a9ed710cb75df86e15ad15715c1257d67e8c7d8c683daf1ee3c250aa74a57158b38eff934f02e6da7e52fe1c2e566233a9899327acf320a3da853e997f63249d4f717f7985c38a237819050f2f8236af39b74f35c215408d896af42814907f0a2ed7fc8c1b4df29ae6dd9ef353be3b845fabea3a167d2812479e2c2627f9010200285b0c282667b4b4b77013ce0f995aaffcbb7f67467e35aafd7a97c382be272655c92344b119a226329b5c393cbd42fbb42dae9c2f5584bcc8e81ac972fb6fec05857d24d24df02d86764370c6f25d03c488e53471c3a627ad89abed40b89fa7c908ee1e8566090d2c9ef65ba210b0cfba3d0bde056a2b117e241ffdaf9862a74108fd6f06d4b29b5d10ed6f11a6beed86bf536eb7fdb911f0d58dd15cbdc55704e2b0229dd9c70e07c655cfbd53fc0858fe79fca87ac5d42dd27a0940a37cf9451c29f647b3e83a6093d2a78eaeb99f3d4cc6ac58f4578e4b001d4da27fa6d7bae18828ac9807b63e2519851632eb486d641d5b6936a5df33ec4f951f92c0547eeb76317b16e236641c12e5f0f60e0d15c52c4663bf5ed02cacd13d4e45fc099fc02c0f1a2da05207838dd5eeaa370c7a0e3a5363543585976773b045e415b2594bd0fec4ad323fc80248eeb81aab1a26865b0cb90edda8cef087e2dfc8fe873628869aed47c2b4baac348f6630b06b90a0b22e43861431aaff4a8f30a3f04ae188455975212a116da6450da8d217a0e76592a1f608ebc8292b05607217e15b38458aaeac24f80c58a5043d886cc93eb7e768c731b9de9ff6e21e98ede97984411d93ba13304be0838328ccf93b88518ea43b346f1badff8ee63c7a1da219c9b22ca743da9c3b7063b55c19894cdf408b098427c08da313c06e4c305f23887b08410b769b77590609e2cf5d59866c03f27c10373c45e97c511ba24e92992ba084289f50f4962bb3eb6d9af298ac54f61e891d9dd5b9cb968310d514536b4ea33c15e5b4c884db2a7508171b95d5392d8b136fa1aa0d71ab9b8ac2be0bf8c8ee85fec24d679ec354ba3177aafb2617c8aa483c29a8f14a377058e1539e4f31207728dd844ce54c0efb83e9b63c35db75861b4845ad299c933672da00dbb2606d78ef76b8eebaeb2089ce3fcba660b05e778d7adb3946e16c50982734e0e0c1f874d53bfea1ae82fe9e6d948d1c1bcf0f942a51dd9fbc9aa54b191f0a21ad2edac5f59dc6282785f67a35a307cbe27255a11cb8f4991865d55eda2bb6256510db231cc9c42cdc63d960245eee56510bd2d6bda0bc70563240ff8900bf61a4baad99ca53d3a3f81a2e1d07ffb05264a6303c6d94037b551ec6084771bbd8afc9227742550ad7383e1c3aa8dda5c44f21b6e34c2c1d0ae4fd4159c188cbfd9052fd365c0f47039d00941955a31582255d77bcfe896f1caae7843658266d292006fd6e75dc5c7b1f5f2c0749fb1ec13f97c652dfa47509f1370b3010eb7abfe0c1849c715b390ff3af7b36b6cb8b8054d4f93a9bec41fb0b1a05af917c5e5be16b77e73cbb9a758f1a1333b13dbdc8951c5bb122a6269860803454b1c49bcb206de0c51ba1ac13b424214d8991249192d006d63fd0caaee93ee6bd52b2de3d8d0d9bdce25774323690ec5a8b68927317f1402b8abf7b5367e2e69eab4f505e1891b23834efe9243576173551381b81e96faa0a00c8beb42d2f3096f36e8cbdce079579e8abe64e7798b9a78e1384458d2b560e6bd50fb54e3ee8dc6586ab5a4d24b94d95a9a5bd92fc0bdf4ba938e39b667a02be824e016e265ab6aa6bcd9462b809f5acc17b3966461ae704cb33f51a13d8924893a6826ef97ec7423fdd0e728e1e5534f719ec2e60906b07438f9720ae23a0c2db1d041e04c88e16c4ab84c099178b9549b8906e62873732a2fce2bcabab3098884505805be81f0455354bf4fafd68bbb37ff90bc55eda51a75bbf1a9ee481b73ebd147747e4369862f35bcbfe0ecfab6ab0d9d6ed4766c72c51b28b08bbff76419f8c5b3b2b4187993d6425e2eabe00a8f8cee02e2e44c0bdcba756701a0121b6977197625d58b895d516bb41c7e604752bb7b506e2ad09aa67690d7409e8510265ef7f2b3cf510240b01f3efa7fdefae931af5b6b3c14b4de44062505d11fee1aa486bf714fdb45469e9d01aeae25f21bd65e79902253bc7dcc40ea1e1f03e8ba797476ff5fbf29c3a13c807e3425e3711342795ddc2ed6b443d2dfa99d0971e1ebee0ea4ce4ef0887bfdabf43156b3bc733d2f05f016e9059ad49e6c9682f8db3b8892a0c227f20d567e72afc21e63c633dae5a8fab9016f79b9d4997e090f3d01253cd041b6bb1dd8b2b696d44fd9a49da3ab476768f6c7ab6180d00917ff3adcecf87ef1fb503d819b1d016e2fd037b179e43b6d84c3250f6498d23977d1daa7ae3f0da9571feea1d50cf84b6c15e4fe2df901779996a8a0ae96ff7e4a30f2c5ad6a8dbe4b6bca54ab6d12ce4c44d585badf8e0c5519bfaf53792f1c0604dc5ec3aa37a71ef55c9128775abc3073910e276a68946180a8734ee1d4eada92f4de8fef45c3b5bd0197257cfbf82cc9854b0bd9cc81e384f168de7f81b33c0fca0cb8de4b212d03b03bb8f4d5af892dba8e6893723959bc87d7ec26d8b027940e51050f6df42ac4bb0f9b925f0e150256e7a3cd79e44a49cadfb75fd41f9b28084619fbfd9808128442fea0f36d1356c2abb0e925608f1c0faaf37421c1a9a76be5b94e3de8c375064ebe99f7b4323cbcc70a5aec7afe5b4ccd581de47f0000200f87f6ffec9903fc10b2b4199716d3f84c80ae524a69307a6cb589b9decf02c2c67ed13fc36484953a576300891f0db316962665228bb1dde9890cf6737ee7362d33fe7c21c104a644a4d8c960300c5ee2ab3c40bfab1ef30abb66e13b4cd419f37a2dc91b9993627f8f1f7540e008e3466ca6c06ce4d7877d2a2877e50d2631ac9887f160977dcffbe51c04625e7a5bbe35223e2ec3a5fc8f9274472f1f86d0bbce9a6ee925bb045cf492be7f90774786d97eab881e1fb6d6e810de8d3d341fd6fc9dccbb969b30665729c90b2a1b21cea88642a5af4336f5e17f8eadd721ea2a8f6aead264a93ffc1a1af272ed8645e7e03e4f965f6b1cf2ab9fa57fee9727bba0ab4b20183f2a3f4f5561414787488ba867bf29a0bb90379db6d07209512703e97046eae031f419d5e629bf16d914ccfe4f1a913c7d7b6024f9257b79f443e5561eedeebe577a12ce35ccb3b5b8477469900cdddfffe5b50cb74525ecf59261dd33b7c31f32f9bca39f0c4cd40cb850928fcc8d435b0b6ce9c5e21ea395e909c66bd7e9ddd89286efd0b049451ef663ddbd873697910be11a23f89d6564f22374748a9c4cb771240d9856e17d0b036c5e2c0272de4645ae4d5faaad100ea45269f69d0527c36b1e04a586cf78ecb12a884bb620a6809e7cf566f032018b8e78be236e2576066712db32b40960505201b1e9b2eafd7423052518e09413465112ab000866128179a6b8b9fe4780a69be9b5baec9ffb486d9edd237eeeac249207d9a8255f2090bd0700e8d27dbf620a3669a2ca25ae25c67e25283390ec1fdb573cfd398c8b245d5e8adc55ca3eae05a1e3148459a2b21f9ae9b532b1ff9da2fa4c248216d419beb7bc7a35c94e4a70c0e1151a170c17988998829ab1c8c5d1b9e71fc9e78b2cb061f0571de01eaa5d7d3205b929c0495757f7bc0cd3e6b24feae6573f51667bfbffe3e674d6f0b7a369f2180f8652f1572b1d89d9237803fb074c5d2a18a445c24ce4b1dd787ccb68a386d815b0ea6ec367ed546306bd555cc3a0b4988241205b248c0d6f17bb582b5d35568e265d792c5101425f9ba0a870456faa102e4083258c2adafd1e936dd43dd5ae284a3ae7b5d661cf65c52414a0d49190e343476297c5ad89f60e51171acc70eda0845fb8ad2cb8dee3ae8666e43c051237068dd8dbd521019f2971753f16961cebf206ef674382e73ba7d8b06c202963cef7abff4158f35813e13dc1bd50a7b579e1b8f381235a426b5cdba0f6fb6c139a81bbe8645b8f7f3a58bf4952ea5062d1a447d8c89afc7c989ebdee27ecadeb6080c036339eef4ffe5b8a5f89260038e854242494e94592f12b9cfc58441e8e8769a7a32ab9a85276fc480c03d1576cacd7d26f306631fab5ca517350a287955d93cbf32f74654f9fa8e8481b526a35caa0e25a9a625aa5e37d077105bdd74ff1651fe602ebbfda119c0da19f7028e05a75102e05f7332238db7c23e4c86e654c12fe56869b1561411f93d470f828e5311dcfff383a8b5094f0fc3d9e72ccec0358527ae1d4e61225045f83865f148b3ec174d794f956da9476b5eb75106b30db96ba178b0b5387dd4f17506b2c06c77b9400261f7d7b8b3b4ad3e3ae1e8c89125b07b54cd7dc439f4824bb1e6ea2c03e3af5aa463d9e707bcf6057ed391518c8f8ac108732a88ec0298dd39ec51c5b133387e942d5db32cec0a04d98fb174f6e9ef0761a87986747b30da50ca72981c0d63bcd1e52ef152f9a69d8c749c159824441b900d0107b6673a4ab66b4a898e6c04d7115785313d56d3873549b17eae1eb36e6f2cc58a3cd7127dd6b1d7085abe106ae9a8c512a3bfb79d7702d7cdc1c9e168bf645dae6dea9475dc2301c88b9713e94f25ea0d06da0227f5d8b624c6da7d94c322bf908e9b4648f1124f6e365adf2fa8ea029be62ee1009a82d94de35ab1d53e51bf6882cc116949dd8052bf871db075cc4fd0369fc7255f375f7cd9c9b5aa48177e4783da8456836bdc226f9d05ea1ecd9cd56a6d29f97b43c0fa054ccec72d07919c60db9c7daceba63798c897150e0e4d8843908fef82737e8a170cc5841779828fbbc2d66c1834d4046da0f01c3f2848b7c6c3aa4d47d1a8b8d327cac87d9db8843f14efb0d3f63e6c633b367063481cc309750b601e62b3666bb9e6a8bcd94743cd4f659c0fdf39e68fef93feacc61f5b38445f9a9192754bab9b13ba4037f58b250810dd9ddb7533507b8a1d5717a04d03d7ab3ba1fb4d04fe5fc32aa3c5cc9c5cbc68b3f56f65da113a7bff5dc6883d67f23afa875ca8b697a801fca06f9e1e1968a126a9c3d0ecb490646cab65e3d9b4225cad2cc1d0483b0c37127cb328a1641e8541dac911ad1eb3a90c30af2decce9d5e26c04adb1a178352561478ef83763c1612996bb77efa72b72a358d2c4b1fbcbca23412249e98bf8c63ebe0386b81721f61c688cd0c377eb6bfd149af7487ceb6a84428ef63d51d34a8c01cc0cb5bc0d62935997534196401536be43885a3c64ed65f62815961ad20bd1847110bfd575467bc9f1862abed6fb41d1f348380d89eb9a44925095ff19d101a4d014ce4c4d6769d39e7521ad119fe06130a1a3f89fb1890300a19f165a54e391f9194dfcda80facce2636a285cf7fc8e6f2afba4277dd75a4ffd3ca0831c5cfb2aec974f558f683ef4338d2979faad2c7af69c90b509fde726d5854c381d9056be1e6a257ef0c07270f1f890ede0465edc48abf4d9e3c0dbcbb19762d1f1693c1bea3b343eb4051f01a20ed2bf7385542dde66263da8ceee9f9217f69f253e8875ef27c38f8a7f7ffedd272b13edd091f5a6720cb534977758512e52a5cb23b0b4073fac566ee9907bdb4ab7b002df5eb395818c7693b3db615cd3b1f9a20770839a0025854bce02ff59f6dc15332b16a253e2482d39d8dd8e520310346f961bc7743fc00cbd70ccb734e756cdbec25269e5b79c758629102c1c0f35c6f46f1544580d35dc4a9101ef9aec8e65432a3f39b758d064fcc663a346189472b192038a3b68539a475e8b2731ce19b4b84805ee491c4d0147c475b38155e264edb549f6bddde3eafc13e78922b37a8c867b1852c42bbf3d8586db86027a45f37c4cd48fc5e85c2bbf2356e788b2a430372858788bf2ee1ac619d5e5eab740b8a3dedb70a7cdd7dac775e2f2e2b73b7a02fe3c8a3fdd301372b536a6d92a8908705ca06cd90a6728bb94a32fd06fc84675b14771c52dcb38b4a069db48914102c7030caa40fdbfa5b125026504d7c874cee7cf9a77965d2b036f3bb9a04b9605dd75bf2c68cd7e84ea4e564755fc4f27c02a3d92940481a734396ab6c4f311a15406fb711e170cf0b98632a1f6945ffc3a2283b5e3a05b3077ec47376b1eec6e1a6c83ece8cdb35de9e806585f1b77386f7c12426665c63aca27722ac86a9a01d7abf57703b0867c1093982710c77c8d7af62fcb17e548d689d060d8e173a6a21e9a035cf09bfc3b193545d66f5ae8f0ead90a0fff2776215cccc58b2fa10b6ac87bdd2fcca869c6faf6af74b14b861e9b4f823bb19c3ea3c2f0624ff92baa50aba0ac779e8a2c32cc353d8ae5d857fbd646afe59db04276bcb07ee9b43c522bbd1b0c5de81d43a115b08c65af7b43467f3088c649eadb9b61d2f93d10e5933034cf5b6ee24c2d07d64886ffd0470277fabbc1e80f363c1eda705715140336de738ee6beaf590e92340a57b7e112713d38d3800a7c522094c049032912e39ebbc57c6793faf4d2be17a6339bf40a2cd97d7701b79c899a39ec8ec32f0816ccb6f74a56af2d9c8a79f3517bd8471c6e1c6ad6ced92ec8eb704772980b5568b237d755555016417e7ec6899b4132513e08ac8fc6d0dd09e13cd0f4d462c9fbd4fae37bad4ac257dbdcdef9ec20fc44298c734ecb56348445b3d0f2659eafed3c0922eae71998e062df1c615fc45eaf9c6f325b5632d99ca47feba19b5f17a6766be60f4e571f5ec49d2d0a9830db16834e6cdd2d39e5f8882bfd910da10737aa19f1168daf759b738543ed8c1d4cd2aae84acd9275da9f804cf41d67b5dd108e71134955083cb727185c4538170770d94a927f0e0784a7087f0ae5557e238c20a0791c377a9b8b3c0f9234e580e511d633075ee20fd02d1d6362ffe0ac7569f2699bebe0ba745c7a668ec7d0eada48042bbb8d21c2280e8296d7643a507058bcce3983ace0ef31f5dd9f450c82492c2ce5331f1a2abaff735e899964fcf1eaa979bae99812fc12c6740d9bb67fc24d2c7ec5b87b08203922692f69680fed9eb8ceac562400e704b0b2319a67ad236054a44e7bff1ecc235cbc56abb582d8ce5304d56da6b7b80673268f219f00238a70ecc15903eaecd521e4908972455744e8eb50af809e07ab88a765747b44a44c4f69fb0ed067598eb676a94ccdb8b55b58bf68f5a9bbe99fb2d828ce353fccae220636d4d3059cba4deeaba9db75744a9b7a400ea0767229dcf5c474855475fe1a3beeb581050bcdd285a9e147a76d0a948e98792c7668c11a6f116bf98c1b32c118f8450bda45e28bf7e36a3d50e2857773163bd60bdabb01f61d794e4180fc9ad7efbad3c30f321b9dcd8a010fa782b2f2c342afb71815b47c58291b4312b28a7ec92fc5535debdeb9e88ecdecc8ec7b26b0c7a194deda48b031667be711ad7756e61303b30929446432f7197c08bb9d476b3e002cb0273fed36a35eb0a116838b9e25ce7c52461eb25ea71569561387fee733fe6927db5e794986837a61f0b2bdbcbfb62be711c8bd368932a722b9e3967ce43cf6448a1fa1054180948695bc5f2a9b38b19ff166dac7a6fd27cd92738cc8d3b23adb9996f58fe842d371d524bbf7ee9e36cb77c9e5ff74f19c32eea3f885b29a1e5f2b923f29ffe5df55b22bf301f6aa3b92c88de02094d7b5c315a62649da2dfdc2b736099d3ca80d5a451ba10a1e52bc5b1f7166f4cc251e2af7e7c2b029897a8f0e33b19d5c44c2148bc4830cfac20eaf265c594ce172700f3797988df54ca809c105670de33a96a069b9aed359ea918e85600950599f785002e8cc798665fe71c861c0b8890ec6e2e90525ed85776a8b0ce70716654f2756e846da03fa1472e7ef299cb1cc9d93f140379726f0957af8fd4e0877f1779e10b01eb3b3f51b12cd3893082d90e93229f354f7bd9988b3b95d92257d2631a4d9519b682f82b2db944d6c2ff0e7bdc184092e119b1a5c1515e2b588eadba0ba25b5dc8373546983babbb50aecb1c91e4623e5250038ae5ae66e2e1c183781e786c0071ba2ad6b5e428a18fc2e26def338946a60540c98728ef45efab213bc4b88a8e0f62ae88bfbc7c7b60fe3e0e6bc49ec079b3af2d4dd328c1a049eee5a4a7e8a87bbd17f216c8614bb0b13432474191a456d51a7049c26d594b78ff35bbf8188ccf93d7c00c74f885971710bf25526518af60f99f5803e25f7c0d3178e4362f82c4bfbb7acb6e70f3d08dbe61b6708360bee6f714ad6ce23fcc559665af56d5be1423bbd4bec667720742c4180e3c305f3fe5b988d17bcd612175a1074bf3e04f50a4a77c291dac973564f7846c31f8d0ecc9d1c6f9703df9a18e9c88fbe7a2d986eb960f2ca147050245f839a90788c8bc22a2d0abc954e22b46d86657acfce170d7ce3b61fd5f12ba4a7cc573e1984326d9bb33230982bb8c8cfc38b8691cf792dc509f17fa70ec8ea1063562791554352ad7b5f644173b1314a094f054f4c8a17d42a9aa76dc03b6082864b87aa5ca0a0b635e5a0cd1d61884fb377c77c6d5d16adc3af33a5d952074643dd4fff0b115acd520099b8efac9371e8d8e5bc433bd8cc8fb1ac280dac9bff517de56161b946fca85b064a48209853f7d16e702514cd8e9dffad4f3f8793ef9980ce78edef7afb970dc9d0e285b78350e41343aa0723119742ae0d3a31c6bdc3f622f65802bdc4248f0f100ccd8710a3e4b69bc69ea108d8ec64cb55b9ca47d73898e75a54f70bc3fa7199f6eb9b1b5776b4c9b6361cc6e9f5545f8fead4bad8d4037c9612aa5f17beff9a80243752a8b643f398680073153bf88a3cd3ebaca89047f8ea6932dd68a046a1dd61677dbfa4ec3dd81b16dde624bfcd6fb0147cc0743c562b1668375515d5fa03a0513588ad60e86a446ea5b49922835497a481d33ecbc94f129914cd2cac1295bddec1cf9f101f1402196367d7a8d94a6b7b211c89c06dbf5897dd18143f80f463b37f52417670ed5f6ac942015fb303276a48ac1a190f2a97ac07584f9621635bad3a830f89a9c31a9aa484c1a5e8e72893435323f299aeb45c53c5c613f99132578f100cfe3473127194854b9829e0a0fffb0f42a2c6922465eead8245aa41c0e07de58a0f27cdaa884d9af1d1c40640e9e3833d76c1ada3f664eb8cef13d94ec42980b2fbba50cb6b88143d5065fdd0c5619962617327880a9c164c8d177fb506450cbcc6970c924129162b27eb1c9326d522cad2e5e6873c4927228ab1bb771cd509599546f3a6332da196983e129580ab1d0daccad3656ee5b606413cd572fce601a2a247117aaa8e85e1e5221d27edeb1476190f2aab34368f54ed0c96c45a37c1bc207cce710cfb7b7487aba060d8ba4b3d009419c5844deedbb082cafc1e6bd9c3037ec7a4d1360fc2cd2ddebd7e9825c40cbcefa0e14ee2d4d039a9722b5d5a96bab0af6cd8a4a5ee5e9306e4083de99f13ad1a96c616e467c386623ecae4ef1592954018aa07ecb5098f7563faa4c40a8adffce9e4b3a40aa1b9a4ef58c9cf9a26d674ec6206807cf1fac6ad5e05175fbfa2abedd67846864fbc733842818791d3c201ce8dc946ce2b7e9dfe3c5f95eddf2279fcf0c4ecb38fc3ad25938f9a8a41ac02b5754db65253b15b028c5a784f8b78c2fe2455075c4d4172240fea39ebe5d0353378f8cd7b2a7d80434e1435462b5efe254d0fc738065ebb0526e293bc97d19a7cefba2420e3db9f90a546380128597bda5d4901e25dce8f0a49dc2da50b5a7ea3333ca9e8330afd3581d1456762b8053cd3cdb6a81cd505419225796c5191a1f972313bf5e5d58156d4c2407c64d0e0509e25b2a8fc237e8095758c85c4abdeaa1bb175b97e675761f7ac974be9c5ffe7285764eca01cacc4bab3e6319e79d0fd3d0690d2db898c188aeedaccf63359d775e82fdf9152fc90a5873efda1f9cb78c4c45fa01686441be2aa52c550330db848980be9b700849fb015cd7be2776d4dfbe72a2f00872acb794c69871cd0233ee0aad3416f73cbb8291244bcc09e8a60ca0e765c05c8f7c9cd0d9a77117be8dafeb4c7d3be6b05e79e51b1a9a239a27340ad6cb610e0e7574e9a19a4fb3cfc3cae470f7954fb2c4f4ddbf4121ae380f2994dbf30e7b529ca21247d9b428e48a57cbd062c774d2c569e22ece512f0f717cfb1f5df367e3649e4ebe172d5c063eab2c8cd34ac5c5f1fa192d75e0ebf69106fd0791063f5b35c36195bbc9e16f481ad7c9b4b92a40312db01929e44c84efb6c04ea40059b720513955a0e691eecb37b1adc7053e7751815aaf40bab188c98ff5c87966ead11bc91a5219c6f27e8ab3fd31733dc27bec345bdb61a7e0a20ffff836417fee49e0c8d6cca1a9e8c24e78404b0068f5d09e47599135cd8675844cb27396a61c58bccb7e406158aa54e1596ee2b47d0f0c6a1c57ce677c0209bc99f75a8a6f53789958be800b46a4bb92ea080f67235f8f464099af2f2a88f7837d5ee4d29a85ef7f83d3cd53e1b45d8a7b9b051f2034a994a30a6ef64ead161710ea7c2476e820f5066f49f676b2b597ac40e7c8ee4c7be72a47f8cb3bebf754afd85f36949175f61613b8f3abbf1564f65713d69a848dd954992c705a89624906b828c25928d943a82ee56f61a7192cfa311c8e9363c77dd6755d3450e9ce671f9543678fa5855714a3dedc11ba6f0cc3816b45cc2b45594f577b09eabd9e00779349d879b970a06ea6e526af606f0c2d6ceaf668c5a13614118eaf0c64d13b1ab42c28e03b4408a8dce6f90e9c8a719258bb9dbc731b8665a361a6f73d40bb463ce603fe750a98aa7485c3b5f07618b4976e9a8f2fd1dc55d1465418b3b8228d099d7d59c919af101ffc7eced8a9e14b97113a6b0a8e49870b4c33afdffa17a946de24e2612f96d06c092aacc51cb906d93f476aba2c569d3c234bb578b91d32ba6e9bd6b41f7c1b48bfcfdc8bcfeeac83c2167cf423d4f675f6f3fc9ebd5a794acd042fde1c908d51958ba5fc9ddb02ffe74a90c9614ef5e90735ece253639893b5e3356426c684f58cfe2316f1724e776850f39a4a4b39ed3046e898eff86684491eb4cc56219412face9380c1ea96051814d94b969fa212e69f46278fd853a2709cc7fe50ab5b4b1449d9cd30786c7a2fd7ed7ed40916f735e5f9bcb51902d7e8e16fa60ff488ebecb93fdf434326312a4a4f77aec881d346971a912e718a9dda19907413cc943e50d4fa58753806291ce3f839cafc56b865e1e50aab3ad6e8caff0e3b8c9ba3eb16dded8c7f496ce5251bbb37bdfbb1392f99b6c968d470a6f374b967d29e649abf3ac4642096d06ff12d524c2a48e3af86d17b571876e44ec0d885e57972de4acc5e68a1f016e890bd59b3c7d7b17f5348b280853b7bd43f47af7c302085ae7b991e70a8e847b1cea1484013e1c427d443f399c6ec2c736b9f1dab226826859a424427782206f674212a170bd4078063dbd802425d1a7291be609f94bdeddc489b64f732a06f06e2234f1fdd5d4595fe3c13c659975ec025f59eef6b97f40d29ab54f89cbe63163b21a582972fe66ceba10bdfbd345e47cf77dd2397dcba2c4cbb2364ee7360eae1fce91d7a59612226927a8de97a3dd477a53e80b3a9fae88f90ce7f16804a02f65235d72a68e0bf3fad77fe8656819261e827504773566f2650bdd6519bc345b18000345ab940fdffe7ecabd0035325f87aaae957a7d0d0c55de08d46fc5adeb4078e3c386c363e6038625ca8f5dd550cb0ee18f2d2ff392abee283439a6977477d6a575f91c4a870358c6270d0211378db64fcc653ad983204317f0156bc91dc4d1d8e844cfa610656f468fb2a7a7a470d298a94cf56c1081153f2c3f02e3c552a86ebeca5133661ecbd75ddfcca3702ce1d08ba1efb74946bebcfbd6ea0e5ae793e6cae804f685335952f5ee7c42d3a50a7da12c4667639ff20049af5c4df45129c8f8d8cead10868134ef942ef15873975111d79c9936fc654e4f473fb5f5e59182885950df8c9226b03c5ae20fc56c06043391fc10a6da6a51ab9966021488def4a5ecaa8bec539b8092f0bf889394667d9d20f0028467f54ed7cc4a5f405e11f09f37995e1ad21930fe2dec1072313ac1cad2e20feaf5514b8ee54fc025806d8229a135f443bbcc71016c54105b01183d3ed5b5dc38fa141758032eda61fd15efcb721f0a78772c858cc1f257e63abcc83866315462eac2c64007df4a84202f8d6603c104150e54329d4c72aa59f19786aa4f1b6346b4dc4aacec9ad911dd20b3aedbc2673c211a5fd7598e37501eab82aab6177063fefd88ca90fd4c7066005960e456628bcf63fe8115a7286b177c3771b73cd9dad2e99bdbde8e174d8c04667a5e724d5d8bea9109d393872d12b92faf6213f599f7d28c5d393905de1e4a0cd1f2317f20d51d40a5c28685867d3b675d28ee2b0155d742d2bd38d3124d9a36f0660fd6542a1b6c8b8b3b06613b9ffaa7ea909cf8c5cc1d90553c66782b5f135ef61a3a7c97ee1f5ed15afef9d685743271970b65e330831c4676bb0988ca846e7bd6a33c44eff88d9fcb8aa9ecc71def07d0cb9a1551c8ce9408e612090ba8e95b3a210193619851b7dc9ce1a0599ca0ea480c339fec61f9d671a7d096b09f2078ac15c6adb53546e8f9fc43810f9053133461f881b53200e2fc04c0ac7f2b0136539fa7b303e6cdaa5063f095f2df20133e47f240567a81f6a627b8045c473c1d1dbbab7db888a43467ad867f44c2eacb5370d0602cc582ef8062c7d37100f6f94da0dd445a34d8e5b34e82d6b27e684cffe9fd69fb3fcf7df6376d0d72dddef98d8f0f6cb34727cfd0cf802c456506f48cc2baad9c3837479a1ebf44c87c1c4202e64d5e2551541e1fcdbeedf9b7b1666f49d9f7e9d45a7d4731e2bba6f833e7e6225a03a5ecbc815d2ed5617d783245530e5a3d3f6f4348015ae7553d8c338a020bba769a68ca73c306feefa47f1de942e273f07dd092d716b286bd260ff1cadf881c1d2d4c351b823b35b0f094c3b3aa60af92cc5fe3f8d7668c329c80f3f93ba09ce124ddbc8dafa53b4c0e6624eff72ca0ce3114cf37d2c069b8efe7e8b7fe4c77eff084a849eea4a3a4c812b061c3ce62975a3c7ecba134ab2f73a13af60b079557448625561df2a5a4f39aad2807c2f28fd47bac3a32247d94eca190bbf2795e517b633333917cad7cdbc4a336cf3357f56306dd874928edbde00c51f34eba45743d30f39e04c1798f8b62e4751a742c193ef605aa3b5cee04baeb7acb893bba1eac7e80df835258e727ad3db7e544bec1cb825cd53380d12cacd3148c8c1cc9a74d55f8bdcca334d652b7207cf5230c5f33a3236e323dc1a26d861139641e1093b604df24ed8016b0539f4a7db0aaf34972f08864cd978305b536c60eb55da44ea7d0ac19d2f7adad5c72000696afcd5dab2d2365d90b9259341558a7c0ada2ede7aef64d26010ec7210190390e8109f1a0be4592ae65fcb51d1d2786a988216ba65d58a8606979ffddc04a4bd20bef19c9a521ac9248fafeb3826811897e970096fdb65225aa0018fa0dd9c3cf8e38380987dd95358ec8cc44b17898c1f77eccc415404970ada072a0452afe54ec3b31cf5e68ea2843f59a19f7b836e18e80bd358d240fa1d0a709db6ed435a0a007efea7914eca8762dacae33225088441d416255caca28c994f1b5988fdc742ea7d273f0a97a91254724a4acf84b62be4cbc1202f34e333c27a006a6fe0c2f86f67999806e838c36aedd63163e315bb202a235e0300d9c3cce0ed32f4322818a3deec2c2c60e7b60cbea076901380972d17c84c6f43f83f8a852133a87362dee5c1b6ccee7ca67cce9d89ec15391b1fa0f74f78afe07994f4c18300dd7d04415712b8c72974429dffdcf63fd1d5cd2a629a5fe3fb6a0fb27b2e114d7f52708d7d6804763e190c932a11c1b2c876397f22307da7b0a0a6f443d09c7349527f2d57100fae8e713aff49630e05d927cdaec26337cd87917bdbbbf08bb849b9d6defb295a6ada9c273a1e6ccf6c9aeb26c6b3bd834ce75984e360d9271cb9f538843c3b5229f3e5452cfb9283b21a991199ed32fc24ab6e1fea225c02571c9822d38d94166e352110b8c5ef04ba6997f92fd06a5fa7b457ae16a11fe3a093c40eb2ad73a1333e80c35f26845c09e7e4dbc1f494f44e941cbc24b1e5b884140a2230024834dfbde99e10fbf0014c91301b5e07c9915190fb0ca283606c35c71cde9f91c5a803c9c15affab902f0e708bbbeeacd925d67085d82cb941ad6ab170867e48565bdc72091de889145b781efd714edd25bd31bec0a6b41faca181fd467c1fe2e85543216d6d62e1e4aee27ffed179ab7ad8063a5921e43f23b4ca8c2418de66c8dc59388a9cf5e21344e33ff4f497825464f0eede232c00b35f428dc6f2817fb3491594663695b7da1b0372db1c07aa03c40fc5af4f2702b5963b66621c65dc5a082db3ed7e895ce5afbdfdb9052e5d88952afc44048170ad8b00e4f0519e753357bc84b9b0f99d4a78a081a029e9a24e44071c61c3cf85849e52f4539ddf5a238c4caf74f6a2f0dd75c6cb706f42f8b9aae1db6623de321431b2d5192fd577c68d131948e924dbe45548bae2e886cb31f525ef2c2f90c661a1c3b13c72e1f4557af441e304d8cabc6e37d42823990f76e3e860b9b5526e58c485e7213a75367d446ad5bc0d87722027008d637f6479323f9a333784aa8b2222ee857cb67225f2f362ab9b3990ca40c9eb38b970796289339e691939b875642693438719dff5571f0e54c1ea49853c7ea87961e1e8e2b3bfbc4c61d841cd1e4567ad578c974d9da486945ae81d34ec5403f6aabac477f611950066de807e37be49897a22dccf908f5e74d7dfccdb7cf9b2938fc7b58f86f80d91de78d1f21661ba74f240d3bbdfa24247c19826b27917026ecc4020024f7cfa0cd484a216c4d9148de2db23b547523fadea53e7540c9812d937b5d3139fdcd4f1aa0349329cf14136cb3902a6c616ffb9452fe887c0163dd9cfe6426fc5b25ad6c25837e90ac9f6ae7909f0bffeab21886660aed361d3544300b345489df4404a997cb7bd79cdedfe3804b9058f025798d69d613de74a631ddf6f89d411ef0a97dadfe7bf980b988c6f1311f121b4535b658630d6e92a17a98825d2813b048b359eff8800675778191b1e9b192f171eec815d327b5aef80b6e56d4fdfb676adc7965c81c3a738b1e1dba9cc000796570901d238f57717912a37714f4c2d7a5d48050cd845f37077fff6ccc6c2c4efc5e905a09d226e3af36d261bcc3139805cb74856a0064983e230afdca89be366ff078c22d410f33576d9a9640bc412869209bf5fbe5803895e54df3d9187c167b6fa4b53c7bc630875b6ea314189efefce3d52335f851fb1cdf45b4fae0dff3598aea780929b538894f090e0c203843a493cae32bddce85be2ab70a03e1ab00e610a0fab716a4e6c712d549def034442603f6da601783a48308d20c19780315d58303920f641b6434d5fd4c09f7d04786cb4568bc844e81d2a44798176fb1a0e7e8fcb0731f5d666f2d28734da91382c2856dc45ee141dcee92c3b16b39bccb422b8f459b92180372a734b1e662dd538705c79f78a0ba751bcc7b3a197b1fda4e12864887db31dc7875c8b8e5ec3455ede0a1da26317a57280f273db0f5587f01b20d7bcf2a32ac37b458b9872caf19dac39959a2908fa675cd2417892fc7e3d9e13aa7a57408dc5af53f4a23be62420f317592a837fcaaf2211c7fa1c62e95160e6cb37297448ae182a51b3bd0e004bc80cc6f3ba5e67bb0a7ae860f3944ceaa07b98d3f9955a29d7123f355013066c52f4933ad7dc39973b610a38d478c554fad9cea34988e27604854462693ba2ae321b743a3f5c2d1411dc0d85f593dd7805c6303f3db7c751e41f49f240bb6c3f1afbf7007cf87aa52c606790491ed588a425ebc8fc91b7bc3d7ad76dcb2973c5527a58c04b33c104b55e0a04d55d91fb35477ab22759d2515498fab3b9c4602ddda291d83c7e86e9a41c28c547eec3a82c3275cf007db0e8ac82054f0493d4921b80c5e65f52da5f20157b22fb06d11998c460c34c5752d2b44d95f5ac2112a19c387693cf649f6b2252f6b0c8f543bdde249a08f14876d4607bf4479a34e1e6e3d42f8aacef99df0dec4fa7cabc0672384a43eecd4b2dc8683cbfec24d824c39c57ab1bb0b0bdf3b0c8ffd62096589d53917a11842f64386dd542d1a5524d1b1e24f44057b6b68bdb732286d8ad940ca484bc2b5cc0b88354be4d02ac375d6f1b25831c1acf085b28ee24f3a438e858ee5117044db7a8d4bea2d588740f28f77dfe7c186df135b72be6b8daeb60b68b1ee98ea478f496aebe4ea3cc8acf89839dcbc9d08f4ecfce5dc66bece3688108ddefe264ed364fe2f108b465b456c18712b3687c702ec6f172b02a37ced168891d9450aac22bd65efffe44c2811e8e33cca61d9404015fffe029881c429fc1c5fa464f7a9ddda07ed23db7b9736b2237bb56103f62bfbff9092837325938476656bf18b8b5346a0d4d37b495f85aee6f079f6e1cfde66eb2b673f9ec9288a79315edc98db0f098e39fe1e28ab73fdee2a66e2366203a0888328a6ceda006888ebf57d9afda3bd63ff8e1ef433240906f1c40c42af0c2822c70bc4a41743731c4c1d5e9b6025fec7096cfefaaf5980be1428c7faa7f9c3681c77a53f59e81bad6e23c18187e8c3acabfdc66d660b7010a0d2eb67d07dc9da5a5a7f7aea38d2453cac4c458064d6052d01f03e103a6f06308a40d313f078115ce363394dc3dd6f55f1f0949cc98b70863680924a5c9f30556c2f22753b406a0acaf3a391fd36069cf9b0fde24a603f036009150899064721125b74e90292482bbe0d471173bd3d5b97bc2000c1b97e594371f2fb767c8b121f821d21c1470af477c7a7293a713650b88cb3efd2849af56d21ec15d38430e0c89e3335d20e6aeb41ee1901bc8d157a7411b2e5c6b0dc5b305e5759ab3603bb83787ec8e6ca0cf86f4868d5b92db678262feee7eac8fe6cd4da2f32e22422d4d1bf012c13ff21466b597997570745fd3e82e406c6e6b2408fc3973699288b39e5b317e15a29cb43e30decb717f8def6ba6a74d7b7c689aefcbd631aba6a71c2b4b505f3c256afeaf4ebc1cac2ab980fd55d1bb6c61558935738a43df6933ebc97324a2011f4b7dedbf0a740905583067a04675262cac0789f3ebfc32e971dadbb84c0b36b5501f3a8bde579f3ad8307f7a35fe1a0259d4ef7ce3991475cc33456b5d5d01d86a1f866630a599d037213222a5ba25b989d6776af5157334d871100d610a2378f20f28221a193ba39af63dfc368fc7fd1f8ad58d914db6a1f296fd3aab912a595ed3d5810b30d31e77c7a8bf37c3b333cddcbba0558e5160ed7b7bc9ee7ac2cc98adabd1fbf9d86fa102cd5082372f016a0cd285133d883655d93dc256570c9ecf1a559b030198ee58ff953c9c0e76a70d603371f5a4237ddd632923033cc652e959b2557a62a1631a17939db7884de6406c93931ea52a4b742536e21917f75b39eb6a46df7b34ced693f26fc90185016ea1f3de265b5a8fd8be87312f5ee27f7c1fca2cc9b8b1c68830493025fb02de2f0446472ee5a4cf8934669c090ff35b426d3e5be2895925a93e87835de0f27e08f021f2fd25093c522cc793ba3d3d29966af7f5e353c30d6dbdabb10ad2a2d855818fcf5fbbc04f2fc684c83d8c66cb322c04ad65c3d79896015227bca0f2ee1a333c1fe45413ce3f8af2a5a264c7c1da08d096a9f486468ba0a22642aae54e71a048ce6203b756a694aa46ab8be966828579eff2998dfb510045bf2f25ea8868ac56cb17f6f0949fb71ae8705845b60ca3d8c122663b3d0781795b05fe980b4093aa14a966c460f3a8c4229ebc8618036d05379635917e344588acc7f2abbb328e2ede389c48d40d4c08c0055311de5a282693a413f83490433c246c49c7466ce107045349e1bc90a731fb12ce44f488a355e69ff2303f03bb2ccfe16014ded0e1642b88c62499fd6cd93d95039f30d3bf880080bd392f404e1e0cd54192075859ce43a70048bdfca9f25f50dfea3527fdbbab8256ea95702674bcee20a04cc96f727b0063195c13611a883fa29692b43fe89a1f9b14237e3473fbf78075a6ae0284dc6a7ea076e7db0bc51aec4c4032034bc89efd1df5732cb7079f5dea57eb3c05156587ee1f9e9046928f8273a4c9bdc53d9c3d13f03d9e9994138aea8dc68b1e2c39fb1213d999fed2cad828ab2cf0e72f7a6a5639963467bc9411ca262c19a157ef7cc47eb52237247ec68a7cc2d9d8c0d63c75e685678077ce023af341c974dae1f80fce50eff7fda6d20d7740cef70106c463d39616aed41e03144db8f3dc620a8a8f03f51915d64423fab4ed461ab582a1508eb176f7f6beff0313f42263ef6efd670ca971eab66a472d259a5c0f4ec8dbd2d54454437a3de59827e4db71e45209128ee3d1e7158fc69f18bebb5263c8e5094c47b51010de45d934925514f5c9a84e09ee5d2c62451509d77f60182047a90ebf5675913b1aac36708a63b353010c360ae5cf1df0ba1a0c4f87afe02da636819900abb9409e92cb13d42b2665a246cdab4cc6b11e08e353ceb1ceeea657092a8e18228324c93bedec882abaf346cc093cf9a94c3be93b8b00a636722b5ee5ea2f03d47d1681498b1b9004477a0a531e22b51500f33915e44c2c629e9495cb70969f461b389e149a5bb4a18673faac2a55100baacfd6de082aa463183f6d94fa37f7154956e4aa18c06b851011372a19444a749f06996073f7a0d539dd96e4e878542d394b9eb0219550904c7c5ea580cf486933fb7f8a6ac9f6ca6632ec77913975340582b28179ae5050e517d21c6aca20d4da93c62ce66725d678ab62f788e19b4b4ef231d909bd03b08815e7b2c9ff8d85d6c75e5aa6333296abad62621bb7a7b4533028e5beb3a2050ce4202746c63d40664dd996176c50ede03bc869faa11f7098663942e348622403e0b833c66147f70fc6cacec85514d26908bfd50c7e1199b6fc6e116e26f6a3b93a21a69b55535f0e63a7148d28e00717b5fc28e06f7cf3902e907b5b1a81895a6e95a5c4717aaf39d9f98021769148366f3ce526fba1456dcf62078af451b944f76ec2921c0a1486aba6684a003df151e16015494fe8ecfe7287aa04d5db3412885a8fd2af3ee1f2cb901c4a34ae0421163ec03b93c9f03f3eeb2665b7eeeb7729cfa51678e794c7a596dfa657707ecfbd590d0bebe42179b4d1edd6dd060c68665b90a46aa16df6bb9e8225ccaed11e285328b78c224814a694b43abbcbcc4de7ffb30e02e23dacd134c09ff4361d8b9655bdf0e3c8296009030ccb01bb92ed8889cb908db7f88a882e0284219926e30021916e35e2bd643b5a20d0abef2fa216d735a235ec1304c14d3a990330b46d77b2f1bb107cbb4d6ea297c581f762d31417be3516f6726c51e7839d958f1ee0bf1f0b073806226e702f92315033945d6bdd01b7d7e853a44dcfcf52be6e9396d17745f6265f4e1560f75d4d2248e480ffc119d6b5568fbbba2e9a3992d08d1068f6784b990915b70ef878431de865e33456bef1fe7fade0e2058cf96255f5f84a5777c2582791bff32f4482d5fd28cc4d81594912c7fe1fd9214b6fd5a6ad66b8e6e260efd6aa72d7316e987a0f3688d0f5c7a0b956defa649f796686cbcafb7b6616a97a7f5490da105cfb9d7b7f952d095e418ccfcfc2f146af48478ecbbf3d5879f0b6a668d8a3a28747f1030473c332001998ae0d9a6617686f990f4632918dbaf6b26545b0f3fc247e015411bbb36407757fd98bd0b226f0012c031a9e5a3d35a846630b345f0e01fe7b0a99daef8b78ba79b432296684f2f15d568c1b763c138feef1f5ca1c6427b6b2c4b6feb607b877e842f3dd561e6054973362fbf61717b803b9a6dea8e31fee33af468cd14794673e72d3b81e4c083c9e795af5bc1dce222776bcf358f06e957af1e7a1e19b635fbe0bdf81f96addbde2b640e4212e50f7fe06444f98310f0792bfd83539db1b1bbcb73c7d1ac73878e3ffa2e7b31b91010fceba1a87877ac8aa83a9f91cb1ce1b42ea74ea71c228f070cce4530683c78fec812c70a872f62720e3bda02b7aeee3317ba3092679999e6e8806e06be92997507acdbe7607e4b04d8eb6031aaa163174f1cdc16ead03b30d8c64429d2ad95f2ab682ca625254d0ac48da22079ec4355d8daa0aee5dc19975ae98ad42faf3bcbb8bc941deb60e6d1f21a567fd751c638c1248a202c0173ddee94fc854890def04f0487bb3c39728403903cba81b0655d0b1e4c3f3c122be57a6dcb26fcf5353d9440fdec16ad0ceb3cb4d48a2918f1ea21e3f845f20431f1055ca70570638fde9dea8b6b11cb19325d695ca5ef3991942666d9e5506346a87dc233287c9682394cd0ac45e64fb6f180b7d35c2d686a5c41ccff5452cbd1f1633f7b0383c16d26ed76b280e20ec7ddf81ea3fdd0bec8d927d775adb83963e7828e3d5f7b0b24e9775ab1d47e76ae4a2fbebb3c29fd64de2c777756465d1a2fff129adc0486bac8535d7e27d6056149a1c0a8d914c7267a0a3b7cf65c07568fa690d550dfeaa57b9ba4cb64fd72ab1fc42974a80237f613d181f0b9156d2787d78e5e51d9dbb8496c09596310293d6c968a1e444dc239e83601717f1393d945250609834d52debba60c6059dde264e67a6932d508fc23f7e69694d7d2c3154392146acb85f1dd4067ef5253ee7631fffa5cf54a3dc3d04b3f8708ee700d078801ddbca423b62b89393337b1c47b6658691e8fe2f280ffcc884443abaa9e226299c78fdeea6ee3d0c5f8f93241e3006b58b338e1c6726016709b81d73b4647f43aa9724602f6c9779fde67d11ddc5c1e27df043385d6588c91e60922ba9d42a4c0d799c82b791cc0f511776be211160c7d88060d4bd2a2374d613b6ab32b21f204512d2f8e18ef9c537712b37e84b4ede7b5b486fe5d9ebe448f8c110fcd0ac7a7d2f3ce15008382fc8e7740dc8766706fe4095180d5427f59f2fa745a51fbba521bcf8b57b947e24c0c673380f720e5c0dc9258fb38cb422baed524975bd6d87a27f2bf61ab7afaa1b05ba609068cbea784ed0be6fa7c269b8d7ee8c93f2961eaa69bce91bfd2a2c3a16856154ef1cadbd10d8cdc401a4526c3601f996fd282451f3bba3a5c12b0f0aa0b573f1d21a9b5ae8482ff60cd56296d7d7f2d8e625dc47aaac11af4473077155bc8dd91b10c23a8809235f8e81a2715b83112202d790bb500a4920f86d61ffd230153410ca7c4facf902ce433b215fb77e08e9e3d7119c0f7d21a85286188081b08037d324ac7d862bd51e50f7fce0e721721a88b2dd20d930819dbc71c67a57d4ab37e4cb90f0037d38db4246cfc0fbda280645798af145f92edc659aad147c2dbc6ef8009fa9578f6ac58eaa959a609ce5a62fb8a56288193f88f6d058f369ef95ead932b53b398bbc9fd822ea0c025c3e9bcd8bb08c5f5f602bf14739b1e65204ff946614ea57af094f8d5d8e373f849c641bac90868553fabef61e6e459e5470a59e27c89fe05731d89d3371eb79ed6cc13536f90e7fb3efd0dee488bd3611e85e99eedac8d94715bbcad1ee11a8130002ef1ece7dd8b1123fd0015fc0c37223cf959d3de4a0d2a8971a5f78af14bfecc747fff200df1a0d24f1320498bfd799d368630c183ab3bdc7938287a0b6ae569d79ad68aecc9f70d75b06ef8bbb88cb879feb8fcdce95312165173b2b62348694f4aeb547a728ce4eb9e27fac9d98e4e1f34c9e0f8ba8b29de27849f94f8d3cd769d5f793ca9f9222478736b09b2ff369d78c021d4a4908384a0d58145d6c93b737ac081789ca5448ffa1e7621115ce150063c1f67857f88a6774bda88a3e6c35c5a6f6f8c748782c8b39f4984e93e70b8cd798176b53c0d04ce07fb542658aa1c0dcda3fee9dbd5b254aa98d5f65dc5016f32118e93eee28b71662d89b6a9920a57307d9fa2e87c05f4a126164126585f2d1908492ba55ac3b3f37955955aa6b04fac5dc56e0fa0f2aca8af9487fbccd0b83370d63d09431fad25fa3eb7b4d40b09d4b46cb4d6fa0710b7e7512d20160045995a09c44e14fb6aae19a99d598d44f39f7b4cf422bc9dcf0d4fd623a9f6eafb0c2fdbfaf9679e844a96c6e81672684406055ddd9e103301224d34fef6264572ebd60fa946554b7d609219e67b4ba43a748509d191c5ca689f13347145aaf556136c543e0506f4e445e6ae86a9b6d7eb8410f47ff17e8258f3b3f2421b8146f6b8457a759f076dfa8517f088ff6276d46cb29947d00df0ac76dc11979adad3027f84da3c47b4d9ffc6763c6441ad1135076e7b06d3ab7aa24b6636749cffd3f559ffe48c1570f0cd55f50a3cbbdcc855b93392fbd3e5b6243e8c44deca93f49fb2cab62fe9a0218e20858bd2f4228063f4832e695404b2d65b3c7fdf02829d850d1a441e2f97ca7f3f7f9d1cca26211e756755df9a6c9de2d5aaa56d722c5044dbad569182da3b4ce6ecf4cc4e0cde094ac90a81c3cc6aef5565d42bfc8c63ae1f9a865f971389b5afa0cb1cea9858a439f55db211e934085009dfdaf0bef320880348c62edb140fc79477f6c8e12e658962ca54e47797c2f1b0bc588e4a60d466043d6c6fad73d642532ff235c1774940e0d7afff3500f80209053b1ceda186016f35e4f4ce4e3e232e0bb4db6e060ded5a5963a086083ef652a00497542dded583c88897b0bd918a982cb35f2912a8ca2757151b77e8f35f653fc8bbe05e71b9619dbfee9bbfa4e40eb60d2469d75355bce1517faa67f5cb132666d2e16907a54c20efdd6efd47ab46ec04dbf1786abd796a97ada6b408a5e5419a66d27ba72c74b78ef532ead69ea9722487ad4c106027c0d5858551c66d56e71768b28b8f1ca23ccd57a886990193906e3c63700b40f14def673e56c97f3126b48accc991805d6f8b461b2e20129e9870927f697755d8d0aa25912842a2ed1a5c7b187f485d3fff67a657235328a0f813726dc2b40ed2dd575f24eafacf8c656f19fd97712c21c3a18207fab98a3f1d5259cc473bc69b906704b0ddff9c50632dcc81f3829e81fb5831372bbadc80e74e7b5471643a0b336043d76c458084b01ea67734866e31e721e82caad6500146c9bb2b9130f581501d246b65c5a8d0d0cef0f7b24ac066e4a22fb1601313fab7c9381673c9a99f4c83304883a77c3106fb15106ee03a9b1877d8979f420d299d18dddcaff9ca9e74d5cb1c709c9c2293d0d261ebe4fe9e057d6300f1a05e852ccc3bb5f3f70090c163bdbac6f5e4512189541149975c8ce19a2bd8e1a2f018f35a5f1ffe085ef26e993134455a4a7cd9f75b9f210ccdfbbb493ac91d0d041209c416d8db98dde01b2ba18007560fae30ca55d132bfe2e28a4ec875fdaafa3d4b7b7cae9d0475a4ebf717994b436b175dc79f48e95e871746d21f9696c449d167715626bd4933fcdace936d883d195d87115add42e2f47bc88c14b4b9712187a57f563e809361aa9596dcb34e80117b47e8116f7b896ca2f50e99e2fa86d776b86d159b37debc8a9719904af7ca22606397970d74f85afb3b5b6a1707962e72615654278fd2cea22f04a3f5c86d3a841f6e8d08238d03569a2fc7ce58decbdf01bc7fac52c73af892c02810419bf49e70f19488fe2899df828a726e7bd0f71de4eeec0a0683b197b18dcd042be5dd6fb067b291bd224d7e7bbd2ee26c70ee76bd027931d9d6c14bafb893c0d4bb065ab50c1cfd9e273cedf88427b764241cace5a99b58f2b640f39d8f4fa82d3f48ca74502aff7b16e026e844ecebf6f60ef8649259278b77d4d2b3c4249d5375c7c740ce7d4ade158713b359c84a69f4936f6af3cee07cd6eace185406b6a9756624254bef133e60f6e907f398fd914939275222bf150c6a85ab4ff11f0ed20ebfefa6efac90edd3fa71fc5db7594d725c1abeaf4bb7901684cee4b490135538a2788b0a488d87a87bf6661a55599904e9c5f757c1d9fc623a4dd5a275ca0f5d1d3ac426296c93c7f217b457b678c41f7b8dc20fa243bd7f0fb5c0dc61541c99cbfd4cf4d016f3e6f765814b379c18bb31c6df8ef6dc47d58da0e561f46077d13915a42b12b99ad8aef855e0a1918a3580b7d0d02f9623392a96c1c66e7eb3d4140264cbe70934dc8e16de29619421b9694de3c17b43886f4a68d68b33eecccf5c482e60b9316dc8709db26254238360205dae359055579e1b477ef6db77737cc7d79f3969266a3e1f88505ad99d1c9cf5155c230def0cf17fbdc189a0c4e14f0ab6aa669a8f6924a43e79eb420df03f5706d5bfb50817cb71acf794b4a2d49a6ebfbdd38ff5e8e20571cd46290b1193b8888e66ac25854046c2881f277e3daa08a84067477d974bdff3e7b2d58afe313e5ffe24521004a4b1b577ed2babb7101bd329b543495cce08fe800319487b956e00b9face4ba977a1a4d097a8c465b6db2d715d30831454a12f636435e72533b6959086d1e977e9a0500129483d09c0528a804eeab2bcf005df8ce09a44222ab7118bf9af818cde25b867f44f224af5d0a9e1723888a623104c72385e020466d9bd3c20fb0df4ac955a1074f8a3baf0093e15e996bc0b043e404f57063e3eac20976b1d3132b74dff662f2a91955b5e9da0611dc56e9de98877d9ecd9aeda92265c57b84224b695c5728a86c31bf0ce86f81085565298a36660087f3b76aa472393b9f98d604e02e7a4a2bb6480115dbebe14eb283d61e6295fe9764d166861afd53ddcc6fe8495514ee0f71c48000ac57b321dce3132e1d867df7771adb10ef298e3ec7e3c304b5b2dcfbf7f6e0d28c4b6bceeff7313c5199ae895089f3041c2f1bc0397b6e720d0563344f986dbaaaf78299b354afdc99946389fd3b6d25840251e4500145610a5c16b3a1f0bf06beed415bc1192eff9f6ace0b07dda788649b743351b55d11c716b9c32975b6063c64ecdcc7bd794246d39838a290af842ab264b6a5cb8cd05270df29fb3c29a5a105601f89632203e932a00afcf6e6f7431a080fe59a50c0ed4a7a2b197e3aad614278ccabda830e3ab8a8eadb88820c899984a5d38d46819ae3afaf4848b4baaa8251bef888faa9d291641d0da75409685af55d37ea36a794c11983733eb9d7caeddb86498e8df91df3d28a8a870891886d2f56d1881707aa93ef1fc89c4479ddb84d82e3213fa77b3e92da49bb5b23e65b7a2240f8d7abed04ee1445b3040082f30568a7a77d840750eeb6905fb5103c568a950f89e16d9d043f23aa5f58f168d6cef560b1ffd14ea3141d0e5e67aaa7db8a039fa51328da035eb00af52779d987dca5666834dffcae094f02a717bb62655e66eb6259d4691ee9a48a3edb3aa4948b50807bb2f9222ffdaa058a3620e5c3432eae602fb8f6e6d9cee343963ff2742660ce002de24281f2b74e201e5d75268a32658b8bf973a70fef7485a75c541e9f9486049d7723c6304bbf9e996f42a2e45b44069b2d5c200c1a3e35ca5dfc923cefe458717ada47fbff6033e5484717f41703fb4ffd9fd584d0af62e86c3a78e76cf277fc683dfcf30ce1b2e72d6b7f1e39615d53f6fc21bfee7d2ca949bc618c089364ec477f39de54c68c7a5bad1f69093d987f4f6476b2722217aa25c160dfdd44ef2a119c16d79b8a6b625ff4f388240651411bbfb9dc1001239400f037a0b5600beaab2ac3a0a4ebb241fcf6218d62b9a61f6af8294895e3e7e6c8eb99d96955ce9d97260271fa8db1bddac6f191ac2271dabba49fe236a7d19ebb4bfb5c1be72e62c6d039e8640cbc93657e4aebf2e47afec65e17e252661e5ff753ddeae7f655f0b58040f92ca86f82ad942bc2e427625605a9243f74a52b6fcd42c46dd3f3ea3c66fd3747be3279e9fd60b9150a5ad0ab1d70cb56ec8c85ab3dc0c4be3b2db8b3a767bdaaccc479a5d2b6f64e107a7d9a2f3ae6583cebcf4c74b0d353f98f82e9a9e95b4620786b5eef6aa3966b407320be5369e04c6bbe143cf7a24352e60c9d95ddebb4b89ce51fe6713df0e777d93d116d20608229093c6274062b24ca705f9634389775c0cc140cbc7c6278a82ca33c39bd218611b0436bb71d3df4e90e22cda41f861d563bce46557996b9224bf7095cbfcb9a6a1bd1f2b358209627dc86f13e1814744a8964395da00fad585868a6a46fe39b51b72d41e4ff6b5fa8955931ea11a4fd72c0224303542f03f030229a0d03fff44f5b3ea187b6f91e31a9283b81d77d938149f933527bbc41d88743bfdd17b3ab3af9008ed8f95951eec1bea6b06ddde58fc71869ee7d2fd7b95ab70d23771d117c70fab7659fd15701cf3bc33f83ba8f2de364bfce0ee146c87f510ff910ec288f3dfb3c8a446a36d60ffa6437dbb8407207d6528e690489ca2509137a3f884b4b550ee6983bd0bed17d65aa092cd3c29436e193ad27e7f2dbcc4cda1bf075ab82a766ce7fc7d15462322543ea41724266ae5e0b9eeb2aae4aa332fad18a2756676b32e8dce838a392d15be73420b579d73d9a9e20fabf51bb8d33468242b8d5cd38490d114cd99d54c352eb90bfffbf79a9186f7e3da4481379abc31bdf7e740458a5bb82baf94028f1f470c6263651019229318f1129fcd346fdff2fb30a51c9226265a411a6ed8730bfdba91b04b0372df59e0954e34765f7ca9227a62cb860c087a09f5d298dbcffc62574d1cab277cefbe73ed4a3c4dbed5024debcab02a5827fe2a7da53f5e9b8746db1f370230f9811d70454bbafb15532bd6e59be0f359bc1036cc4747c45e627c42615fb69af5295d155c38da9b092dc9ac5ab0b43fc9d34676f0753e21ba063aaf541a15dc06da366b2411152f4a6347bab4825b13834a391467978184d02b3917fa0e9f79ed200209b3911af4432e4d438b3f28c6cb24734c031788c05c9f5e112c28257a2489b090338ba54e6ffb5d03ca65455c0a0fb3528d709f822a9e4730436721ffe0dccf5e51cfc8a971f35fc6f3918128551b16a3397b6912cf376812250c383e5d1c37da6153da0d5307893bda1597dc6cf0bc942d22ff68a2dc9b11829a8808721b6a915576ea7c3a11b66f2b9523d415c97eaa34077bbb308e575ced55d8ff67f5f96f9958aa729760e16722a1d5f22bf70d95884deef518e7695f0fe653443d492ffd36add3bdcd74d870238fa442bf944635794bac188e20f8e5496d6b8f1d1dabd484bff08ae6fe34ec3db02cfb8ebe9a6d1c1b7e8bf42e768cec26241f82517c910b88ac46ce0681cbf8d5514f462a4329aaee7fdb61a95ac3b55346c55f1b96d597f0ca0df0075c19b36ad73b90baffb4a423b1f15858ab3ed02ac0263454a4582e987056959884d496b283f90ca185b2abc751ebd6ed081029d0b4121eb7829dabc4db29f0f9c46020c315388fe96b0cd79f7cdc54d5a9079c06c42ed6541e1be6dda5bc76e46e1414ead009be3af56aff96f2e6e05d3b74fe9e3b799a7c99c367137f95c83ff87cc716f0b941d3329502be60086d3f8f2999453d51efef28665f0c6eb4a2e347ed7cac7ec59bf1f2240af1492532b529f85923b013cb6abfba96769de34d5315eebf397a2e70bfeb6f4c4f4d6688bf18a0942b8bf294c227bd3b037ef9c0052120f0aadbd1be0f02e7238ff90f1202bd22c083ed54dbeda675942e17d87ac9e9aeaaf3debaa7f838aeb194e7cef301fde01d76c208ac5317519f49248d951f79c8bdaa500af9e19d25cdf8f59fab7b870378e682bc9bb6a24bc4bb936f9ffdcfb4b1c746c6e621502913de1043d32a7e060717b391f3f21535cf34d7d52a0a32c7c02c47c0d3450718dadcb35eccacaabe61f89a0b719b01b71593232243b0503223699e69e098a00da2735c7ba2b859b7d448f1e68b061cdffa74c407677567e76b440f2aac748a071329ff3d729ecd16937a8f322fb924300c57be0197503097b2d77710ee56b1447d2638e58a87e059d944dbc376c36fd57fd7b452ab10064baea4257b7f14f36da39c89231541381cea16602a974556a066eaf54ef5312ddb1e096765bd7f5f75dd218f0518ac8f6824f47afe1e3b640a84ee9ef5bc515eb0584d999e7b840b908be671ab98dd46f65196f6fb14bfa522250245487b2d89125dd1ab73916b0fee6f34dd0b9d15e784b68eb225f1a12a2732b7a831c7bd7aa358571c5d9f32653025d2792243ccecf882b530df67100c9af3a181ecc657d01e01458ae52f008a56907cc83d2cf4c99d222b1088ed75282e429e005399b14b1cc0372d50cc33a7518ec22e9fb6015fbcf459bbc48754fbb9c7a8dcd108362538678619bca8e51013244d26bdb2c814fcd039d3f3cd553fab5cdce7ee197ebea3a20d57b483460c34f1efd8e555f205b9b0dca16e117adb4221b822015d63ce2c06e939998c6ddf49e566b37baa983e9eee3101eb41f0c7c0e03f5936624834715e86389c050bbd18a792f1240da2d6590755312a60602502d822d8919f60e916307b337cc0181826b8dc46a175cf63565bbdaaf76dd56ebd1156acc2ab7c6094cc4aabd3eb6b37d4d568fdb7ad51032eab7ac7afbcafbc51f6cb08f5de9ae0b03fa4d717b7475cdc8d1b4a958d81debe72cade19736e5f7f5920f8d676a5adf3eeb900a1c7b6a3779fedb314efe8250856a7eb6b28695095ca833ac963f6d041535f0ce6a200f6700fc4130aae88c71c9c8898af0ca7198c3fbf3e72cdfdffb91a1864cca8646a4dc8ddee08a174297dff34f9a3393e1cea09912892a1b617969006a27c05f483854b3588094a18c9704a4a3db3a065ae7e6877303a1bfdba2bb2b0bc49a421566a2bc45a61eec8ba2f2d97a25ddd70a2ad59459ef39735c40641cdcd0e74db868b9de48b05d5c57cd8055711fe5415e894d828233a1c954bda54b3c6866dd8b5f3f7e2af7ce92dfa1c8b9133c15f743880d09cc98f09d2589e976dece30c50d73c7cc9301f4d547c7dfaf194995ddf6b7530e76b057abfe8166de17a99847af7ca7fab81d7f9a1575d68746066e9d0b6efed5c77940a262a1ac6acb3a510eb98ab1ee9d7a061e8872e7f38f42ecd97d8d51a0266da1e46659760708bcec2e5f69d53a4b7605123cf02b6f9381d3ce69ac5628386c89f52321b7d9b17a028efa89097ef517ac7446c816dd4e8ec2b1f2ed028a74b15e3f32e05aaac84668ca79b114524a87a2d2f0d0a0df828902985212af892c0022055efd42fa21a64fe4aab31534447aed675b83cdb8d7115fac783c0c4cf022c5f29355ae25acfa3680a49dcd32fd4c8c43ee4c5464caf74482779384cac595eef1047ab6b3c6f2e8aedb0c9f5f8fea88dc68ac8aff94c99b3aee489d77b1321f41739cbd3259fded525643b35e300e0e7bde4144ff12e0ad7b4722e930d64deae3d67124c862d74a6554c5152e0baa6439b977cf002dd412b9a3a9c69a9bb683fa821733f7856ebf90b049737a5564f51288e4b00b3f929ae3db39a354a6f1de333260a4149d78b14b8fa74f411888653ea09bc42e7013294bc2b88598cfcadd6ac2644048f6ada257b9a0e9835e16f6bcde4e8aa2c52ce9cf7345be1b94888efd95c70a3727ff7d79de2cfe7cc8c1418c3ef252ca942b674d0ca2913d3864ebcd2b5dc1fb5cb624856edb895fc3e8a03894669cb83a599738576564f856d95ade5ed0125eb28279fa8a7d120e87667be84138c9df6a62d0964633fe00e4222a7614c913e10a6a1a527f423230ac6735ea94f9f96dde8c29ed3b710821dd02e5cfc63662aa03d1853422d6bb9e8a805e47d4f92673718373d3c6fb408910adb49cec021d5007e9b0337d21538c3489d6d439e8e13f7f79dd7ad09e515423f46baa9ee5909b06eb3c92e80c4e594a589d1e175b67838ea6c16849edd94b83ae0c25d59480843994e1076fa8d4ddbd99840e025be6eb7f28f3489163cbfba357e37c55e9bc14ee8d273eb94fde6c9129a103b9ba23c35775a1333fd244d9da416dd9fe87a41c962f74e01ce8ab7167206aaaad9c76afd22117c3e5b75704c5eead81f4d988779e323662f820bc858eeaae10999bbeb66089485d3fb07b3e758562a0d10314e1ab8bdcad3e8115bd12c88b1075417f76f24c4b0bd23c3b9f430c8c1ce612c9ad973fff119e7a8f98d304e11f1db995873f16dd3898bf0fdc0de6ce2f6f2ac647c988fd354a410aa0e147792f53bb58f04afa04b0b772a9fd9dafb57f31bbd25f31153eb20fffb1b4d6173cf353d6bafeb7024584d64ad641a312a72014f12f2d2157038d8d19edd6b33f03938390b959e63584c29417d22dd88aea59a4f17a6dcb5d4cb1414b7684cd6ab3083304667e8d238a69d07682dcffd4e1e44fd6c270e960f6c95b3575e447ccce1ac694a3df714969a7f86daee15bd37359e3b6791837884c93b0a0ab6e9d10111f7ce2616ec58654eeb23ad65db5cf61d3ef7480e6bf62342a191602ebb321c85ad1cd1a71327655178b55df5d06d07d92e81c96a22758c32adb5989e5b003c99aa67423ddc48e925eb59240d960c4e24dc269c22e0b9272c39021c7c641e84519c2f315ed553e33a4709850c5ae7fd18dd5c9fcf8ead12cd9fa4e350f880bf1addd372ccbacada656a3a6a22057101cfe9416fb79619cabfed39864a328f6a4fa9b89f60de7a84439093b0c814da9f638bc3c0c2157edcd7bd3f5d2d1b113600bc21fa4b7811fff751b73e1dd6f781b439187a7b3e5d77f460950d447188f4a06f08518990dbbc0e71f873809db9bc7577d75994a22004376756d0df81ade0c8de74fa4f1a07a6286ecc97622208f5ceb2a1a5e77753e83f33aaec09297f4f8a0344477c6e03c478628902995ffadcff0a0dd81caade1af56afd79ecaf4476752909c18f721c0936e804f760357f160a9e1a6e3df1bacc3a432e315f3b49b6eaed96f2e912451c0632a137fdcc3b72da61bda2850dac03923638bef6a78ba716a4a1316be2711d3ca75d6d612fc8ad14d388121c967cbf459510362cb056c6ad6337827bef8724a2d48a61d8100a778c1ddb7585d6ef3a796d4a611fe177927fc12f145313c6a0d00b1dbdc4613dcc0d78946c28b7683f2c27f8fe8d90b88d819f298ef0dc55847354e3a64063be8a7d9a5b4388bea931e5f17c84ec8bcc08517a3662023c378b0b2d07f74d06677fc2e7cb90c5a697ffd87131d6ca0bbf04619dfcbb76f052e68c682cd924e3c86ec1cb0e863854d8597f2614fcaa8a4114d0a2f08f29e92e0a808699632ab5f39466cbe099c21f4c250ec1be6d6c2782f29fc08119c5eee8c068171e2fcf8dff294f769933ceb98371bf9648550f3dd37d70aec3288f16602eaf61fdc2dab5814c8e087e96242697cab92ec75dc2a739655a6ef279988da2d265ec452abfb59c44b91d0e68355ea81c2ceed891038c1c33156e00217f42dc51fb938d5b9d4d092a205376f0ffa2b04569321ca95cb1be7d79fa6d2164b257a469b0d10cf80e3a6af57160de8c10c74c73bdacce363d87c6a59e8ac5894d62f80c34823df632953863a20d02b3d75f3fed1c318c557762149122dfe85d8878564ba92c224be912d52f4f0ecadff234583238c03789d3c787046fdf5ae090b677a38cf658c729495f91905d24727608a6d8cfc94bf5d8d6bb214218d5cbae0cae49b7808c1009d129445332b514920b73f3c6e97698e772a6b05172ca207459aa29251d44cf5cad9ae3609311fa07fbf22de898d050b937503f6debb8cf8647ca454ad45423f8e538735fc5627c5f694edf922469121758229c3527d08eaaaa30a99cf7581caff44ae038c6b8e977d72f5dc967f3b1d9750a770c29edbc81760bdf9695ac9a05cdbebb644636a5654ed09b7b145f87515dac793bb2f184308a91248057538155f13d9ca4d8aff36fb7e4dca3e41bbfa1614071689bb9d4e3e8d91cf1055008dd862f7a44b418edba32005332fa16be09e2ae5aba54d05c33d2e16b07a005365517a14dc800626d8f9cc0aa566cb3682c509ac6032aa2876737716ad395cb2fe61a9b1437b64c2d77afd899303de60daa49dc0ba6811b605d8e3b32827c723d49ae02d95ff12d85317367596044309891eba4f05c236e5580b3cfe2fd8bc81bbb85793af3820000d5ed68bbd491c9087c461322918a23e5a04bb91b8b2d801a47ab4112c00b7a1c38af066bd9a5281282f2c4205520975f4cfb61d27ce8c3e4452ee79c9fe46c1c08b07383add58403f66ad9b8bcf4b32854c38566843acf4269eaf3df0f300d947abdbb40a1ec880127876c2acdfe65428c23028cdb0b959cf4735230f0ba9b97e539168f74b883ccf6306344876ebfc2866f158b1a8dfc1e9013f086430fa113de1041ef232b80663747df0712e2b04fb5e1abc87d0665c728c1f6bfab6caa6234795c23fdcaea862d8bad584a8f0ac4af6c2f5c2ecdc7e21d0d1e7ce234977ad09ee54e64693b81645b467f76cc11e3201fc2310d768e914344891d3c8ee76ee920b7f2444f05aad94756a703a04587fc869f357c46164c8a9abece09564a60969caab83745bb27896c72a7f4784a7dff5d465867144f9e20058a0b8c26a9c58dc7d6c2d110f752a4af4aceb9f5f184cbee1049e1360812aca8d08364028d28873f44ed61af8c3c9a3356ba031f4b3fce2450460e68a01fd436fba91de6a3ac1e56518462d3bdf577814c2db192794fccb279572c720274603d208c109e2a7ecf570e9d1d9eda682514b261e537c5b5c7ab369dff2068baabf1dbe6f57eb9f06b73bde3b7078ffc60b452de92821f57411d57f79c158315e4eacc147796c811c5022a3caee91a8dcd1bbe7ac69eb4a5fd93642e76b095913638cf50dacfdaaf15d126562af7d829f6eabf6ec27883cff581a2ab80e9eedcff54bab8e2e958db4f3573a1ed71d90ff21a3470c47aefb7d8d5b0edbd2803f26326cab6679ad540a850e4bf8dc1e6922114e509d67a1afb34e854ba0cbfd6619897d60ae83da91f78e09aa5a1e564cb4bb79843ab91475976dc45592aa60250f8ed77ed01339a4c6b9ca0eebf16a3a11fb8362096e61cbfceaaad0c41649d377052af996c93e0aa6f816315d4357cd9570160eb011072c315d105c90adefc1587b149970bd422c4144c8b2434cb123818f4e92fe8c66fa0e60cfee7f2ef47b43544a507042bd85d13a70111279f4718a85810b86730561368ffe3afa12a11f6d85b436e16c6cc3630c95521e9f0aa0bdeb5df9b6bd8a852ebfb1ea6a9bbc718773bea47f596353d2de6edab75bd27764283549c59d01df8f125518cb68026d048badcba3a3e06716ecd2990b60643f78f9dd1a8ee0a0483a2e19b5b0950cf5dab9e8de5759dde91342964205dde0fc773a94756d85864640e38b7d561d758b1f6de8dc5de3f87b53ae866968e9d8bb4fc9f66dadc401bfede5bf6fd7aaa52e766f31573dab8572fc23aee0d7c5650f78b14f441af106344499d0f409a1709443884876fd3050c75f30ba35069f61f4f044ef4571687f1b58969e7b39e2468ba08131e7036668850ebcc104d761bffd308e774aac0e208df8221c614c8a26873a40698a0006a87099762342b887e136a9d80ee6e13ff2cb727327ea98470bd67700f7c32592110899a5834e0f1dc050e991964c72e5662b69e72dc2a1f7f5e56a5106735e6a790b78b90fdee6464fbe7453054f09c84ae952df8ff089f301f9b35e30dde5bb213c1165cc3b8b30ccca9bfc46a073e5f60c25e866485ef268daa35862e8af9511346a444c47cafe9f810c9ef8d4c13ddd4f258a7f2d35139ac4ca84e7b98aa0eb21256622ff9ac2fd3ae3aca01a65b321049fc9a80ec0bb193e68d1f53d104fb7732422873557f6ee7be5a736c02ff9923cbc6754fdfc7366123a3b206b91a38e0f7a378ca3ea2eab261007d5c9b6504dcc58df3b6a7c9478f3da0830027f6794808132a1827bf8c139ff10cde50439bb4fba0a142ed60b607353a3da27c88b229bcfaec6fa1d615ef0b9a181eecc5e882f7d3ef13ca2bb003b887d953a54f54c07b6142961205253473e5bd865fa61cea01a1e77f6d884c2ef78963fbca4b10ddad2e4f863461a83f713ec386d2ec6a2ba5fb996c69e8f404cd9af300b1c4f53cbc445be39414f8aa8fba2a3563c4e9dbccc0dc496a8962f3d6a35ae1f34b9cc5e79ccef7f6d2879dd0979dcb44a8a096662022b23e0537a93fd5dddfcda3c55f409f1b7db25b76d4be4664884a5f3e54fa417a919d37f2bffeb645f7c7f224f6348c45daf18dd2c95e5134c58a50958897ac6f181ece476c463bf0f35976c6bbf7742507b23c003893ccab6fd2dd4e78afebd66ff6d55beed28e13b41722542736197e8b05762e5bf49532e3f1b67886e3811de9415672a9dc55281c63b1288996878391380f23e1096082e0770cfac10f7d59201d32561e83ce4ef49b353b438031713f498bfc2b86575de41bae450fb14172a662f107f888e499b4b4418105fd463b874c2c418e42e98b2e42160faa1a233e6d1bceb00fdec9cf17e635c66b5d82620f944463dcce799e516b9116fcbfce53982783aa87198b5c316f079dec7a34171de9132a535b0001ca56a875411e9c7fb93f8e50e9ce751cc748248634bf8b69e15b3f55384b4d66480377e35a4e6ba0b9d7a073edacd1e8123a72d521bd98edc2ee49fd4dd557ab5a6c55c41265ec7b5e57681ec19cd1addad6916eff3940bc289e1a0e5a6b41f4eebf6beb9056ba873aa0ce52c9a77613011e6c2f9ed7da758627cd4cdaa7a67ad32af4f98818bc605c030c684e42bad6dd11638db9a45ef7ed8f5ab49f08552da626499dd65da92310e09d169b7a4db36dc4d7b30f8dc97d3b3dd94cbe109160a243786cbb2cd7964b83dcfc9ddd5fdd2669704ca6bdd6a4fa2db8863c735e915d0e79f1dfa0e5473481de8d170794da45908d42afd4974772f9ae19bc6f7498d6661843c345be7c3ca4fcbc267f140135e410462ff8e8901f3a569e366fb21583eceb1c3d63b097312d53e06d5d9b80fd9a343b9576d760256649e8289c38ed3f44a45476cc508c3122fffcf0054a158cbb0c83cc8147b69ccf0e405439ea76325d1d93e12244379e376f1655cb4551facf7458863c7b3295f18fbe77c66f847bb459d1f9bc80a036c2ff92bfd783a3d7ec207cedfc018db6860ae611611af1a6b1c24dff33d201122c5671446212e69cf7bc0e2aa2f9b7339175e009af155eca72ae3395a922f318a8d35f3bc58b120fcbe59121bd7dd8c3ddeff83e9f33f2a1390688b6ff37597d1f8e60d11c74a4277eaf9795f3f2cf96138866f8fd4a7cd6b77eeead77e828cc0dfb9d18c1ae697136c7e7deee88b15b0a89e494f2ccb1d8aa32962b531ab52473892458ba465ebc7842608d64fcc1b53e2dbfa7d3f2d0343db41271905e5bcd49e64125a81ed98f4d9d6da21a74622550b25d32f24be409f0f437bb3c7ba788e2c716d4cf413e05cbc47cf92f40c28bbea3fd9bd8c5621cb444b460cc5baefe3371e3a92a1f6a2ff999df836ebf91203621bc440ce5494e86199df3b4577211116e342b50f14b41cad40086f33d177487df1f13428345c65e712e63d7dd2091064abb6cdf8ca5da76a2d683713c1f4f721704eb0a07986f3fb73e120b9c44932e8de82a23fbe94434650a5a7aa763cc78367500bdc026e157f4b775ce51d1d412d7ec555bf4720f0a5d13265341135b99eb956e6ed0e9817e36c5f4ad0097b19f290995f40fee9e59adbcaacc2aaae16bb55e349292dd28877da2695655b7522377b021f7c429ba93ef57e9a5536d8cfd757bc131abe855f4acea0f20725a9c799eb8e8e0446aa7ccce91614e86cde331b06e04d489d6c18e63df48a5490a2b7cdca2765a89b79869cb885b9b949ca98b5d3b5c04058a93938427b32430d3c3fdb3621fb3c4ab8d5e83a5289423de64c05bae74a0eb5e9c4450209d810f1dc135102f93c68086eeb8073031312edc76a4fe3be24aed3bfd641c70faa0d8c7e7d3fe29141658aeba4c3541fbb97b6b114aae96c14a3692b61ac501b6bde57d4a211fd35b9a47569cf08d6a16c001412eca8fa40af437cc3ae1bd8d99ec904e2e4ba4e13c46235d850c5cff485c7f5c29066646f74788af25b0376621fdd1dbb719343c0a22651c68df70f57d2b572255c2bd755d59e2d1c64abaf35f38aa02cb73d35dd88dea21ea9a7b025d999d6d591535146a3ac723786c6d7207c9f2a65f27d5d60356f4fea78156ffce4bdf0eca0e67cc26e3f38eb2c9f86adfac3606bebbd5c34c345442ceaa96c657c17c854bbe9d14643b2d54f24903a7a7bfbdaaee5b6d61389d38d876aec60b8f63ccd533b44e7d17e7cda138b77c713b28510d3fd2aa491653bbded805445e4b423a9ca44dcec4e6180b5a47fb6605c16f6e05721c0de0d6c33df13ac9caba31cb58c41419fbafcdceb9adc35ac5950ff3f6663d20b79274197dc31f9dfa4cd0d5735c78cfc5330cf7749c8a0d9254142f47e69434e1b1859854700ec119be6b8d73c4a5849e74209805b4a893cf5240e68749df8c4b0c65d7c3a1d54f5d7d05d753b90d6c6ed5c5a920312ec789d82afd5948cab45059e83f59f9076af0bf217f5ce683e5337f7b08ee56b99112634c053996a65d62fa6844705c84ebd615601a5eeae09999aac091d55afcc64a7700070dd1ce1d01367032c5e6195ec5673e2c436220769214eeaf23e557e9baa5d2e213c1ee36c4b81d371c2f4a16c1e08c7e0f2f68d3562f54b76718956ac5e2fbe756c7bb130c8d4e062debbff3d48ad3190928b51342a59e8d06a146aeb1d354d7fa458b16fdf60ccc09287cf640de9ca122cc9f492db1ea94431073f78a76454f74335f79707e3cdde7a0d17fe07ade368f954b33488c22591854db55b9f52e42a22cbe33171aec9ded3d80881eec6677cac5a909915d735b62d6e477007f943a0234003a2b6a8688539865fba3ae270b60b19595dd272cd8f94d513b56ef46645bd4189bb04c5c8613281b5876fa3fd08e11bb7b0b24ac98b89f5e29435a2f9112842f5d59938217a5fed2dbe64024c3b3ab442ea220b75ab342bfae3d32b7f82c4038b0125b173db045e39c97f1fbbbfa30be2f2c569b0ef1881e119c02ce7db00ba6ab98b0a0ecf4da5e353e5aae3aa94433907028534fadaed02e68be161be3d993f15aa43f31d2660b3e789b94b4976156c24718a51a5f801ba81ecb7a0111bf3607742f41a050ca7ed6a2e925b3bb78cb8aa033c8d378f8d6a5e8a68e0927c8fa67b82405ea50e75e45e713045a6a4740282ffd91b5b06d21952253be9252fbf3132a033354eea1d2b1265469d8fb1963a9740f8883ef50224ee781639db62263d3e699d36b8f7dcdad7dd46be8513a396ff5cda33ea3316feed0233525bd544e5d6f82fb896e8f872a94be1012ab59991dd17503b4e8ec5fc627eee9287a2b06512a32b1efbad278dd43fc9b91830319b882edbd13313fd308ab75bee3707f3fb4f249187a9b6f69805439bad7b6c584ea1766b2ed162e14106364fcc849a38fc3dc44f397bbb2b4f3f3117e8d9fe8da17b74d809b849054c9f40deaa0f309464333665e5b80c33f113a1708543c104965963c62aa452f6313b878a8e60688a23a84ce91ab55143b8396c7f97d220d1bb73c78c7f31d8ecc9414ac78549ba3ab56459ba575cc75ef884cc2ef75ef6f792b2535e4487247cf99bc7dabbfe440e6bb7241787b4df31a8c2f2c0ae87ff13ce894af9ed88646b97aef898b3a8e8da6ce0d8c859f29f9f8a478e509139637e359f04ca915a0e88d692c927444487e8533087305527f22e823a394fd3972f615115534b346600de17f2cb944e5b3298a7eb1e4905e6758507b42ec23587785dd0d2385742ca91d40ad25ebbd580cf508b9ec59b9efbc60fc692323afbfacc025ce8a91bf0fcfc40e6d3418159d8375b3c29851b5bf3eb63eab32d7a4dd947f9766cac7c91c7a6f0eebb77e3301a82a0531eb5ff804af967a4e9dd94795695c3a5d633efe0ff89bc29278b50168371e9f629daa1da353618d25cbf5627cbead777464c6ea1e4a2ab36b92f7b33f7302b3dc82be32fcc7cd927f988777751ed8d1a54845dcac4efb1f14797b4dd83c67eb17ee3955de9432be2a3fffb32d6a6bff34a6ce06f134be894afcb9e55cb34baa8a8ce26a25a92a9922525beb4fba21d1bd8bdc06d5573e7837b17e94b5fbc6c79d6e69e50dea837cb1eec38987a5fc56daafa9a2bf791aa20ca77b84ebcae8290510a753c7f21175f9b633a29deb2d83adc908b09480459ab97cf5ed50b572634c51da9f11047b6041c62c874053a2bfbab127a19df69babfa40936e32f33a360b6b26e722d378a2576808fcda6ada814005383dd529b43a67f625da72c93d1df244126a467bc564a32ed46c17600871506da0a796a4f4d5031655242890423b9a1d5b43346c1899dc9a694fb458f48cb26b4e99483bf80e7011d4d2c9d5f31da02e2717f924af8d5cfc98b04187b6e851588d5d8269a2b741527bea17ec8ed5ace81faaa9e698679789f23ef252f72f2aed05f8e03e68f6c165beb1da704a669aa8261cc13f341854cea353a8f7510da0eb9d772d7752916e3530af2ca06955c1f52a0e290ea7020d846edad1a6aee7e5c9bc5ea4789fd8bf7692a37859410c1b91b0b9524de6b73573ad2a0621ec855e1f1c7d271da9e651e9717ab456ab790ac7ad66a7c226611644e157940dc28147d3e034086febc3f5e5eef80ad6afa290ca93912e21585b7763b9ba4e512ac4f4e4be852edba86da06a0e5f8df4f8e356741f8ee8bc9f9eff7d914e77f360ae32db63b0ed1a841da8129f5b1d463da0db83dbaee8fe2a2c0f47e56cb4d52f45084abe88b4ed66f10b471081615bed413104d530f20e3be33dd6dd95ef83ffecc652805689c2fdfe2c20294aebc5e4931daa78a9dd1b254af1b2536aed0de71530e039ce844b958bd635b7cc3c69e806940d10ee193c7abef650495d03dceded19ec65de848e6dee973effc5a41e1b64fc9d8c51913221c92655dcd027d19140e846973ad9b8aef4785481ca4b6e3c9214700ad3feae16a78ec4490de6ced267ca1f81562b4968e7e755fe3c9ed028d09fa9c832b34daaf0f76b37c3a7db8c4c2ce2feb1748a9f0f8e9e2ed90af322fa54feae61c63007a222b3b629e7ffa4cb14b5994191e6d08af409f8502ca54eb1321abe86458b8303c51b01cb144067089312cdffc1705378698f1602e4916b250e5308598938e6859c5da911b69cd0ef28ab536cae84b0d6b0457186025bf8ef9fc2a54c4feda9b6e280191463e4d3226535310ffb3964e5fb50dc8a8d8565bdca10829c62b44064874c6ec932c02d985938b6aa8963e736138f22da3d66a1d487589687d7d4d19040360aed6edbd08d56a08a21dc5deafa93b8dc5ccfa88c385279d5efd6e7f53a868d6d3e4d1f893f789363f1ea32c7edce65c9aedfe354d7b4230a92c6433a23c047229a608db65c3bd5d3dc446e64164cb2b2d038805d98a39a6694c03854b7476e7bbf8852ae1eb12a168af09fd48553861b031f97aa0d692c886f58c548b6f054862f8c96c2599e3577d54c6ca41a53f418c73a070407b2fe52dac665d04d047e5dfe9e5571aad591f83fec9c1b7dfc2a20d9fa718d9f9a9eadafc4d0f8b16e05ce695faa44c4e0e0a5c03bcd654a03bde911fe52d1d405b1108c6b7de3d618d1b5952bb3cf772857bfc9c8e69a6866731c31c4049d80f36e39175b9d71bd2ecd33e265d5f0b38087ffce3422d63ff27f3058d163c7755324d036cadf7511c8f427c2c506d82284a0c68b7b0d83c8dac0449a7efb4d5accc53ca673a4d6b29e65adda4eb5a1cc0e627385fd4ab62fbae0bcebda7fe2f9d9acbfb9330b9c563d732c0347c208340d30e4b15848f2023f2a2dca08028669483094cf9a99d6d71b2db3120c204c9454866df2f54033daa29d13bb6527b6562e3aa7c71b7818c9042fceec65c6aa8ed028f4f56e55bf082ae5d645cea2ba894679c7d4cc3b2e3ae03e421ee924c5eb2f74411ec44c35647332379aa317a826aea8b3e05b267321489c4e1c3c33240bed21f552982208ee41f1777b48ac38d7ef907a7c67be5f9680736ea2a57104cad90da67ad97edcbad8eda6e3fb19dbe36dbae009b1ec96a566afeb9c1eaba6273f231eaba8c01186c82ea2994d6126b02435fcc266f8d4fef32467d909cf97630f4fe489b9fca3fe0e9ad500ba62890a593beacfa3cf2e7ac897de542762e1702575d4739ab74cfd4c1497b434d8182651dd32fc3bd120d02be021136bd17326cd8cbd8b3f1b17ac256c7ae2fa1d2c6a62e113e356c47b99ff9f71ab80da32f3056430990777bdad5d623b8fd8b109c06b83373acd7fe3813271f13d06e53fcc751b0c4e5885c072ef814482a96830089b6290f95f0d3a8ef3caf6f1ed64532e54727f62a3c6f24d2541976a2ffe62ce112c9b276b57dc343ac0cf84c4d17a0bc21f633dcc843baa91952a575667ce5f434bc03efaae9674d62c6f266c4e736597b510aec80fc675b48c913a38287b538ca89cee146cb9027f9d5a6bb20c9f9e2d8d34da23aff6bcce9b0452c16ae54cc9c1321c5407f210797f98bf51311229859b484bc03bea332dae5e5e34bbd36e7a8475e9b1cf461e3a77a786608b17d09b197914d56886189e3ca408b49d262f2c520b3ac8bd949071ce85a5e8f467848389c883e4bb60e89799114e11ee7fc651f11c4aab8187be42d69dfe9f9b413b7a0dad4905f2aadd5e1273e58cb6b57e365ab28747db417b943d30e89e062cf29000952d56fd7720758a8b6ed52f76aceda0077f1a68d64657119732eeb4217bd5ec6eeb9d1d63d1d68b99bedce3378b1db6a53515ebdbff326e5ea7e46489caae2d2da931bd486654feec9352c50aa75a5325d89e2c4c463a9cb6f5cfbb7eed819986586c148e5107df469c97dd0565c9ed3f8763e0af391334b5289fe192a29c72522e35ea134d8d673897e8ced5908bafbf4bbbe2668690fcea9c6bbd06a838cbab05d295aa2e2ffb04ac1d81cb80b4547e7d7f47ffb34ade8f0cf98410a6b8ad8b6b73c80b438a50f4c6f7e1b69b9114957ddd39c392c666ca33326e0759d883e6e7660183aa501ba4c287f248cae290c4fb409d646fd463e3682f19d1258857ff708a5ae2252197f3edd7879356818cb5e141812f780c133896ea8413c69aaf906626fca5bf82698243f2db63b5df572bc082ed8594b702ed2f619a94d9ae537d38f468ae2dffe7ba7852fc57e2bca49860c658ac3a1806b364696a6329bc679ca6502e4ff6cf48a873555d5bada2d4e3257aed79293169dae0b0f128455dce9544e94a38daf785afaa7a5783c6823523b9f28035b1916d8ec5fb380d35f989497116e26f2d6ef8a412a60673efa3c5f5a81834afaafe48c2e6d6aa334db30f8efd59196bc84851c6d36936188241887eaf3f010cd11f6e0af5cd2efb516c7b9e3e0be2887542a62a1b2778456efb8842a001675146e7836daef3ec80af4ae83b998933334f47d5b44f7909713d23b41bd525d46d26d72f86e1f734ae7c027b716996574b8f16765865b66171686965f9e7d4d91e2d1c31380e19dbc42c1aeb5725413ab27aec8b45c7e8bc99216db84fee366dea9da98842eb04518755bb7a01d1203a85153191575aa2fcdd1bc8154715cea3a4f3992df7edebfcb1565d6672b6c83dae7d77b6a2a311a2be120768382484e3962ece159bd2bbc7a6ca44e69f1aad3a70c73e8379255551421851221dc41d49fc31abc58cf41c43409e024284b2c86b1405e1bd8817c58cd770a31987a28a7a830d5d6fca0f94a1475e77724a8e5cee8bc9a93b328cbde166dc2dea0a94b9f9c439eb731fde854dc5d53f97623c10a951f1bd8f849c1d29c972eaf96d36a5e1d1c2da1feb00929028db2952f204645d491c22bbd3aff8cb29dca9f69e1b4b40a0c606d12f664e27f22b240f19a693849c4d9838eebd6bd4ec12f185ed5991c195df2b1616b0895c4ab3a7f5c45cc5737c5f6a70342eb0d11811a244c0afd743d4c21a134d6182c36fd4b19cccf5a28c1bcb19326677862d9fec2f5406cc4b33a7aad77ce70cadd4aa74c82889e20f24461a73aa4f09f7b617e39a7b4c6210d910e62177f8a3a879dfcacf91e0dcfde087ea477b96d9438b1707c20400808191fb349105ba4f22428cab850de36bd3de7676cd46d988c0f2abcf85723f1bd7d51cbe587e3955bb183d13cae8ba860660ef124d91ea006f5b84f319f1d2de1cec41ff084c8916b46c7aae26bde58b3a7e50842d9bb0d851248b113bd9ece1643a0ca95c0d2586647fb2f886879f89952a139f887ed060bf56a81d8e20e2a17a6f505dc95c7691b08e9ccd6c04246c3f2671762036354b947045ea03500d6ff87cca6c1245159dd07ce2f370a05f5ee2d4b7e3d4110e50bd15e3f3087fdf0e002d7b89510cec4f86b289a28b4a35340825ecbdb81de29f05a0b223ff5d6399a1ccbdd634326dcde0785a8f0d8b88a3db92db4ab1bfe8bfbc8c298dd91e5624104841c8551674fcd2540d41b6efcfa51e1de928f9ec55e1d06239fdefd782a8b8d28fd36186d85d6657c5dd9cbf1167523c8876b9342e44f73ded3d1f307d5b2b06164f3b3a50cc215856b5597f3704e10b3c9da279ddd60a6863c07c7a81a740267e9d279659d2d0a8a9beeff08eec2e3a3e9e4dad3221fd65af7aa15af629442b4cd89b5f1956b066217ed71b1a0c9e23fa6e4d2ff7e7e2bdf5ec078743179f55a76cffc4b9fb010ced0a1f4ca17e3ccf001df357882f9b6bcba5e0725b07d4d23d82149b0ff5e2db83333b8dee015bae5040b5dd570f92cfda28b0ca1821d898824a91da0cbb034e47f4beb1b152fb295d6c0927a493bbd8a3472ef727f97b832cf09edb2df8bd0f1335dffd32d7075e50d29e552af08c7deb045e54bce805deb17a63b735cc77a5aad4d2dd47b1a66111f8b4bcdd5cdb798e400d5ec8972cf6e0cfe5d13155c01f135a7c1b091371f2adfc02deeca937c2f9b6476d3a07c345f7f97984ae0e1061c76d629438df92b1719269e3cb5a5e346a158a7b02ba1bf6c078a1aaa493e0dfc75f260132eefca270e8bb421691b9538c798652236b787fc39433136f8264540f32e2dffcf3d0610e6fe5ce1e17b6c66564f5622605412d90163fa6703b9004173a06ce1bf1e8fe824b7eb65fd8c242fb2948e921eb78fe758c8e3e1f847edf02b63722fa68d6431081f74ace0c6a60dfae909bff48b25b6f878afdfeed0f7d6b62d7f69518bb852fc407c1e90d70f705d1e9f1e964c5d413d1763a93d612f1a61276c57ade2788c61b0d5646b34746fbce6331d979a23869cd5039d74ffdd52ff3fc055625ac0d3f1726152fa303216bd8a85cfa478a79f83425afe8ea1c6594afbef92d3686771f5947e1d28e11d330bcf552ae8bfbe80156d10b1a61bf21dce11729a54b645fd0b5f74f6cbdc42914e5116781af44f2d10e784e8841bccae00d697ef2216b9417ddf321668577cf8e86430b4e47964ecaffd358b683afa6d6e67f22282cb7ee97ef5467c214fe067e4d6bc77acd328accce7ab2f406b323b74f6d569a941dda18c64eeea8d9ec522a3221081b688e8d4706a58b7998ab7576477a16bf173aff61dafe7d1529a8143a13a911714872107a9e6605499fb51589f9ac72183da217094569c27b41e015c7626efe4c33e5eb66f18e7a6c501d7980f383781e1f6c1558eceafa186b93b9103f2ae2bb04da37cab2e9eb869d94f9d09580a0a0e1ad8137a956f2c046ba51fc3a20ce9efe8d88cb988244a58e7829cfa205fde909a2b1406233a51228d00a76517f04870181b1d25551a706d977eda8175dad5fe17a0c445329f597778a5f6778e19af9d178f6d3836a35e74d10ef6f7c16f0e5d3c0eed65abf5a901d30449f2b25fa73053a8234a127a128ee5d3fdf884f09c7b9160c4f0b385fde49e23222e71bf12eda2cac730297da7b564f864c27eec9d2638ed220cecb997ae2d4e5f05ec4e5b7b2cce7242d78aef59888513961f0a0338c89431b797903b71e4c77fa00ef067fbe64eaefa1a64823d13fa0a57d367c72f1f0d3eaaf885e0363c901f5e5be28b68b2e7ff9e24c06b12cf56bab385364bece7136576034a1eb96df5aaa259ffa827d9e068d14050f2f6d9dab436c6bc8426b820b0e6229333f4c9501d9b9c0485a1f866ca75bc898b89e174d25b61d45a89f50d8ed72c01d6406ff5054d6bbc71180a31fffee9684f5d84ea982ac7b3bbade2ed988cb8be81ecc92f13aeb599ffabc901790fc5d554fef09ec58e8fd15b5fa5ba335e80baa24ac2a41fe600969e8870454c2b0b0c9560c7f494b229cee5495ae5b205b2a2fac88155c19b9a77f3c3a20dc08db76e876cea8d11ada7ba375a2aebafc17e1046717e25ba2617d665ff03a51203b9c3571ecf0fdcc54f307005c0c1c7bb7b7373eebd3a60ddaf7a76d329f28f53d74dcd0b16bbab305ccb102a34ee005c20fba1ac7da77831f33f07f12becd0c5764b67ce07574e2c7eb53f45729d467ca3aab74e213cd2b65a4b175e9a275b10d1aaccc636b787386173c381bf4b761b5b01327f5070879c05a50841c5f37e767c7e3038b590b2d23a8837db5568c1979e9ac7ffcf71da46088549af1c9d59a177011ce339d60efed7cc0413e1b0f139d55db44b83adfc5e2a74e5dbe0ccd3f156f631352da966a8fbace58af4ad5e3751189c668ac3df203d98c57df6913b8c971a86e044b86af1549bdf806aaf936b9cdb0e383b4e8ae9aae1e0c31c4e916aea27d4f1a17595a54e456793220a0c3f26da89f4155be09ebf233de3c9e58200a0288642b9214a6e6e02ac70bdc240427d206824fabce5c7719d1bca07524227a0ab70b1f7a44765f8fcb5bf783da8ffebf043ac5a5efdc362fa9546aca64517b062a7787128b380fd28d2ca06999abd9deb4b18cf3726cfe50be72578b00aacfb883ebccc77ca66d3d398eb7d86f8d8ca5bfd6a322929f393557ea1d019f9ad65f1047c300539df364f29724d33413fc49d85c1cdb957f26f36cffb9c33c54108569335e1cfd6139009d89bd995e87daf9c414c97f036a3b31a9f24fbf0730bfa184b9c2c94386d2b577afab3179ff13a0ac84ebf20b2ea640a198fb13b8f3f50b4c98f881a5cc28d83add3f1c5d41c03f6abd66692e9d00ad76d85d8685726d61026a39a18d99b84cd767a28e004e32d2b160116244fdff9da96c73e068e113f10c1e2144ce47fbe4435173561c109b847306e416514b543597c6544ad141acebda9e19a4e5de105fa846d8c741c18f98e012fa7597e0e2f43c789b4d9c562d6b54ca991855b9bb2728be17385724f295204c734815186912150baf916de3d1a02f95fad742cde302174be353abdfb0fca24a0e88eafb1e1cc8b17afb41333c4ffd4e50911109169e0470712c211448ed315c632be73c1a367bd8026ef538f7d6afef0af57a722a4ace05056db82202a2b8bc1cd89f4663519644d0ece7febc55728f34685f2a35de04c11cb8d9dffbac57befe9810450a5b504bf8f3f4642f8399bf8c19535fac4b22eef23fb1fbe969973a09a2faff937c35a55cd99e21201fe4530f7bcb1a088a2d449230e8c00fb5f23643a2ebc372abb969c82b00058097849222a8162a4fb03687df9fcb59d1058a4b9b87dd5d8667830ce4831cf43d3b40746f7a61064c3d932e2f083526ce8b5ec800bf929353145ac2b1c2c28332394b9e34e1db0cdd154471758cf30e69c4abc0ef190ee0748d6ce173974714a9a6d9eebb8efd5da0a148b6f724aab7154a93b675fdd9e42c0bd66ecba78b29609749c91d527c6435a3fedf5e3b7f6b3d0c62f4e97a2786d275fba90cc7c7ba41dd562c5cab6494ecdb2b0db4f39c402c21d14b269cf4ee49e8af102cb703b6b3a15ea7c75bcbe6b98176285eea8469a265552b98853608e7bd5b5aa4bc2cd93a9665ac11e3f1d30e19d319baafa70fbcff24efd0ff23dcf64333d86c2eeb3c9254551d92e2cc32aca082b7fe31a02ce8c70dfad0b148baa26f4be20484840fcd29752d56ff78598da7f2360f46b357829db9d46fbb7bb798d7b5de26d5a131f1629b4ce48f136f570857b811e75984af491fe08bbd15743e8b41f3f8eeac89417cc449008bb9c35837da9773060987e385cd1b9d4efa908fbbea753b0fdeef8e89628216278658ec19975b68c0846fdeb1471558ad7b00bcbe00b62dd50f3231fda78b4e829654671ae1625fcc227984a5e335bcaf34b5cffb60fa40713012297dc5611b6364cc7944ea5d5be617cfdbf19af035384316f4b191827423d0136fcedbd8532f93412b841d7761d301f9193387d1e0678050af170d53ff83c779836625a3509fe8396b9a5513945f09b9a6857ec172db106fe5a7daa71a97ed64f15cfa7dcdf04fcee7046ba0cb89690e16b74f385ac5fce991ee5c97a581fd148ed09425d660975e2699448126fdc64112654e0f95f1a4e78b491245ac0a9a1799c09ffd914d0f06408b36ea9fbbc9c1e165f622d204f234a4bfe3b13688252e3210c52668695c1daffce970e45b377b28612ab9a7ad651a76cba2b17e0b615e9593bc7619b83e4a21b4dbd15ddea7b62a722ef4d54e269b72d392d8c2e622dd33b3afeda8f85ec79e9bc5950b2c2ebd46ee96339904775f52d05e726ea77504b330194382b12a0732dcc7fdd7d0009162ef975ade320987170b79fec1c8ca4ab547322e013bfd9d307b23e75993a197eaaa6d9d14be4b08ce14b7741b060f1c94675cef0ce6db9bc1d04dd7c4ed51536ef3736ed60d9c337d6f71cfc4337efc152561be57d25c217bf8d8462bb8d64169b377f6d476a300d6b5aa8f41153bcf622091219d372b6f0d9978af2e8417e0fed9f5d7abccc01b5978229a09e3386543fd5bb7f702db846a807a6d7d3bb97ef9f03fb5f8d76cf8b49aa361c54d121c808a98f27b66462157edc13324576bf60f081c88c220cf0b1e2a25edac35dddca5244ff94268cdfa22b0622f6a8db9cbbd25da28a2568c7b60afc159412c71fb8b21f595922ffbe5032b4f9dbcfddc3dd0ea7e9855b390c89016783b37709fe2d735d046a7e10d0d8cab707af6f211a273b915da84b2522fb4653acdd4bfab0cd5dd2894ccc1a4a0dbbd1e9b8d8f6f9d9680436790db096f87e270712090be728c096995ca534b8c0d27ec6c904e08807b45145c5f02804c724511290c332a74dd29b95653fd806c4c5a68b40e20b8806dff07f07c36d5fd7e36e970a6755a750843a2cd8072d906b36c20436de92ea061e5dedc0ca761bdffefa0efcef01b7a8ae500da58db721c0c85f840c8a5b3e78445696aa09118226db0f45506683f3bb0c4c334c03a8ee1c3b34b789616168b6ec28bbdbd3c6ac7c2e11e90d0670dbda612d49e02e5c338f0f06d9e47e9643c96edf2090e4a445d49274d0a1ef6c09e401b4a3add7bcecb1d887f12bf3ed35ebc5cc7e4da35af28b8254e34044f004cdf722f07f0d31e95d71be91558d1268fc5a748619669a42fe0c13d1661c31788ed33c3c5d816add9265ed832b82a9346dd766316fe137432b0b228a71a3676beebcff679b476495d2a7749a5add4526a2d833bcc3cdc4a4d850d078640470c0c1370167c89790c0f7eb3e486231acc424572168385cbf73947e3a1ecf5ed60b82800df37051dfc7cb3883d589bf4c427a13222e4632347c3a92af07b4f42e420b39bef7f88f967b18bd5c98de182be0dc60d1ded0afeb7e779104177c93fb2a4932c685fc00635240015e6a57ad692f2db5364cf9326f6b2c285961b31972194bc1e191b8fdcddee47207d90e42634322b10ceb963f90599f28cc9e6260a95a74d11f924b02b4faf483d82755ba454667143cc4559fa4633854f69105f5b6f1511c1390ebfa741037c9eb0395e1c10aec4c4ef8d61cb251ebb311683f27ed4b442194ba0ce9ac9abf0be53f82bd946d8bb528c1409003fb8f48e6f291c1b0e4cf6d4f5af246703876aa768a0edd157af5d35c9337090806c15d9593c6196f40d602e760356e5533f6100eacbdea5bcdc01e7d9c3547bb20fa24240192265cca48a0c2bb9d064e1f97a649b545da6a8f0468714b2dd23de73696669bd60715c029115968937bd79bf3ebc5a38e649a6482477331bdb85828c454f60302535ec633d5243845f7ff4b724a94057ff02b2648fb82107ca1ab31a51d1891fb25069468446a93d95cdfdef61f5607dac7022b52cbca3d305de8d6de8cc6609a25a68cb33b38a7b2cba849334dc669cdfcc65f302f156580c2ec60f77c20971a82a8a95cdd5d6f01b6e9b9708e4ab0a9f8caa0794a14231e9437e8befa35277e11fa5f9fcbb6f69376af841724fa037748e200fc7f73ad39e3ffe0db7cfdf3645cc5c817ffd2d5ac6ecf9caaf9082d558e6553f3f8eca28875249bb92636d1a5790a11924dd45424e2dab13fb3aa5424e434f15a6c0e26e5b56d5faa0bbad5a49709ab6711497e785f2797bdcfc1f041d20f0ab09a86070e19c936479a3b1cee4de56d15f3365b580082f8427c10bb0d03321295264d37bd949137651eaa69d60ac46fddc03c4dd1fa33850a996aca77df1cdf1e2686002f07e39ed8d234d52250e5b9ae602f7a5b0e028afdf4721f4051372de1503215a93691dd984e9569c99d403311424d1c7c61bcfcdc11334b933f3e132fed4acf37667a077f09257559857fcc3d12b92e06e2ba65964c24473d23e4edb4f662052248656c6be5c63493273c77803f242d0ce4c669a3b2dedfc13f38a1a78653ee8c095bd3d36b6f30e7c719149e7ff79489e1c9bae4b623152b9fb6a6a0fc44036004abfbf6a9d8b6b306f4ffdd899b8a50ba88c9c4bf61ee58824df0bf193145b828d2db5d0450f309c543b51be9a1c474d77e85c33f5eed65a7ee8f95c87c6c38ae5557dde00685619762d30574715d0bb5d71822cffcab2e133b58353fdc83f678c09b5fb8166c3d82d1eb1da809d0929fb7a93de8cd15c74359ee73bfc2375b2d790499ea8ca5ca9523a1a404c67ed70812c23a1943c4ad3888ab36a87f1b6ed9a5e97312495598117766d84bcc430debccd1dfa50987904a50145f9b00b1131b086af33e598778030c54520f6a9be19b81ec838586d5108ab3fc108f132cabea6b5399a41a77c8973399848b5d38ba01c7b24aa374585ce9cb23b292266a41d4156310520d276e4650d34d3be5e66cdc5ffa5dccdcf12619a972e0d6986c2896311831b7650639e48954f2670dcc92e0368ed9b597c98c42943a01d252df72ff74a36bcfd928e37f6125da8d63cbfca1683e01ff8076a544092e80f84c7834b996554f04f638f953b4d9e6596e702748e54ec1197b92cc4f196a78a2869a53757f714f9ad595f6da99cc5f87ada3e51ecd0a3fc88ed068a8764dff1bff85e116b7d17095eac668216c8b18315e5e99590f65ba1714689ea5660f7dbf078bc29407c9fb0a4025a773c03c22f68ff234bb58490e3e2b17a6baed5bf40c3be22cafbc5220e95485cdaf266a34b5824e727c810daee26d950e7acf02a9bdd779d0e5352cc93f554f9c4c5c592d0cbfb6abb4e62ce00a0cb27bc91c9da4bd1ee83cb2d29408ebf144b8a08bc5b7245dfb21aaa9faafd20339846d3acd266b531a6488a26157e8f82e812b340bf297e6b46ceb05a7bb124c706819a530c0b7c177f90c5172c53100ebe5d30eb491c7db8c928396e7109fe36fe550abbc4994382f38f22b8cd3e48eb306b53777d623e86cb7b52456b09f842c721cb95625b8e335db2b10e222dee65a9e17bc7582c1af072b33eebe8bb85ee265f3bc441d0168e1f7231766e57f6a6d1c92a0d781f57289739353cf654413c63a295daf5ee45c5c05210a37ab443be03294d63227d67749886ab0499207de344e7c1551bdfa4eb9454a9ed2bbaccc95e8cff22923cb00b8297eb5b8dbcfb1570fd8b8c21c5f2b0ecba367a241b55d2745fbbab134f7a49616c27ba73f49aeb09a1aa469cdf9c31a4971020d1cf41cfa0589067e750d822965ac4c131b628ec6b71a0cdc2a4c49bdc7bead7fd04c9a0a829bd9ed6667a1ae77ca7eff79bb6e1290ae326ea47c83fbf919736c495145ee1024a1ff535330f7321242f9a6b4281ccd2c6001eaee68f3602e34942cc26da1b76fa9b88b56be68bad8e6cd656e899aa6d513b56c2d2106210f3c2f6baf3215eac578a20af3db2d3b9d2579f238f3efb37d51308f69affb6d949c3ba44a63e12945095badfc6212be0d9996ce0a27e13adc14dd83f4d9ab64820b2ef961a75450bc76390980cd8430ff54a3adf01744307979942894f8d36f2450962bed61982145ebb0e24aa23d3b1594a1da0e1508f19bec3aee440f22a4d3267d8e0b17c7298cd030c2a83d132e8c18c0da96ab5540563788b0adfb2da254ebfea75435c33285f130f2cf134b5b78db06e2bf6daaa349932ca235764e620017e0bfdff2353d1951422cd433cc3e64d9e55b506e1e89881231c3b7e246cd41e3de80d82e21e89f103d9aa887e7f48ce5bfa91cd634a2938e34321575fb9721a9f7a7722b82b0fe28103b76edca769add49603d20af38512555d0197291b60743a60df0dada5aabdb16ef54150fb48e8d6ab48d8e5f9507a0494dd2e96d2dd47446fc9dd630cb46e016cdd277a9cf40e029d7ceaf98014905530eaa14bc183915ef3c917dc313b4795b75bd271506e60b0a8ca755336a657af0dd0d3b785f76bf08e0eee70cb51e3a1109e8bfdfab2ca69f1130dae1174e1895d136b5676fe6f21cd52b9268e3d3c0354257f717d3f6d4980aab4fd2fba15bd29370b85f76fb115ceaa09caed08090f074c77220bee2a6623ce54e433c8864bbb02b7140ba34bca1f0a075c05b49d983e0efa7185f07e185d75f4614cbcce8821432e35592cb7fd47fb24837361739c9fc7b3e2526769b194b493efabb3ecfa8fcae58de8f9ca3a2829555a9fb1253276fab030802364ab9bc98c1c29db6454810f8146d6ccc35d9f6bd5c7c8a449bac5793bbc1df6691b456eba54cd7c9e212a0e78010d9241995dacdb39c07b7f2bef3b5f3ce7eca24339abc9e83b51ad8984f37ba6f1d216b39d416354762b041d92ad6cad54c231a4beeea34740bdae1407d3a0a9a33bccdaf66ae62815d99b7cf9f63262816ed123ec12d5508d3206cf8632979cda3b7315eccc5084d4f07e4a240c28afd9842c8e3f820764d89d2023e22c16e79e96747681ab1df949c1dace01ca0d06e5f4143eb307eab19b39939d77012a406de288b4182be475bddbab5500e04609a3596b0835b2c414c863796618e58bba97c22160b2279c97ffce787bb45ba7bf55ea1d5f41a78f2455e5bb24efa8a0ec3c255f19dec08bd2ffc6bfb84c3166cc91dad026bfbe933ad20ac1d17007ae7dae41e922804fdf96690286691ce35ee8e69369494837e7bb6d8db01aea5d5f968294831fdbe4be5d50694432773cfb10c934453f25ae2bfacdeefddde54309f41d461d3aa646d30dbf79afdd00767fd4b5f299ff4a0a414962c80f6318a637c6f151ad7569415e6ed5616214146e0a33f296c9b9a7552d578864c703652b4d15ca07e30c999e8287bf553a6813adfdf1e4f0aaf543b31a460b6fa2cdf7c7f756fcd7403482215447bdfa2ade7762178625c8a3f9d1e2ace173f9094e64b7844a503e5c52ba13f7dc58e0942f6ae7cd4b019633a934229842dd1ce2be85b9c86b6fd98df1a60781cb64f8ead73995d4ae10fde01a2da44462644e2c1f9d6a13547fc270324cf39fa8fbcb0252eb471556e2da90db5469693e03c02a6e151d28e9c9317527c5ec46b6501a4d473f7e51e16b38c68e25d36361e45215704a328de9c0655bc651ce63088f1fe0d45f39e10a255976572305815176b0e9d67bd745c76d8b664ebcc6e63c799901ad6b5981eba909c7536a42d3b034dc17e0259d103513607e6fbd3409e81248204068c8381968b566cb4ec6ef35111687f3ea0951c76aee08ec20b91356841faa7d8a3235a6439600059d12111fba7de209d5a47af596df76d982b5b0fe4694e5d4528d08d1b11dca3799590db9eff1201408ab197c516fb3e28d4bf29cb1d9135bfe8a993517e947239ff022fb891c7b4baffa87ed2e4e3f22797b7d001bf610e064d4aa096cb9cfb0a04fe473c2b88945b0820475507198b4538587f6d851a1faca539bea37c163ed3b69135e05ec4814b0cedc89ed898babf6498ac70cf0cd0204f23ebd2a74e05ca81d870adae37457804565ce15561ff6d59672b37a98888a9019410f3709cbb74b9ca2a4773e2133728b78af54f9dbf34082924232267f2b9a3d0c19d52e7de6e442946a2b541b2cc555f33717f70d0b8fc47b1f8f241aee4c05de87d18c612b12c862ecc510f6c0f51ece9c2c7f328f468ff55b1cc52dd7e1cea4c592e97304011c7aa8c5aad1748b33010ccac8a2518947e43b5bda284bb4b9e577401b5bf82658e5e32dbc8dd3ac418a1c49941bed7caa32893ebf272d91fe09cc105dc0e2389e68264c8e20e3b6a9e31fc49f44e268deac0f6b23c265a6a0982e22c9c27cf1749a3030e472e8cbe1dd06f7e12de2be3134ae56eb6b05218d44c96bc12d55bc19db8e9f59975784cf13ab0aa075af63eb777206d671573ac568f56962d881f2d58e10bd004ddaf591bc4fcce8fc65ce8de27aca24f686547e97e58edf488e14f989f882027559aebf144933994e5c346dfc7fbd66b58dfd06c3f366920508b95e1f15e29f06bb0c0fec4bbd198316fe008f97638bb1411acca16c05611b9d62ab6ba0772ff5ca6f07f167a5729a68dc2cc04d8e6e1f5a98230c19a6fb17e1133766551e7ba158180f52ccb82c7c6713c69a79a46706e705980cd6a2813388ea814d0bf1badda00a2accbb73ea6e21529b2bf27d85da6130ef86ef82d61127c9137c9095314c2de202635d30e0524c50b37504770601322a1d43a9bb53fa446f9e703a9fcd246dbf49d3eff0d43f18ecec5d8e892d16390a048fc091f1d24f55a53ecc38ba514b780d1f20c28997edb0161edc73d19dccb0aaed1c1f5d9ece10c095a868234266c56ae08ef39d100f1680a53b6a07ecc8f3b54e48701cba991e567afa41565d29f72d4d9347ac35702ccaf3ece575604e040e5948e5213c7a2c04ac8301078e2066724c82ddb4ed261b8ee1b1c0da1af2be3b8703492cd5318e97cb90f78584e1fe895f5c41a0100cf7caa6e96267fb5f07f5dea1006295cc76935455d90571fc60f5d716b97b33cd1b57fb9f9df5252a485891751c267e4da6116f16a6128c6717d822c9ea28b1c824050405788c316a81acb8c69bf40f1d54260645c791c1be3a7679cb6bb9ca3a11be6d36298529b2e261085b06cc72bdcdfbc14029c6f811eaf5e7e884fcf9431d47afd826214484729e38468b46ead8e7d71f2fee410e7f3aee11e246c4b04f1b64fb929dca09920bbd37b767ae44835df6e573ea921975c4a57165f22457b119d7d652f7a3d6fa72e28d143272b61977c3a844a9d040515ac11e86c2681c05e924c79a1a85e3dbe59e7590b7d4a88c355aee44900e7eeaa7c772a717cedc8f6336c541fc0bacad3ee5344be1ca7e90033e70ceec21547f14396cf40421d22aca7e44e702810ef1b6ab2f74aafb8dd2926c7a5de823f2c99d4bfb3f217fd90730c0d37fd5c23ab84f93fcce6b0c2fa92dac3e2e72f82bb6da83082eac780dfee380476dc5cff603adfe4181e2bb874f228ee211fe8834cd06577ca17da7e7f0363fcc7de5b0383413113dd092bbc8750e89c52ae454a002f968078fdcc190d3b684b15857af7b0f1238f570ce7883eb40ea5ac4edaee508f17226e349961928f011122ec2a69d878b2fee4ae653703363fd05c88a37844deb0b5523f741cac3cc28d37c01ed2ef63a39b2fb9106d7723e7f7cb4c1e4b855e3063f752178dc46a55e29548a9a3dc752e33e29f12f7578ea30fe83066c6112e5044758729593b7a7897cf7ec19f16b2f3aadc7baf120bca945958645d59b03cae8a0343d9b6cdeb20ffabd90a345e8f72ad57282042e9ac9a3e0114c8051e5c164e57aba2d2dda4518a2692d952925bc7196e9f5380f857c0fb2dddb405145def9ad618f56c6aec42051f1b8b0208225a539801021cb1397f0a2c7914780e008b8a93b9d4588e2fcfbd213832abc78e233c463a455e0ec0bd88002dc7674ee39212c4077aa163a9270fd8f9b4b3ef9cee861d67a855febdeaa1f1595e4e04834a6164c4e011e1b723d3b1b83ac2da13fad7f2ecb9688740bb540b413c06926b5d5524dee9ac2ccd9e510c777e7355e58219fd1482f495f7113d3734151fcc604ea3a16a8d106938925ad5acd03b5fc646903726239f76beb7799334e76e08bdc36612ca8d9281ed8d3a35c995e7ee86352f859cc27df2378c5d22bfb2e95bce8c44062d79ffea454006d971b65aea9f7300d177dc768bdf26dab4cdd07607e8ac639caeab5b894b2d9019d98dc1adf130a539ed8a4d0bb190acb38947cd4c142f7fba9997dfd9ba7b8e0c06868bec4ac41134584cdeb21e446665915c28f41739115b1f3f8432b3abc560a3b648a0b7266778ed151a6431379533334289bbbdfed69a6c31e5998f17b5105b13fe23f76c3cbed48fa415496b6d6028b5fc26d882352726ba95252893cb9d58642140f94bf92b8a2eb5d59ea6bcd3149a5a005c99512a248540314e837ca646e8c5e76f2ec08c33f63e5689bfa16e8064ff9831383ae311acc5534dfb4d2d76c4611586cf854914c9cace1a421f16938314b701d8588eb30f3a0a161a591810fc924da6f43e532d3c3635a5fd2383e050083051fc73e65dd2ef7beee0ed49d5024ca928e2c4cecbe30299325a54685d8e47a3baec8a6addbfe4b160422c161ad7d405b7009dbf449faa9f8a352facd628261ce8293459465ed5dacb88f63679e78864601049a0301a38f755fcaa4c77d91408edb57421fb61527e1fa00e41de08e570d588d46a965c464408ecd91eda3c4d6923acfe446bce4637bdd366c51ead250f3e2b29598811ccddec1e96588f67ea99bfe5f634ae1217b383976249ed22abbe03da941ac3e5ba852b04e538cb7e0870d87ba473d3b87ba0c4a8ae1b9178206f5d41dd29fbccb6f45b391b23f6b7f4df88b68cafc442309e7a90e601292b043d502bd416668acfdb312ce270c81bf081a2606bc78171ee7383b332dab786aed195d507ff15f730f1fa386cd22767ab4cf7c845db54cdfd47264e32c9b0eebe474ee4d4407b3190ea0e08f7e99b673d7b556453401d0f1e4b46045a70bc5109a1e06a3850f81813976ee8651282fcc0d8e5b194c4638c9e0890d0753282f3e2c7fe6408f6989c61739cd91951466df0e8d85c6f582f21ce29821a1d7aca54c1af1fc87189745ef8e7cfb12386754709f32af012e723afa5cfbadc96bb49301a673c2a89d64d7f1de95a1339aa6814e6a1e490f2c5c285c173a0a477c7b3775a02541af5b96a743de29d30da918400d6e8fe28470ba75a8ee79f2b8cbc77326e11e33a69cf1d4c5ba4edf300fd134a67c6ea2bc09f7738db6afdf21a6b65787c2f57a7976a065f1258d2def8cfbe072fa3929dcab2d86fed9488924e5b623db53cb9a37bb60a1a0107d3d96b5835db25b492cda511396add354efabed04e1429df5c280c34030e4e873ce423914901660e1d9db5f9a001b4fcd33d3e70200179eee6410759dd6cf7c7b54edff22541bc1d583e50aabb579b181ec9ce4d7bf234547671d3811f9af5bf285948f031c9484233f6a2cec66e81e5731a51354c5cf96a013b26663037b5f397f08383127bdf01a011cfde0ebac85ee81142aafa5315ba5cfd2ec4f18d0d046f8c3be3b957656b4a632efa6dc08658d993754dcc69f14a6718106f8ff227fa09a5dd69613cd4595cfc1819486d61acfcb1acbc8f5f9fa6ec3db957a867f37982baae9bc57f76e08df81c34f7b0051d5886c2b39c4d88352c94eeac47437bb68da3d7c58161517aa32cef58368a284289347e97195da8febb4c107f842874926d09b0c88baed7646d08d3413810368385a53acc797f1b33858db6c2a677b5fe4e6365f1bb62b34776a1cb5afce3b61d36eea40fd94d90a671a9b144bee38b79102a4f50e367a3727c4a168be3e449f7c55d84253dcfd0185368c4dedbdb342d585bb1339edc29d7e195b576faf29ad52a92f9437df5432d31753f65b10d5f6ec848bc4902e467aa52b110b10a2ebf56f66b1a360015b4a903bfffc514868904ac2c0a36bfcac71b4f5d1c3133ae0b897c62d6a58d86a268232019887123ef9876adf7d1b4a094dcd609ccc311b6907b083ab60ac25942f8fbae8d7d39e2793da0dfa34403f61ce15a2b3094439106464dead857398b8b42483964208a273e4c759b209a03423dd4813897abb7ad2cd93c81637586d234ba702dce23bc25df32dbe0ddf6b5ecde87945317a04214634ed75726d43371b2bdcec4ba2777c3422cdde6487ee69a00dd8524136fb38f562faa7ec096f1e4d78dd0c98ffb0b9256136affb22c710d8c0aa303c32e4dba4c63e239c51f32864cee7ec6df338a68372fdf01fc6cb70f165eb58ab2a65e5d0d304eaeb106ae26fb8e17b8e349497fbf5eea1f577cf79a2a3251579b0b507ae2de62392d62098717da35648b14874b04212269880093ae1b518bba7a4e31a36317ef85896fbdf705f354dc065b378510483f9d0a3cc64a215a95ff253bc1a73df871bd7c5019995df903b99e697512146b0dd0a83e4a930656d3c833bbff303491ae31c9d274a2ca08ffcf591ab350fff400f5bc7bbe51cb6cf48ce17ba8b0fb55c0b026834a980112acf4b97fabdddb6e9134e3aad064ec5e310ab17b4406e3db889e6f62993e35ed0c0436435b3523ab7b64de0af84fad06a2544ed9e4cec694027c9db1942d97c0109262d19deba8beb19e69305465430fb2d25ac0dbf32c2ca0b3935eb2fb782c0837a9b4a9e0c227738755e4cc1b04f2fc7a007a711176ecc7c0da0c73adeae4755c5c6fb4f4d03b96ae33025ae8f98e99ea93dca5e966c1b257a90850c47fc842c42dd8e14c68e16bc7168ecc7c3b307c1e1ce5d78e807f04c013fcb89c9e5f0a1522e0eb8e93f62b7d000b60b66813ccf80969c7b9bc82317c30c240482d26e4210ef90239fd2397dceba3d14d234100c630f9bcd78d63c6d4aadef7ddbfb73ef79dce77b12c54fe1fdc9273947a4d04dbf683d6d95e08dd4749c3b3edc4751008bdc6407e2c531ac36d37143b41b2aebb510884189601fa9565b46944c3d49cbc0907223a2fdf3a1667bf4ebb1405d66e294bcb16a3aa1ab198a1b4abe9b1975447a42327e617c6b602ff2bc8b6fbeb8e955eebebba1756af480763d80f15b694840d7f63e5268c5a11f1f7c4f9f61e8f8f7182cbf0c0204a750081b1663935d771457d41ef79d370e0eb054fef065ef5fa24797a38d3695bd2f778abe4adcca5ff53696ae6eb4321dd0375e72b04dca84fdf9694f7a30c119c57e213c25cb487c7b7055f4ad83b96da988845f6cc44cae1d197f7d4cf5dc89498cf0be3c9db0dfab54cce8afb6a0f279666dc4152e803fcd4142ce39e33c66ce33c821a4ff17804807f60dce88dc65bb0986a323e48a05e9c7f69f75306b7d786a39969e67b708fe645c7d88d4c39e0f12957009773c757fbdebc8c93be8a9f81e047866d3f40f3357e95e4f44e75ae2ad2bc31bfbed1e8a91e9f403683333acc0e5d83218dde69be97b9a8eaae315b569fb62581964e83d5abccecf620ade7e431674f7e0ff23f018c6ed6bd4386cd3fc6b08d63db1104d2a6bde2e2fb993e377b971492bf8438dbdb620f54a5bacf64883d03bf53dfa6625ae4aad19c8c58500a600e1510e844e511e33bca0ed4e61a08c4fb704b94a7c165b86ea669c72855a68c9dd5f2b7c7e4a7edf1c6191c974cb36ad2a036ff70a16595c2cbd680227e65075799db43059c7ccf14155d9a454ebf4da34411f6dfc5f753a559bf733662c8a8cd41a87edb7ab493af050e9c10c123e99d60794958d004dd3bb4370b833c4cd6830032808da6c40541bcfffb08cc500ec51e4d029f5b767fb254f3c206bf7617f7e994160c246aacd446533c7213bfc43379ae8294ff74456cd41e1cf90cedf05441a36e824e7e79677a172cca74c4338911e9c2b0c8100f4fd5df004e47052501ae5d760cd58158c7a5e4ada3b41d8e113c002a9e6ac86ad227cbc78c4d37f8d99b351e4e656763750ec49e83a43f0ce663c8acfbc2f772347eb51d7d4d79291ff6785fc75fba75c04930abd0deed07dd2dc70c93ae5090ab2d5b7e880532170515f780ca4f51a32d71bb2795bcf14ab94f2e491fc17943a1d14cec544bbe147bf2c89079870255d941b52d5d3fc0d219a38b07a8c5be0caecf59a4ff444b8a7960cd84a2021440618cce93457a1de11fc5d993a008e7e54f2ad5768aeae663ba125f0b67f11e57029cca14a55074c5d89046609c7fd992979991516621e5af9c0ac448fa01ec58dcc643af405b6c3990dd7b95c6728cdd700fbc3ccd85c7fa43f93b7954b2189397dbce42a6555e6acb9df207018c1caa86800ae701e4e1ef455a50234c189a64a9e6f8bdf49ae52c8d099984cdcefeddb983c751ad3b920de658da82b57007d81b8ca0d93bcdc11f2155b0e88d9086127e7b62aa018aea3f11fdd1a10bfa66b8653df133522c3a446262508f27f67220d5631bce1414c17f6b5d4d6f8f41af924a3fe18e18ba6c1b9f2935418dd4800e8d8153123476accf4300d00c9aa4044500d1cde9747ea1515cf19f0bbbbc0aa4b1d2b2adc2ae867e9570dac408f89c7f99e4681ce5de027b616981a146d1556fb8e22105f9dfd5f36fe40a4da8b71d0cf40219c793ce18c6743679f9f509afebcb993c4aa49c7d01a5183a2936df09a4b3e07b7f9ea174da3e39aaf96fdd86e90fad13592ae6de0ece67573cb75f5e53755265c86eee8400c275625e8bbc99a782473fdc0494c8b07a3c9c5d85d130569a8c026edc39df51bca48fe270367088f18f0429a71f87b81661a5940331f351cf653e754cd90990eea3ad16d66a0e99c5d0c7e3499fca661011f8ecf8dd330837da414bad54b3b928d2050eddb8f85127659db93ab67543cfe1edbab3100b57c20f4ec338957e0b6d6b2f658454d0c9e25f1dc37185de18fff6279231dce5061f823e4e7bbfdc4de85b4d781385eb03a7ce3170e54e753873bf521445eacd8f18a625bd3b198b2afaccd90addf61c1f112858cf9f604fae620e2b2a94bddcd18eb3eaa8735b957025f92fd71eb74115b56a1010367848e00ff48a6e970e8c26f098ca5290693276473b7f013a0334bc46512b7ab6f4ee3230edac9f39ca4a87e3692f62bc99d14304a7dd6470b62ca33b07a9c5fb6ceb49f095992c0b43f531ee5df5c651d54b32f6d2d217e9eef8a15b03fd509519d728bb326229d2d10b2a3ed17e45e43fa6f2a4a1d5c9150b30fc3169ff8421c3cda92c3a9b3c05f6739bc99051dfa15903879edc394708aa473f018194595f90e31ce776c501c4dc75298c4663bf3557fe026bb61b6adc4e236a2bdb60b22c6d46b374d1a00da9d2dc5adc344a6058c066893e79dc7b6da1b9b45712bd8f0ebde3a0440e65185e8ab4dedcb6d0ddf33f66dc164cb2dc1865ca8060dba9cefa78a8731216a4a69bfa7d19e3ffa8c898f93559aeba7b33e583535f2fb0e433032c23f989fc5a77f0892bee73cd332e2bfa3116dd8bb1535c6808ee99a24aa4ad68a2a2566405e6dcca1b22b229a3be81c93a90a9ab2c8e10e01e2f50118790522175877b51e0f8aa35b2ea80fe7ab832f563147a0b115d2b76a739d716abd0ef065eb1f9e058bebb7b3763254604fb339c9f5703f3231695885886d0bff616250e87c1f2c90dd710f1e770b4d8145ec7ceca4151673af23c9e9ad2aa36c7368a322944ad9600040cd597fca94804d8fdf979794a7eccf55d6eb1e62d90393a69701c4e9855e148cd439e6d1132d3c1156d71eaf35c7c81cf2841cdf95fdfddc3721410bfa3ce4ddf2af92f6a721a18dd1559fe051435769bf8d75bd39e8c9a05f0ba4f5f99caee416acd9f2ef33b014ac7ecd213e4c9a6d60deee3263e451a6e4e0ec42b6ee2f0448b84f69754e5ef14abfd8ae5903d2994d80bc963e0f84239c798177768a814a2b569ac96169d85e74495cd5ff5f8cf00da5399a851929dd5324963283bc1c1b78682ea3816a906450059ce4a461118d02761e5a2b6775991ac532ca2a2a6ee4bf78f217c7f6f5994f2ee3100c723f20dbed7fe12659494fcf6a9e3881404bf642f270c42923205855b42e229987fbfbd7a01cf2dabdb37c21ebd222acad1ac24c2328c6f5150f73a9ea47850fb25eb473eb21d6eb259862591df0d39d1bca7bfa29d0d401911f75d9a7a479f8a68ac2a4f9d8b28216791fd816de5a3c258d06498b107d915fbc713b74e12d8b465f3ed5c6ef17631f4c586d7122ba02895a9717c8c301a0c81263b827b366e1120b55c88cb34ccd9660e0aa65e39e564d4e69dc04dc5928e2829ccbd71223817ce6caeb3a84aa061a5298ad516d51a7fec4393fa40c96be93a2515c741d9af4d43b48f6d195c013d28bd7232e70a1d8efe900d730bffdad3f67a3cb62cec4d757b9267285bf91c8330a9596367873df6606802fedecad63809240ddfc663c369326af48992b665d6d58b661203e60c884c358ca8fd30e756365a48f46ba39d93c58f9fd4b706dbdcf193a96c52b23ce9ff54a0aa881fe1f31bc3a3a6ba5224e94526ec1817aa5e5b533627d8726e2a93a47494049bacc8e4750e91feaac1dfaa24d4731eace96769fd0190540f2d4837304d58eedf2afac845d3507c684886a15268df66a5cd63fbc1724ede49fc1b73c049a03aa987eeb830773daf2cf723d5f02f0804830fb50d4b722d18ef4ced70220c4c4bd0c9da5ae98ebedae64fcbd9b29d32fdea382f6774d422cea4668db8245caa9bf81be7e772ebd431b4980c7a4a12e3f7a1ff89971e49624a68231d450eeb17eb9ac2f5e048cf4e79aa164505b6b35d050084d894a40f2dcf305dc682017d25d35059982b3d7a125b7d4b4e38bab4f17ae1002664fd2cb386380155e8ac2afc25d9ef5754c2cab95e29414fb223a0600a752035e143d1f9ede53f5b3e43b6359451e5bde292b5df0da521e5928eeb61bfe3cbc84ef51569755b1b15c006ea0ef05678c41422fb9bf6e225142891ef31f336877d52e84af792a5beefaee11d4f15f2f93166d91df232eb1a6973e5e5c74e8e626d784c0f6c45444251f21b075814670d2f60f4e1500db81b1555d5709096da9232f84c413b0b3d564ff2b568a0f400dfea239e209923f035041303bc1445e2f9160df7ba9de203bec618b3f3aa6ba6dc8c10da80ab9a452f2fb223f42a7f12e15115d63efe2ef863564f8c83d1c1178f8a4325cfb945dc1e1baf4cd965e1fda66b3bf2f31969567ab74d89803e3f56f5a39c6ab83ac47eb8d3d9f050e61ac1f65b53fe1610a4658d72b35816e4bc73117d63ffdb704c2377aecf21b14a9d0617f5233c443eb8355bf1a199e56eeef4a77711f2bbf7ddb194c55b21971e1555c80a712058e2d598aa5624c0bc44b22d2b42e2d7313ad5141d969f4d79650e0b32cf2dcfca808af54c3868da57afd70a5754ad3ba35269938cb0438d2d29d636c59d6c145ce181f03146769ac52fdd32f247af81218b98da18f3c8fbaeec51e1edf0184180d79f3f09c98ce4bf3a46f3573768e09300ad491d33223975ecc1278ace7bcbfd60812f661386391967b0ab55681b42a44a039138a29727c590f39f8f3c45764adcac648655b2c63a3560bb00cf216b8218cada89994728bd2e93dbdf20f91d25d9b31dee7222d8d0e6ba4109635de1a8ba193f740ef205ba324b9044a95f24c203eaa215070b62123c1abe19f853bc5342ff6c284b599f2abe55821fdc4ffbb248d6aea189a37bbda63449bfd7af886f014ce6fb56b135d7cafbde8614a7279b4cc2fdf23804cd3b8d8dc1077714b5dd24e8942cb33adae825bbe5b2d173263b999ebc4bb997bcdb075e188978e1ab086e4167f97c2dfe811348c50483a8b2f46eeaaf1da41600e506597cfc83831ee3e9f09e3257e98b038b8d6215ed5a29984c3db9dffdb879fe09b1b572777dc664b619029ecf16e1fe5450f83cdf7f3dea62572e74352f58c2b6c812829e6954afbaa60dfed72daf4a63f14cb67e5977706c12d7ef83e5356006ce4f2e62d974ec1153db90eb402f1ac8d9af94b07d969d26a4d51cb4636bcf9747f26c996895bc79050c22ff7dda76776849091ecc92536e8ed0f6a6529742d7d5255224fb7d156d13cda48849a3b27853bd21489ea1388cdc7a5a5fe75072a05150b744dbb4a1834c9901aede84dbee6c851ed87075fdfe84ef36fa38aa5ed7fefc1a8977f4c733e451b814545ae77e7ca39357c8b85d76192d6c660a35f553b2e61ab543737164c4a6654e8e3a0f492ad0d7ce28ef9d3d875cdb8509b0fea7044f39d2dc900dbaf009c2287c1d78477d80ed1e2b225ce4f0dbef918625a085c5073711520cf8524de0755211c74673bccb1982217faa02dd5955531d49907da2d8a5f27a0c2a9b4daae53ae3ffb0b883be89947a3460fa419cf79db0227b41a4dde179aab6594c9a280b8b435de6ce2188c50f511ce537f2d29fa0f58826d14e39a9502bc14cf373250adc6123c2c9dc9279bf7c2945d7267fa356f5576b25b1bebe5fa2b8ab791b316f01c626bcaf2b178fe40d0c55cdf94fb0835ec834d574805f03963d820a3be3924cec708cb6b3bb628cc7c050959e8b9bf4149e23f5d205e1096e572386d160c4bf924081ba1f791bd648bc4e6a7e89ab1469277683b9769622f06a98d35eb354f158d2f1428aa961ae964f3d980b0f9177057d88ca9c4518fd54d2f8045d092301de409a3a0ee0deb3ccd9f065c8b90524c7d94fb969f2672a9cbd83a0c15329fd5497cac9e4eb90746de8ea2d3aab55afdd73fb133e8f846d8e634b462bf3278f7bfa139a2ecc2ce6056aed49565c0d9f17e7cfee95af812f8f2da2b3351aab44b09a08f0b1b7b25d58b70cc9562e9b1a347fdadd2f5f9143fdc098c927e12f7d5e564d1d0ed14072549634093ba8fda334399b255949e57045e01f255d1f162380602adb3b0b2a4ea0ba3e84fcdcd3584024b331a6898f35acc06847fcc16f5fc7e29e08649b01e0c92e5e939629e4fd1f541e6735db4ee3805cac6135f19758aee5702a4647a68f6d12e15eeb8c1532728faacc7bcb02e3003e17034c40b969c00d0460baa9be886bfe879ad16686a88750c65767e61fd4e85b43b6466fb04754a4c14c496d8077e160b1236b624ad7aa65f580c1399391c78924d32dade3b9b02854f1a9aa767080536faae4a77178d12d1e8c92c2ea6a000014dca34539ed3f9cb76755d59a66b6b7a4bdf2bf475b7000a8578838fc3cfb655b5e90e97374348ddb9754258f4730f16394070cf2fad8b3597d274b78e403f894349e46abe6ebd0ed626e825d7a4d818c8e7f7bec010558601a4f8a307b00743d97b5b6013cc5a0a302bfe5e561299046f106077508a3cadd60067b4f729a9cdf5b8f40dffdc17212895dea89f13084c8d67592f68517be85b51e171b1b3b69857e9686cdbf214fcd0c3063631064b3b2bd88e3f530a2b63c12408702542bcde88029e6b3b5cd6bc595245a38595db0752f6a7248f90a39b435ffa5cf3485b29ceb5320b66775486a154e9cfe70fc20949311cc100a7c4718172eba6bdb8b0c4dcbae25b6a9132d2a0c1e599e5bbd99e3e6dae81e79111f89d64032256382d2df0f2a9751b6bd4b3cd067372dd341e81ae320b75a839cfab88dd8e0795dbd04ae9ab383b1be12ddf09ae5f9917f5ee45688cfb2bab8e290ec84a0cfdf40780568ecfccaedc5a21322f781e2e69d132f1bcc278b1c4dac9cf2313afc386989fb5fc62dd2bb56a7b64f8f62eb16fbd480b79c286afc305f6e88df4008aaa33843edd35a63b68eb03a1b2dc128e63c36d24f1b2c1dd7eb1f27238e5ed160ab7c83326fba7c04cc68ffc9aa9d268d96984f7e447fadc5e00b94618cb25ca247f88ca3628e8e7858592e7a0110ca3132734ddd21123e8562ce61e6305bbc1506662461361302d131d8caa84c71fcbf99912547f570fc5d05936c64c83a970e4c2f5570556812205da9db80c2babe48bb4703007489000033a61edcbe80cef2f7e7775885ac7409dbfd4e4bfba01583aad37a2c8b70efd5e65ac2f75f1acd7c35e17fadc6daf69785c5cd9b91ed4af8990c544d344ae551ec66fb24c5e0a277eb4e28a14e27107aa8a9004ef8cb5adf39550d0a2cd5336d977d3ce4ed66b259930774010b92cba01eb8654e75a1c4b8e73d0a81772b533113ab6ff9ac610cd77ffc71d0a49cf1fe3106db0f60f4a6db4443a6a211ecdce0bfe08e55dcfeaa9b88382042a8f05726a02ff67f8dfa1433bedf111613e4f0dc8bf7f1c07285af4e41587fc7681b50b40dd8276d54a58d5696345c02301fc784d497106064175d820515f7cdfd6406a1dbc33acaf5fa22fc85e6b8baa9c678fb3a78f00569751191bca56906f19d32664fd1bd43361f3888468b5ffbbea7e331d98b83f2cc7fcc5a7c92f0b0b42b7b12574b5fd87be75a13790a94e9e929cc7b31ef53df4199c2aac9b30fcda895091ea762de6b085ba6c9478f6035e405a382be84fddc6e8e10c4af2a4fba929887ef75307e309bf48fa66e049d1f134d8ebf0270593b5f0d1695aae24c66b24443a3c62990904aff12a65028b1eb85b3b2cc9efdc1a4b0bb3aa8ab122e0800adbc54d30e19f92ff2f23fcdd5740892eb8b975e71285777ed043374a86f40ec3ddf7ca6c49bbc0b26f93dc11c2185f441b46f8b047cc3b84404c2c38049798866acc90c047524b1db4d053e75d18f83cd7872c97296ea67123c0d2ed3fae8991122e1784b0359abe35fdc20d7503118db6d25a4f88145f7e68fbfb137e9bc526555795054882daa314bfebd8c2272c4f57e5918d63323f6a063931685efcd626028599b995f75f3743e41e0983cbc81550210b34753087821e5ebeb2a10687e8c7b41bd4e0b94236503a78281bdff3eaf138fb74be8a1e55f73c3e2d35e598bb4de7d81fdb9f5ae59f612f774db152e2609c16a65b99d97bef45b40f5b246a4a16c6cc2655c21e6f8204bd329c80cf8780091b9424494aef6dd69267588866069007ea43c3ae6b08b4d1135240884d4fff64a4ed32918ef67fa52e24d68a6ad84aa0cc3242dfd2f6d5cf457cc45fc2cca5c6aab0c9cea7efed4f4dbf100822ba6203b215eac84afec6969571fe22497b9f75c24565c8f6bbd641ff19b9ff6cf922e81e82d9cb6014d8a99ebcaa998dd473276661cddc45363d96c6e032261f1e7b4cdc5b65a6c80991302248288ab0b13e7795c5a01342dd7f35d639ff9aaf7e43d122d692817533e3048c9b550ba2b105a223da2e0c20809d13a0429a9acb9a087d570c3012641f59a2d823de43d91de201266539bf72135ad72bf0f88d5b6b27999ba25d28199889b2b6600c8994056e8dee603eb18e6b55a1d88acad50b7f6149590a4f308052d838296d538be9143a7dd71f27f4ed588297aee5724c34d27b15edea6d543c48963b688b2c09f4297b1912bb0616598bcd24b07a02ecc3c369e818456485ca6e4322909b33d40b75782d151188687ff8439fa925f8a96c1cc986d384cc170517c6d120497af19da399ad002a811ef4b3c9bdb710d8a43393069a1b5a269cf8fc6570100130c18e5af751ab7ce560851e5235625d1bab525ea2f380f37e7a75c7e10043861c1cbefad33777d98c3dffcf7640b345d8145860f90302a90e9f03a0141b616261bbb44f2c6db99a041e42ccb7f341d5a29ed4ee5d8e6dfec77923e5e6ccd27181246db10ac62a42c51ecff5eaabbd4bd03389ca1e23ff10e755ea658b02d3a7d862d1d4fda846a0cffc020ca2244d4dd0e4d0abb07179b837b44da2c0bb4a8edb34bf3c30e0c74c054c67dbfd42aa60fbea772cfabceceaa1db4cbe22a2e7f7664763d6d8e79520df905599803e2839feee0a1bd8d64bc434ac334deb8c84e3c0c8e5aa000e90f0158c8e09918a9ea247748627cda99724cf9898d46b0221294add3677cce1c29827833fa6dc37bd40b56a3efcfcdb0f930131904ffe746188de4bed7cc8f177f5b1039fe6b321d311f168eeacda36db96386bef04888453e2b1cf92a3abe405283565a5243efb77cc76fbae329830de4e009b2d6213c58eb201c7eaa07f43dd0d825cb9a697d7bc7d065682b82a36c218c7819a1b652aa56429a01f77f677d992d856aacf736b74a486ab4a325204671903ba15fa59910b92b7d379570d5e60e2dc094ffe1cefccb1eb44246414e01a006ac16d84ec2ff1c1f94e86565cdf5c178fd64613e00311fb508382703314bfccdefb95c7b1ad068a243ed11ab10499e03ec5ce7d883d3b7537c2d101c7809f55468255241320c3ef64f8590557e945497ded2cc76ee453e5329797e1a71ee843406837b18ba60ca7beef7536f463e4eaad9dc7dd0befc8db0f99e133173f0b7f39c77ab54fdd033bbb186ad56ccf124172c65599004efbce351c3480e7e9eee25516d62006a038d27c8df365eab969575e3651850dddb1763dce5a82a4d2ca99efa87798759d141e3b8821e67cffdf47cb3fe3634303d49103fd678636f34f9009ea326b1df598223e5893b3ec071343336ee5c8c50a2c0835129c215c8f8f4bd1fb563456cee8ccc1d3cdb0d7a95b150a7ebd2fb703bdf1f17cca27bd677089dce81dfa10f98a6a5d24fefe2628607897b0a8a066ff453c30ff8ba4a5768bde81342667cdb4837aeba409d03d608c4f1b20b0f94d135a887bbda39dfa25963fbb76db043c2856d9e7d16ef2fc17e9089e68ede7bdced2826b16445375ca06df75126bc4f46aed6327be1a39a144c135f58781f83cf7b092327fa664c0c30be6e845f86f5b49858be1c7c002f44778f9189aae8b4dd2a5150896861acf7c6dbea02b980b6c5e70e5382481c359039a3e7ba78d0eed8e51b752edefa9f6932dad238fc309fa35767dc17a22974779f19b4c5f1656d779f2053ecb3cce400953d7aeb8db58f4d373cc95cc1b14ac5244b92e6198dea0f2b75a9c3e7721a822ebc4189f0a1871aab4fb09a5cc3208aa478e77a3bd4ccfcef234a78c0808848d252eea520b09e77acbf5667d8727d1ab8901f646c4483a11dc503b15c490119b420027f0359ad53d6e1365f5f63884ed2745fa1942db0a4aca10d0b3f7521128529dda0e6ceec1748bc854777da022da19b6fe357e85379ab2311d0f4819ca98ca8c64f9d58a57d0abbf8623e230c38f3f157c477fdd53214212241cef4d5f431512dedd68fe7ed7168049354cbed045c91d4a5c6e510cea81c2d03e1f4c0da1b191dfa9098397abca24022601aade44858458e3f34acf38220d3ba00792d076e00321601f71850d7ad755505a30a45858c94631bdab26cb675f8fc68e2411a37ac87fa6fa9edb2b02f7d216a88b11ac968f16ee5021186fbc50ff4db0c54ad685202b3bccdb34037fb120d56df0a34b5d3d59162cc755f00765d994a8a72e2ec983da363a1008594d27ef5819de645aa6109a2d6cd187ecf2a5ba42da9faf542b3826ddadf1c8898f9b470de0b9fbe53c3517d9a28264ca2ffc0aeba09bbfdf608f07cc12512544541948fb0598d821f3058d92e60a58d77820e7b4dafe14c11d79633b689aaff7dea0107db6f0782d7ad555bb00870b3df829634765878b1b8cefbe1f93688916373ed792a8829f9867ac4dfcf78376be447c7a0ad7fc873f96e83acefc150456d86390570ed29bcc8ff40bc8b93a94fad7ab4874c97187f515055225ec6857355a4c718c3a142c2944a02f8198bbbdfdecffb2ecc6c8718ee2f272a12d0307a4228d405d8fc7ceb4011d4e88dc0d69926d44fc5568d7754422c4e7558bedb3a189addd886e8c4842170a6c840cac99c732fa9d7fd967935b887a76cc1cd5365e0239045b03e5916786f9dbcec362d43d69319b95153e650fba753c1e33a27c820e7472ce000626f648ededdf30cf8ef8ac139fc45946c2bb31fe3e1b22d1ba8e5be5c7f2060ad3a349f1aabea233ee7384802a789a267680bd28ed7cde354def03cf22fc4e5903ec86c19fa17f9ba217f7582f50e349519cad198e06296b8ec58a9979a511fad25da17f811b5e2865d43584973b3a45264e7e15ff2e29fecc45bc603ac163a703723762b60966fc9220558cfd1864f68a32bc1eae6e029fc3b1243413498b5f728affa4f52f37e1d590181bca8e6e761cdda5689aec55b37fe2f4cd8a2fe0cbf198491f30e22fa73f69b1293a7282aa81f290c0fca3b4770ee292c8be93ea321a69aac659c9221aee7e007d2dd552050ab328e31ee3c4adf8664a9ca73bf06d8143656ceae1fb58147d6552d78c1d462e62563f18c194b65e1323c939fc902bf46c7459cc5ad0d9db470ba2dfde62b6e1d5d6d1b7d6c07789a513a9e188ee1a26e2e2fff5a9f7eb4c7d567916777995a51acb0f00080c95ef14968fdf4407d1822b64a3d9473f3437087722c37b96f156f4486d16ceff4cae3822958e34b88902697e410e974caa052cdba626178391842d6ba1f788e9d800f4e26b1347d5a2ec38d3e88a1fc888a5cca2a260d4d93b02207dd59a39069f829be6cabe8028f2e5e5a64f774a251ee12d037030285e7a72c97f7e7cdbca7e397f4748e430fcfe900e03d9b04fc988b3deeb64c372efb612158681e55290c6763cc109284a557a7ed12f32d69c329c3b2f8487160ea19b0438c642ec6dcedb9dd887d8babd31de99a1eef1a767a0531e2f69fb5abd0f8f57e7ff0e4e1c6d46c5d1e8d52d0fa261a26d799f9ced7aaae16aa96b828effa6c77ff6e5fb28ca07da15de30821d9818bc52fa36eee588ea8227edd5a8c1a7dde975a06d3366c9c74c169e8ce095dd0b25596b40846cc43db5f8a8838e3be91ff5f7f5a9efb6ebbd92e353e27b05df0f1e4e774d8ced428b2e6a9b87ce668df696287ef6a6e117d35e5143e47394c23b3c080e41dd94b6cbd85c15e11e667002337c1d0d877713b8dcd2f4d10852f66655f0aa9449062369de70e9d7ec39214631069b52a76f116551d01d53c4ea887ad2d0d64fbc8ab047126f1c083bd5075e683adde8bddb0966b0cba3e6c65e80116b7c8339eb5b83730019adadd6198abac40f619fd7bcb6d698a1bc34bddd4d7826ff71e058b20423266e18eb6f45f77292fdce7c354492b164e66ad979947b2ad0d8d504e1b8644c2856292347b75a448376b72f550ff11eec1a114de7aff82dd3be091fd46978091a2470783335116d9a2fa28c3503494616a898bb45e3578ae159fb962a70287cc626f1d06eed9c5d32e2f6b454bc993779be3575a21bd20139f7fb68ceb8a37b64f9b3bbe970c783b0b856fdcfa8e786d954050ff73c7bb1399513d5bf536197cc5d114f8c5706506045892eb89937260c104b1a3653beab5d2769a25ed03a7bda1f5008baa8e3bd498313640ffe925331d8c915bf6ebe98831c2fe0d8b5acf9a806cadee13bab052d188b28021b59c26f9c69e7abc1f0d8527b28ff19ce45718972c6d1994e81d360414162014458cfdac2d7477fff59681cf8b5a17b4354ac006cd4b756363571cb25f0f61d136febd5d6068d0efefd6037fd1eb2a0ae363fe59e931c09367b1b44e47ff4bf9aa843b84d6fd75036bc32c18fe21f06a50c7ddcb69fa1387447f1437b0941e7a727da99bc123865c1b3aefc092bce87a7311fc5a37c7187f3cadbf40e87945d9d3ec4bd7aeb0bf23f46b63838bc45c2464fd828d71054e07e8e16deb820a23768451757c9c3ce02bd26e970563e36fd3fc6aa0bf118f47a3879f52edafe77d05ef10669f97ecd74debecfd8cb957d45291b06a86cf20dacfa84283e8846c42338f7a257ab4d9e2ddd1fe07a1c419e432f229b68ff21caf5ab64a5bcd4e1e74a36b5165d20e900c19d67d53fd1116364ec584272f137137dbf1df4146f0fac7f9454211160f6af63f44b857467afbabeb0c41b03a75e85919d26373711ab879316f94048bde1696fcb99ae7c44a5d69dc614ea1b117dc40c6dd117422c049fe19dce7d8bbb20b24783da25fc1f5bd8f86983bb23ab752214362b343ca0b3e39faa4a2831cde78d7811201b66e92a7263ce04379c6c5ed988f61f527d7e994acfc3970faa5dd7e7bebc2131ea2815a49e2dc5a1f6dc570e6d74c8ba72bfb52ed1e8fa442d2f2723c6104c373be12b10aba5555832edd8b8ba058ac6c44d52e540f0cbc5033000a5ecb25f438d31548471103ef062b06d67186ffee0a8dc71d3c6a0ea9ff4c2906164d0e0701726a00fc3bd6a958898437a0428fdf340a47b1d16ad99b4b69d6295ce99a7bc967f607dbb97b92ee6758a92a55f1e70ac41995ea6b27abad3348da71104c8dd54c87908eab83ce143f0ced0036c479b51b027beafc0961f454ce853c4189605322e28367d974c138534c3c89b50a9a17bede0b500bf60729c239a1409d58641ac83934832d310c741f7524cdffc9ac8cee5aa45bf653464c7f9f5fd050dfee7f874ffb63036e70df5bca034a0f895516bae3f41b4a4bb16caa10653aeb6e4eb9fa1765d06a45160a3a55bb9771d794f2a0725626e875a09c0a940d6a2f8a772a5076054aad29383be6c00fa99eebe23f9ac0c3419371d0689b889ce9a9c1ca7ecc28f3ebad5c6620a52f1cdd1579b9b7e2520565e3bc10c973649d13aaf1d00a03c4fbbcea91677d33b509e4cffe34ed5bb4ce2b3fc8b57b93574b415c9e6e78d0f89901fda2eb70c2e8699d5dfa5ce64c56ffbd4d56998284b1da9d49148fa13fae97eb96fd05684284ec6357402bc68365432cb9c77ef0ac4c9bf5d37e076647aaaa445328414721368471a001b88cfdb8ac907fdebc97108361a8419cbf9ed615cdb7053b6701509e39689b354b138c193d26a5fbd876203e58df0834e360ae1d832b1f6b873d86b6f9d78741180a0841ca0b0abf6a1f3b935c9834c6ad76852b3166b2b037e7a2b18ab8fae2595b47c526895ed154725bea2aa2c89f4f2d896627fe6a1193f8624b46e097f15c46d431145d39042b43630f9b21b53a59f8a211553b69803b5ffa525e488532c837b48dcbf71bb5b0d0a8c2e237270e30af673ff41233102d5f53edd6ddd1e50ba4c46642e0e77d58b2e3531dc7b29a330ce90d7b0a705212b8e9816ce89e1563ffe7c9edfc84f511054fc365de082c941614bf357b4c6c365bc33bb9527774e5868f1641aef287b26145711972d1c519443fd5fa55fb007a5f751e9fb06c2b75e4df6cf31ddaa34dac0775add84822cfdef4e89ff497bd8759231c1e9deec4b56f6641bef36ea4d5de4d40889c4a96166db4d5ef8b604aca98d81aa4c777362241fa5d3311cf874e5929f6ffaa73d4d831a5427b9a8c00df2e96bc2d3b99c7a43e12a6a4863df0764c827340386996672452e3d3c699fa008d9974e3b0aeb67db57e37ae936b0b61c461e0bc6ee28c252bddf099d4e2fcd335c8c1bded99f182d711d477ec8e3740f3dc406cd3d1386c822d32f0ca3c525512f1d94e5c76517e2ebc9a86d625db14ff9c085fa58baf495210313206e1439912455b4e93cfa3b5b693f113121a1ea2b468576f5632798f28171761d438066523726a7a3a80e271be8246d5cf2b2bd7cf3c7ee924aae74b4282c7d3f94b5e0c77ac88ec4d8dbd26fe24c157d02832ed09efc400539d9183a489c37fc3ad88c0ec237267d078f1f5b5b239aaf0f0a083e46794c40946b4f9cdbd2e3e1d4fd63ae692f10901f6103fabb7d02b4fd140fd264bc0cd704451f3e823afcba8cbd44f9c4437b22063180a82644590d786a629d7dd466b34083388fcbbadefee29a41a027065c8d563b360edc0ba428c5824b9f195f06f8dc9fc7137c23bd40dc425ea6fcc7a5bcacb91d421dc647128756729a9087f614f00930befa4b4290b741a185b0123aa676d92438b143b0e274d6cf5efc8a96a6017755012e6d33c452b14c65e08a44933d7bebe4317cb3ba7a0d0eb6e145b22f8bcdae09ee71c840220adbb013329dd248fb301a6f618db9b0afe574ce87bf7431e3dbbc809e78cd84c8201c5a889e9d3a3b7fcd7a5195f75627ac4040da31417db139414487c66e98120fa7f572fff7be65fdc35b82d7dd647b7188de32df7a932f894a7d2b61be480fdb3fd33c9621d506349484c48ac2463a3087a5a3473af9a0ef304ad9ff5b5d8da1aa18faeb86924c5a463efd8a5e3c712a4f1ac26e7c2fdfdd2c82f41e3e8eca5c45f24a3c442cbfa12e29362c4edadd173dbfad5b197688225904a44a6516670058fb60dbdf6afa0330fab3f5f2003ddf96e6fa4c9ab313ca40b33a3f2c725fcdc02bcc3a71f45790b0ead3e657df8fccbdd52123d869160754a920853ec1a926896147ec7c442500f1351fac11b816022d94b4871820a5498f21c3c9e244ec0069e9e9ed02146b3a1f3013c3c31abb99665e5a5a9f4359f1fcb1ae2f9b3fd281f38a2aec6c37b30645ef8286c17dc0f0803f882c2e8e5f362bfd13f379344df81a3eba97b9438c3eef438fc5e107794879ea40c2bc19a1c1916542aec46d7de7c785e92f48e7f08c34e653396f4355f4cf38ce572e7b614a2069e3c6b91b4a2ac6a2e8fbb7bf4685a588e9181b214be00df88a78c4025584cfb398daf19ac9b1b8ba23df794a1806918e62cd5654e3152b7a275508e02f6f6e7954281c92011b2d92f46f09c0b53a9bff0b67f45b428f5b6d05793b1a449f15ff5a989029a4aa68f673d583affa6bc5d06116a02592dbe4e6faf2ac248d4a213f91c4627131c8f28301325a9e72469f09b7cf69fa4e9e08ff1b2a91d54f961256b8a36ec5399f6f925c11573f7356ec04967708faa6c58528f737d689df269e99f89379bd5c55f9592eda25c365e938cd5ebab95df2b97fad5b83531e63d5e812d917e19d2c4a71175c46a0449a8eebe93e3aeb7a06e99e40eccf8d3dcf66b8d368df814397fe87a31f988177312a1a84286880cf745b9d077cf15aeaa1dfdfb560b4d3397485896de56d083056c9b449668dbfa472ab1d0d4dfae81d703dfd21f95552de565129fe81696a81868757d91d0e36dd200ec88f39c33cd5f6f00654ac4ce5097c5ad5b4bfbee25029a6eca225f6813c7c3f3cacbdf5f739c61ba8b50e56a2b20c6541e9e31ff1f317205af88860d6ac1b7a3b4de70e13a74c04125881daa517a52060da9ea34035db8789f12eb29fb1d9f88bc82bb862629c28afd723bba78d73a3a9878f82b1d01f7c957f2aeb34b05a8227911d8c4933fc8f51bd3c15caebef0174c0f0382813ba449f96fdddbbe8d34d5a4be642546eb6a5c614215d3ea447402089dec8fe4f01107bcf721ff631781cfd982db413dfe22644e8ae60ea23455bd2c8a506598f48d97d2628ed781e0e7e391183bbc4d8b284535625e29e79d80cb0c19690527bfe28c2ea8e29040cffe5b137934eef104041da8851627182c342aa2bebde468ea05b9a6c5ec8f12bd8db4d424aba3f903576ab4bb0d5b122db66a85add720cbf9a6da6f2fdf03b0b13eb0cd477a2effd033d9e83c917c06dc592b7716a8b3c18fbef459a8b106c45a3429e266c748e40883563b6307d7d64d46343792f6e8f442bc1873ee22c66592d57595980a3320e7513024382152839dfffd927a3849c1c5deab1b22d9a61de93e002ba2eb77175b95e5aa994d6c27ad990eac8cf5e654c647eebaff6af0861cd0a1d9913262058ea10b30c661ff0d59b76df12822f07d69f9b7ee0207eb781b52290ef51e7c8d092e8c7498e3e50d5685798196ce549e250703f109f48cf7898844fd306f8ed2a39458434f27a9c677f20c1b9ae405f943c5eb08f9d57890121609ef909b7dbda364cd890f5901548d6eb7c92d1baf31632bd4f93026288c67242fbb397430a1908b8207e59e991642d8ceafba20fe59ee9f60132b47772b1d680703b1227e82831946e2e5213e50dced773ce1fe005b2e2e5f9396f59e4ee1321848458e92d739f2a7583aeb9d64832244bdda09a21d8101298eb8fcf5afc8b8b5d7853d07de6d4f78b0386727a47c25fba1203a7a18601fd3401230171ca6ea4d83709a13193187d1ea4a8262fdf0142cac88d9ebfb20cbd035b7ea5a89ea20594692cd184d45622a7cdfe2bdb7bc6b467d08e1ecf0c4210cfb9dfdae762feaaab7d1f71dcd58e0ff2acc25cc351a19463f48dce2aa56aa763fd17516d7b92da5c167c4994c69e8d8bc8dd026278ec6a87bc703d995d147a54867484b8e6da77aa97a856507a06738d54d497fd715bae83f1fcf7b0318f6c0100cdaf4fe779a47622e5e94707fdb4e1ece0da4ec0bb297ecbe32d4b07ff9e9a6ced7ebb2cf93232e504c71e8c50f4f5d8f131dd552ea48e668dd2818952b05cd55f59583a5780854f61701d2e5bf35e986eae5c2a5d7db822d0a81c66a46c6212d135a66ff41feaec1c8b5f1bef9b4507375e68101cfba076d6fa3363b46a54a78c597d0ca2b381ba3b2c9fdec05191ce036d802208b912b14c58d2ac48b5ba0283f641b467a955d05f32248ad3d21503f3644250928d5709da1919306cafc6a0426204390587a82d7fd2d8281073f8e0a278d79a3cbcd47148b8bf61f48b0fdc6fc4bc49098504aa61936d70cc90d900f9c54cbd299de2c1a6eff2691e244eb45c07879b5b721b25eade3ff3b9e49295d708198a6917674e68cdc6c4f2603ee5531222ad584545c351387ae46a0cb1b12021764e38c7439312cb7ebf14e9a1c7dbb79389d5ca1ec754fc002a0431827e0170e22b0b440ca016b861ed8a60d772f973514d70bd447363be38dd728b372bd4500a8bb7e83df7ab9572a3e7e15e88da958d0a36219500c8a1ad8e614ee17dc0dba1d7784d8341c85adb92e4e042aee872cfdde9ca8e32ccdf527fd0db040763b49f01548665d96c0b64896824aa794aa5b045ffb2149957b5116b5b2d81762186e31573d79090679f846adc47c83e32baaf34d838749433192d81e0e4aa8fcaed63dbfa922c03dade479f556904825b61be165ac32d47095b84f13c385ac2106786747872e865524d7f05a2427d4bcdf9991f2deda3472e1f67dfacc00db32d68a85178da23ac28b2dbbca9be7991447ab301e9c1104aeef02e0ff59e2d2c734ca2cf4a09fb32748d6b0b72925696fc20aae0ed793f08c400f8807d676236a7ab0968900784d06b967f31a73002d24d2024e98fd5fe1be8e3f75643883a0257e6d080da61adf0d2a69962bd7238359db152bdc84ef8572080318cc3fd14f5df6ee02b2a98cb70ee4e1751378389d724441e72b3076a8de744699956de92f76116e833bd6e4783673a4aaab206e6830c075df0ffbfe6d2d9fd49d8d8c886ae0d05982a7a928c8c09e416b2122b5bed31cccd0c81d1510ef697eaab6e625e202aaf97df244a5733aa966d7a9ac77f78239ed2464b5a07e0e16b4177bd3bff26a84b589442cc7c61eef3de454ba055a1d284ef6a0c4fb7ef320ff0050c1952cf77a68e62291cbc692a1b8a0daa31142b335d8083ea158e126fd9cc60ed46189afb9af6182d8064788dcf6c2ce990c42c6fe16ecd0fddf1bbd286ccb044cc2228925e39aa71b87127ab9f2f80a51e74a9be3baa62f932719822e5796288a3b704683eca34c789fdf29fbb95e9277c276764032aaf0f5d921504a0550b4fb2c50eaef628cf6f6e765ffa591d174a2ef8dfeb9a95831e1632c338a1a71c366e0510f6f07d6c1bae4524bdcb13e21896e3d4c7687aabe03f6c69fd6f4c1c2c67cadba62b03d45993bb39d103f0fac4804efb20cf65bfb9d48ddd4464f0ee714120056e116dfe1cf42ecabac0a442ddf5f2604fd0e59687b41e7f9559fb8d6268f14b5532fc890b9d3c40179a641feee45c59da34f5d3429f9a25de8a4c56c45798078ecc6a42d319c2a3c574c94d5fa1941b402804b5e99f2740ad57530aec60b82878e94bc67dada3398f922afe73f0e019b4e94c4324e6750f1833c457ebd9bbdb4c1217705dccb83f9548c96f976081e089fcfc93847e1824604f1e07f5cc8954921a6039d6e52f2e3533245f39627914bf363f0a9a04889954eb05791c8cb43809cf605cd4b71075da7e17ca254633785f6cf0607bd056be2e49ef396e23feda6f9938d252022c2947eeff0b01b743c2cc0e7d7eebbfe2540ace6add5cbbd04a500fa04f104f86104e609943c1a9760ef5296f70db4d40f7868d5c66d747f141e0d7ea913fa79fb0701ee0feb204a7b289f31a401e80c5e1b41821270441bf2a17b11d0e451fae57d8e27eb3a3cda34453c25d8016754ddacbb6d9f276ca03642def821743c72fea42c2fb8ad82a8cb093a68342656d85134a29c1fa4d4ca76e3f74e25ffd977ca19dd6c78662044650844b24d7cd68c86efa901d0a6b86f2ba0e3f46a1dae42f7c6fedb678d887f00b402ed897ae63ce784d7145f259e425509e060b5511c7c57689227b6287efc75ce6c2cd1cbd071d8cbfa39f980584058d8355bbd907431f5520a818b9a68abe91d2fca91229a6db265e1a03ce308ccd573a2150c4e83cba491da74092a13ef93429e92649ecf7d8ff73c987b0d356e0ffc42998c7c09c782ad7be5a45a7a16f1584cb2ca2c987f6385e9c925287e7165cdf305da2c06d4df0252db2379751ff7a77059329041a163fd16de0bdfb48a401c32a88b0ec8c24cad1d99152ac1c16da478a6e8c78dcfffa87aefc34a833e0d11996e0673dc50258dada1f6a23d375de98f5195be83d58c8233ea70136ebb3cfce69baa62b748f8b265a50a4e6bc97f9451747bc618ee3de090c86ef355d6048fbc8bfb9512f1eb1990ad5ecb5d88536ff38b49a9dd1cc9ed109785589013c1b8fc4a7808150e4006742d3bda0c6bc2e0b36d0447f4c551b61375bb0f0c393373e6316ab53d288b6af543ff7b71755a42f494486f8eca4780c8ff1f802a92d5f41393404746eecfe6e0ef4a1bf573912c09c8cb31607f6f438cbc4cc91b4183a6602a204b5c31a1288beecdddeaf8128b43076a3bfdd8c7257ad8db98f05810adf36026979925b59d5c8611c2e41a81f351f7ba3cbffe648383fce28822dc9808059b284989946b519826b705a8e1903bf5a612f881015e1e77aa4f69237fec7090287179f79e0692bad5059fcb8e3590b64b4d548ce9a62e19480d9651a69091f2a75c4ad137dcd84727052d6f1c82f9b2b7dc4204713fa64a699a42fdf6786ccc9ee6611a3de80b38a20f885334edff67c66f0cfed561af9c0f27a81b069a3718711b930c195e805d4a04eb69ca06ddbbd43ac01e248951f1a3cdbbef2a6c84a5e303306cacccd4b9e02747dda0c4c53740be44d31add03046cb8976bd177ee344ecd724234040c904ce125db0f9f9fae70fdaa60c18e1953cb8ee6e8dd69c6ae42a60e28fe8402d79cd59db5eb5c4dae903316b12bfe9649a1fa359f2d0e19419e4b834d5d787220ca969302086f01c9513fdf9fb1ae745820c6c953bbb8f85738b346ed458aaf9410ffe0ecd9a3cec7e7a310d7cea3613b07f802f3633f55a8d65ee5496090bb080cdc6c7259616c9feaf25df6062e7e38bacae2e55c468e22e174dcac7b302486417838b023fe553187ba16a07c0053d0e6a8254a0d4fab27072a0a68c88d478ef0bf812cee26e4adabbb043a3c50262485cb365a696a6d3cc903c59054b5b93cfe245426658103a490992df241eb005cecbc8fb38ce0ab94d9a7b1917b9f424a1728e4a0b355ab8d12aa981c446fd206c75c26c95abc9665dc4b94566405f36c171adbc73bf6b55859e7f834a516d2ba6860295233ffcabe2ba3ba3139854a56008109beb2876ecacbed1ec92cb079aa43b0b6041f84cba2069f6de82e6a32920cbb3f27fd8f6558e6cf8dbba2327fbed2ecc0da24803134063e599645245c22c63463324a086a948009236c0fdeb808100462d433778f839dfdbbe7b21f7992011f13bdbac4334d55d3a16def7d279a0d11e7cb030014b86e82750cadec274986a07e9f6ccf79a62025abc2534d118085422996fde6e340d4671660bc29e94d372decb3505bfb6389278ff131f7efbcac0a08cd21fcf1f3c92048cee5b248c90a2781164012755b50ce44b87f68b1b44e826e81d5a57811f56cbe42dd30eef313b622820fdd92bd7f0ef0e9acba78e0d598aa7f7db518764d32190b1d01973a57a972f2d17af766b2514b75d2fa1b18b227343555eb21a008f34d64c9c1cb0ecf16b470db4e431a6d8cf745f9a2fd01af8633125bc1bbb1395b6cdd8bc98b66820af9c9e0ec0c557f4e38d13c058dc5e6efc92e5f10b63032f38136029d4485d38bd85e43f2d4360b882200014867f68e0b64d23a33dbc0a6ce2244ce293fa70c8085a4ad4a18df6e37fc14527f0618df0e3be7fe14b22332afa62b9fbf6845be90f826639a2714935c6c8e4e2c8024e818bc5ea2c4d71166eb5a640bd988487a74f89ded020280235bcbcf17f2fac157c0dd30f699d34f534449f3543104ededec9a22e625a8abe21a66b0bae3a53e41fcdb18185f16103bea31ab811e24bb9098191ea8d78b37d20e27a6c674b6b801b338c5c239d46875d44e12b5a0738f6222bbdec52229f41571a96fe841559afdded4cb80bba368ea554807b1969d8ec88cd902b60e36e2492c695c9b251f21e1ca520efdd783554a7cf155dc9afd7baf761587128e7835b9688aedd45b9011ede4fcfa351f0f7790c46564c17874c947d0beb63e79278cd3ef723d3f2060a52d5ca3089e7666be0e7eade8e04f08922bcecc456ed66126c5bf3ccf32bee486e47dce12217e3fa7b1152b101ae05a64ab831a1ed7f1b1be26aa6b3ae22643e85e94b47cee2c44b4759ff799ed580c8e9d0b820b2cdd0e8822a7825ef68076a9a1e9fea08b7b1a9bf14cfc4dc0f0fc5c76dc9f09c3aee7134f93f09b4c4db7edd522ac92c2094f6ead53198e18ee8ba5ab82bcf96bcc8bbdd678b607e86f5e4ab27e3a9ef898b3c9226897d8ff19a31ed87c2e0855ed02159831928752b11f82dfa6ac1dbd1759ed23b713f01f356d01d5e115965c3e9c83919d64ecd8d782e066aba8f2d73db0a01c98bfa082c695211dc55e98749dfe10a401a8539b1be3d294eee4cc5cd22f8ce48b1f34909ef933e47ed795ea4bb17a987ebe04eb6e066e5803a8ca6f121bec1724b141e6dc563d44e4d8d587dfa87bee704e97bb4e64ac34967dae18b652fc896635f38d00525bc68bba1ceebad015ea5ea9af4e421e45909677b5e984d409ff43fc74820c7e56a3639c4b378286b7dce8d113db8435d34217585fa5a17769e807eb212a7f4270b26d75ec331aec2fdb23276d077bc2db92385606af1a31bd80f8c478d12bbe77ea377b55f88d89fee1eb04395251b48921998ca3b9e125d57e4c04a854ab0958c6a6fb99e8294f5fb0eb46ee4098d8ed16f05ecb939aebaac66baa060e813a2cb3c3b22fdfe4047beca82e9af8c0623e14097857343413df37e00c0a33d15b4a4273aff5eeaff2412c8ee7412ea171422f6f9116e60688cb2add1c0f052cbe0fbf94401df4709890f371db058d24a8c640631d95d0b051407eafc522dd9a10ba4079257deac881c7bf6ab60cdc420889d608f21a6d65e3f45884b95291aa8e912cc482b42beffda8d4d88965a7455d7332c63f83e38912f2761f448155864ca8f0e852c97571a366198ed3e29868be7d9d5986a4bb0cad5cc85fbb810b2ea7ab4ee65e74e658ca3ea0448302118b3f5ae6f299dec276f0fd2dfd509ee403d4dbbc300f6d738bb841400475fda23e066e484c6b01157805c551328be3525b4dae7b0a0bca66c9936b792aa41acbaef588ef2cdf05feb74cb6d783bb905f2ef1210c051f3f7d5e841b393a2ee021ffa788b9fefc43824b3b0be90a51f98f304ee770f4c08581eb061af044625070615665f7f45e0470d233effe335d8507081d2eab070d43faac4d6c3347648157dd2ad8c541ace89864b1ac70c63a946f45f2f230666def928b61bf78c7c4ec592cfcdd6dc5e75125332ee933807f5d47ce6814cdc6b0b03014b654039edc28134bbbe9f716ce6b5e366b51ce858cc8c0dc0d2f5ca27481a558af880d50ed22f6077bddc675046c3131a68004df46dca1f67a25d8c3636889a422faae73e1537b5d1e5a815dc5f0b4ccb91c718a55529ccddc678d1753cd52971281e4533ed5020ca0c17bb1f40157c0d3e286cbdc3e8d1c1e9174bed32f571b0f12b6c2f6d3d0d7443245dbe22c3fd87f9ee399be84d8c82931a52ddfb8c6044b302649cf4a9296f3f98a7f9a79235fd0b995db57c4224ef912354b3fd6c48aa9d38653924a17eef311c15eb01514af562d30e63ff0d9ba7599be5e12091a0fde98ce47dd39188b7b74b9fc0e22795320270776014831d22d231820b9094cf96a4e6ea4a4c4ebcfed578bdd7952a782023cc3780bb372a0d5c650cb28dcfc00a033f08b9f5df519cc0adb8506d404182f583123d208d15cb91cf337c4bf7dd10fb493e50877795a9d640ca39e6799fda4c645411912c1805a897ba54ce5951bc20d9d6b5f4eb726955f1bb194e3f1483274243883d478c7e4046428fe629e91cba8d79d61f90a292cf66102af43b270f2aedc0ed5578ed60f93c36ec18f27776ed72375cda2ffd54b7b81a7da86ac8153e3b6f1fd8ad6dc0748625d50f534cd4ed07df7fc3a0fc5107d8819aca40cb6f3056feee654d822985386160f03efee0ed7aaa08c79fd929df20cd527b7168def1b1fee3d9bf9e0bc595d58027c67d3d155e1d20e454e8bfee4392cebb85018bd169a208d08a4e770797e07b7847736d5cd3e7205f25133a40727d07eafb64210d83a47b1d22ddb473b377627eecbdba31460b10ad72df3651f54413335312021be3500c28d34aee88a6083ce01501252ad559df841f30b51a50954d8884aa58b6cbee9a660b52dcc4996bcfc8a8c44675a479263eac43b7aeee082eff8ae40dfa183b277bdf4bc4f2bd6983b2ea69920c4059542bbdf065c3b4c5ba44aa5e83d5d1a098000e3f893747b37b4da26312f87a8d460e1b84a01b33b400b181fce4081e82448642806ae0a323b8ea17467d88cf65d539f07ccab4dc45be9ff478a472d8a349e009b6ce785dfd2ce7d9bf2a7dc568fcb850f2043870543ec431db38c1c2d950db8baa16e2207a6cbbbd3c09a78ae123980f4e1e5c41a0543dfa63a3a6bf48a167ed98989ff3366366d632ad048576f85c6bdade5ec0284ca0d030b1c230dd01636a99fdf891dda8340e778d122f2b9b25b78960dbf661d667ec5e7bc393b9f7f6b9ff51e2fe6913c4f117c7a2932f636138d2ca2a7dfb67a036ba74bb86bca72340a3bc838298b00a1a6765164d8e2d14e7b98cdfe5b18cb4c3ca13b5d426900e1dd1049f5a235aa0974d8ce5040c7b0b607468b0a790fa449b042e29333be5b8ee652990da10d3eaa7b49c05a7366b6357f6cc3860130eb26408552d0ea341be165d84d7068b2485c49ad7509565b533d6862ae67f0de5ad62917cbd891e9e65f2a50288cd8e16eb8827d8e52747d48aa3ff7fa1ea87259fa759780e4c854aba2f28a39d1bc38c87cf87258cec463ae18d049fde9a36bc073a8a15ce092e0728358effa47279a7e4eb985998acaa5a801ad603614a7b15ee6384f72c8adf437240fd6efd6bf1599776db9521d28b11a75d8c3dd18cca731e4c905ed5dce8fd49be92bc9715f543c2e99a5982a70354ac953b59c422d475e56c52b97d8c6a5c6236eb044f6a37a3bf93495ec749cd74889400915f7a546ad155c1071cf82f9beed2fa2517c645c1560ce0e512e0a16a2109d632da52b0a20d177569257aa868377e3bed06f2989de6c16057bf9b003e0c5f7710c3f7b8eaba3d8f41a4daed483345649741acf9e1c11497ed15906f2d4d0b147ba1eba1404a1fb7e3733b4c88cde1049c93a5f360eae3e086de35d2378f1dce7974e01dbc14a28372a3c5adb2980653011f85a615427cd84b6f3c13f1127abd4824ce7ee3298e46c5fc853d229095f2eecfd0ed56891896fe3de609736626a18b6ee1c93f054cac38fc8e4d73c344cbf93e8f2bd76b67689690bef5c3d472ed47080fea1ccd718835bb82be8018e8f89d77b99e8c0a5a733d761e10b87e4831ddccead65add7834ee73e42be2ce0d2287179450782e31ad32ab0620f0fc2b2d0227504ae3b2667c4a4daf6ba172acde428b2b9de4c962ca38830d03456f7c53810917c8ca9afc2fd604c39c0f2c99687fe0c979ba1cea123879817f5174fb6b4f2395d7f261c390f1f6c11e4892de5130e346649681cf10e2c67a80e9e4ae12907019dc6a8b13bb63bd66d6372f9730a0d6006032aa1bb8c11f6f355dd6a042ecd45ba9be241dcb10adb89c64cebca45203e0614995864ffab6c8a974791fa609964bc868df6be1294eb5f9df002a4a191b9d030e9fe9a4dd1bc4df78d3a1a2a81fbd3a08dfc140f8064e868f3d119c31e5b5a250eb1fc60758c096da3e8118e3e42b8c447b0f8b04887b0c6a7eec8399fbc1fd72aadf1d7ad3bfbf9db2610cae40bc9def92c8af872cb7e4fa9bced370abf4ba67aa85cb852fa1389762c4442aa2c184251ff91c123e519d072b986c2a9ecc09f9a3a1bd7d65d0ac27bad6159f81982c84c6e9d74695091928f9453155f645386b3a64fa664ccba4161afa064c2449c6242941f1dd281110e3adf8c0931070d472882cd0c45d15a251ffbd204bfe5b54628f104df18f313f3f1df67efb9a6b0877a11afe70882027733212007dd2599b4bae4c5bc6a33b8ef083fa0acb99ec15f180e2ea6cf0c3bf2884d356920e154c53f5d7afc50ce0e0ef941b954db088d7081f2c1ec1948657c6ec3a711dc7dfbd6dbb9d8d37aacd111d98a8a41d36d86cb3be065765f536a7b3fa3b7c18a70f12d4e7beaa7803983355283e5016fc46d4be54b3d26c6cfa519ce0c83790ca748a7172605669c9ccd4755f9c79b9f9ab76e0c7dce2b80831d4015a8fecb59acc7e042c9a0160446f5b8061143c96870d0f8b035cebcfd79b3c23f203605fd3a2ffd0173c86149a5c1c54ae87be1fa98dfe85ed8a48c98434ab521b227befff9c1ace19bd1c1505cefefaac00d8f0a4eea5cde827d824409936cafbd075d861fa3fc820d66cca126ce76e67ce4977e5b42185a1afcdc662eecf482546ae0363b4437dd6b9d8d52dc954a6b3a55cefe1d2c4b654d4184d70a1a43ea5e3d52fe57f13678823455abf7ed912fc9d1930f13c136a4c5ce1752aa4fc8965f4ac825e5213d3efb18b4885c2eadeed81e9616c3b8b5aceab189db8574064fcec82aedb8282e35cbb779d8c47b61a29fcf850292db555cb072f373507b5467d0b9aac11626169fee11b363f1e7af86746cc508743e473551fb71b6aa7b908cc9ff6b62b7c4ea8d2151db112314c685aa0d23657e952ea33cf5e72965274082267956016225a44977338840a4de4dade01b749c44648878172aa86ad847f6473a0e84ed490290d686baaab177fd955ad719f03316241bd780f7a63e503f21daa686e6caf52bde3dca6b81716ddce463fba276f39faa927943145159391dcfa7c97c1495cc6f07e29e794bdabb23270827e97625232c7e22825958c2a70da934bd71dfca49f66cdb36b714216abdfb95519312b06401414b0e02300b470e38f40f8393e1b5b05d7d51d885e9d3105d5f4103ac143fb63de86a6eeb99092536831b34ef332d1311b181fc12d6b07e17d0ae5c3bde4ad03972c3eddbb95473f01256cf1fc30ced7a705aeeaa29b56fe73b5da8f75bb4c1b61589865bd480b07bdccb7d270645137d039290aee772dba35a1dcfa6c646fb93ac1e883ca98c13140fc945101fd03e4805e2bb9c1f1824d31543b0af02a2204389d2e83cbb2982bfc5caec5d3832d278118b676c5bcdbfb5f6c44f5b2b803d2e3d854d80f7e1703ba3f0ef3b07a3d1a17a1c77c6e6b7391df5524225e6589f607c3c5bb1a052a1be5e0d478f92103b21cebf298b4af738f865cd95a5617bd674f830149e3d449d00874eace5d84cd9764ac20330f2db140c27fe7d676ada2b9ad906be64a6611df70978272f9ffdf1ac0ca53e96fad6b49c3f2d148bd7ba47783b6323db884527d2118f9caeafe835cb86cd6d1e95933b1e10dc6f3df72c26b3bc2bd8b5ac9ef41bdfb3e12c240309502675e4bcc7c1337f9a8466d956cbeb32c1e22b63ad5e8e688b80020dfe651c056f7549492dcdfe515120ebada7f67a7b6f21bb3ad7c57fb3dc50382c207ed2039a0bd8d90fc32e669c2f899c5ddc45d7c1e965aaa20240819c6ab7c818c9989ec37f027c5f9681d31de6293e25e3db571e29a98b991692d0ecbc1c8be524cc150f266345be437c00989403e285547d17f9689a2fb5116027808b4aa8160b97a8daaebe2109b60eedb52b9683b926bc54d40f7e0165d773de67dc8ab7cec4c7296d03f47b0bc531f690c5bc3ecf763f6fdad62577ee39e2af81119b7e00fd243468078a6cc2fdf3960c3e0bdeca56748ba3f297bdedfad491a9119f2f22a9de1db6c2dca8fbbcb54081ddd95c59cf267ec1cbb023c03482af0c1045afda36673b8fee0defdd9eaffca60bb870cc6148ad433f0aee09f532314870751d82e04194feb4aaaba3d11b65af911042e31a7ee7977d7b878c5d88d6aa735b5b1788f55b59f39c52a8b53c35f3807c392aae15fe9ac5fa36fc538a105356cf09915f721379b0724ea4cb1394382b1f6f4dd96c8623104a8c67b164c42eed8b5ab3da83f69db8f0fbaa57361f84bd3f3ef1762135d53a917dd752cd2ae705aae91ca2687b03c239c9a182fd677ab382910a0f370eae2a673a65abec5edf375b0ac1cf44e4e9d7465979dc7153c25d2dd359dacefd21ffdb26043b197d3138429528c75370bf45a7c4c7a84ae2b7941f901c860be17c6baebfeb25a485002de8427f06dd82257baae79db8d526386c7a67439f3e7d79c58709bb025c075df3047e64a1dfb7ac208d69d9db7d4f2fb7e20a3dfd5b18d0e40549a79498b8d0a7c44d5ef73f860cf4b42ff159cb0bbbb2b8c6583298f33dca36d3e1b8e9126fd1d0dce2f6b1fc73c571a0fc76a4edfc0f206b1ca909fa1eec0723679e8c3e088cfb9cc68fa7858b3601c60a3be94587ddd2df2e2da762c442e5885be266998b008070881fa4d416dd894d1db54f517da880a9d48a72b4f21c87b318f6e671eeecabea4311f072a27c7e70386b4eb66f8ed3419d4ec5eea4038a4bcc9c60c38890eb09b2207860d8766bd8602f40f562b17f072fe161345aea5a7f0a11767043dbf9e36f0916c1b6affcbb00fed367c24c9ac24496634dd0266e55bcc40e656a815ef85a9b4da663f68e2eca949a12255483eae9c720736844c8c927509cfdace25ed0e95b92917cdb9c190873d07bdb431a5f1c398da1c18caca4998b34b8d513538ec6dae7a3348f14ae207b3a14273aaecbe70ea31e7f100070c40f3a78317b294a1b86e75dba6ceea2bcdb71557097de05c212e6ba1a6de3f2742b04226abcb1cff36436552296f6881ee98863774d8183b8745db049159e62504c39d6f3d77de8529fd790be18ded60e899db298bd03141dfdb06af6ef966e51f6bc48ed7592b81792d0bb5f52b71a6348a04e79263a96ac4d4c6765d0378a7450602e0fd27e9b39045ffee3ef88db830e80d035ae2f880aab0382a13f03863a02a5ded736e5d5b725e7a63c317977aa1ae032030d42750c418b8becb0399cb6630a0db5b5df15c2fc074118c7619de05dd58cae1e0d6fc0d3ec4936c65f7c2c62f2dd467cb0acd6efd692d30b0531037da01c3697b26eb0e133b457154152f8ebe903255cf66fbd7bca10ea3cc3f071f3b6457d9d696aed6aaec6ed39a94eff64833233d14c6dcef6ec59ba9983603b457e6f030484544897ff526743c2dffe52f6354186f0defd52ef5e331b0be6a4d55060176a115f2d1a3721f014d72184638e379e4fd63f8925b3568c294f7b583515bb108da2528c645a42d1dbb5e67e4482c39e13146df0a01f21007c7abbfc7fdf252eec5fba0f34a8ba1168691aab333d3416d022a3a9969c6af1559b6048900227dccc417b839929c7ea53341256593569d6c42d81025728b2da9f812af8fa446d4eb764a4130c480d9c75de8017e46ef50125fec02b598eab927a3a4a9a234138cb31bc1d5e46bc8b234066dc0905400d1f631d6d948714d7ca14f228b6baa6cf108e6c533e750dd1b9fab2fa5ae8dcd2383ba43200a48f94793aed69d4aacdf38d149f0b3c52ad4d0b16d0f52f0eab12e92d6d80ff9258625a0cdfd8b6f383a89cb73f38735252f4130a2a57c3b47407aa0dab7b3b7c8e48f99d8344ac4e8e89eb3a30516526165b09a5162bc26ae27d4ce8ce1cc7b7f9ae321e911f950e4ea4bd648be17c0fc134dd8623e56ee95a5aebbb0f92ecda6bd4f334a5da0d9c880ed142b2cc547156dce9d1648df35708f797b1771b3ca76c21c26c8df206dbf8c0554dea7b3fdbe1aa34986384715ee3df1d4a594c79b2a17fb31a66c08d9bf08abf63510fa6678cb25d08db11086d652b29c33eeed5b490de7d149531ffc3f81bcf9ffecbe6b8654e71c4751fab5fa2c3b10fc854584eda95e4e9d85b44ad21b75f41245a1d24ad31a0da031eda1ec104be35ff2a3531de2762486bf3fa4c412b478734961820b007642393cff1cd63237b1b11f20290ca19f2444b35c2dc0af34617b8a498d890c2c2065caf4f6054312510634c2bb9f8361f569ed050c7ad5865a560ed69d640c44b145a32ff153a860132c9ba396005c842715c381d7d7f233d0128c322e3ea4a226f2be2de67729cdeb7940ff3e2dbac60f878eba07d37fec47859356753e424f1483c8c5bc7189f2aaf11ab6e7b1b154285ecd245d4c4a56904f0b9a0711914b2bc763ca99f384c574b7554ad56498441ba0a1c7da8c57e5b99a408de17dd2c6780f0187980c595d396bcd79f1af29646d55956ca8e663bed1266508f22ee962abfb13c951438a25d50edd5b77ad004c5341ca78f15e410d9a6ff622a14aadf07159af8defcc2139c0f72113583543424e703c0a28fc067f9db7d118ce5e8bf158928cc63e1922fb3ca6f90bd70b781de2e1898606d4581a4dbf6da38e8d9f12ac07d99f805bdfdb3eaa77a5867dbe0b8acbe901955701ae310f229b399b28ad1d3488bbe7cf86bfbeba17f323ff08a2de806ce6a97445d46012a0cd324cd7f25bb6a05c6c61e444471c0a466c27cb21502ca52fec94e811e598dee99d8cfe46d157fadcf76e9114e246f2203553128c3037ba7fc582497f3e60f75acf2fa1f216efa5e7ba1edfffd870eebab1ace869feb9c4b99ed156f42c94156ceb88edac9ff07fef8330053d589c3058eec2e6a94a8603881869c71c98ca5ca803dd207856fb0ae9a998aa99e4d67427b9bcc656453cb84bd7be1c80567a3d5c6c33f4b5cd3c2acec866240465d980e7be4058195bfe7a301bdeeac811823c41c4f5ea552a08024fcad01980635d622f96f4c190c90fc2533b7e8f48aea3639de671d36abf538eaeb8c77d0523d487377f1f482d6e54d76525e600b83e126e3f5515243692c9a4eea6c52f15a0aa1850064a29c4e444276bba32983becad084b24b77789e591ad48fffacdecef63f2f5f499fef9b7bf2474785732e1c53a6e310b879baa0daad43ed3a69b86d2546814964bf974fd015e0a1bc7e1e587e7297bfc2653440c64f0999031432ad5a4c8c823f0961551649aab02c5d0afe45e1e7092abda1f14618af46668920393c1ebced054ec253a1df007087a19324e3d2bfb608f6374d8d77d92a179fba09f27f8e9d08d36340d963463359595a63bdadd227db6f71ad677254c949bd409ad2513caf0bde39691fab7441557395a6b6cfd1bda5e625166077da58e9398d5688b8df0459834eb1f0cf929a7f58b7791e8e51525e546c2b650af0cc11ec8d894a0851d3c7404bf344be0efafd417700f80ff01a28dd6c6f968de632b0c805b46bd3ff16273d53bda8261ea2e9cc9a0f641ff966048bedc1890a7b680dd2b32b8ac83009b873e136d0752b270e0664fde5af434b280f2d388ca0e4ce87ab0a7b109cf86b3c2f0b739e41d3df3132d4bf1e4d4756d620a07e1ad5c5013a6e80be9c4271d7668992d6fd681077ba0d040b49c242413a38c37dacbd53a91078ce0a50df0b5e34b00f8010b6346ec367bfcd4c4cf293ca1faea56707df35dc41e1d5ad7743c920d593f8755f6eb2cc93ac64ffe962e172ea74fcf41ad5d08e8dfcc8cf368bac8285f8bace084d6b1b1ffd74e7594148a3a361dd4c91c333b4f7b44eff8cdb0f2f1938e78aea7b66301234ab309071e4289149c0713667df7ea90131fea0f845234ebaa9e702304bf7a6eab0b25b950019b8b628d6c5a3e22a03ce7ddd7f34f50fb191cf03c2e64b7550a5b53b151e94950c14231c49b994bc1d9249e888f2006df2dc1152fa9d8ad11e64d5815af43bdea0c00ba7fb96db477b9d5870ac160b27eb8e4d913e8c1424120b54a92b99423346584feba2c17840df15b73c93b110e0c4b7a9869fa7132970427229b838a3bebf6ad6d919a95eb12cf4e09e4a25f4960ec39c88961a14e9d7e9a21652a9ea0e342b56da7fbe6fe585ed18e37589aec0ed76466688fe05cf6f825aa85911e53895642a0184ed76cfb7e1504844391765d05288255d9899a0d534f226128560cc797e76cb92039a6e5ade42626bed300e9b87a2c2864bad840292363d32f3916e32b7e6927773cad72c8f0bf421a8791a80b3d416c82836e5a763c1134ed8fd4f30ff0d3fed97b20ccce3cff48a96d598aab73f09d56bf7dc5f1197ab658dd1b0494a8dd3231d5d9a3028a46f04f9a1efadd3daeaaa63e603e58398bf5681ab4978d2452d436f06b08853041564ca87dea3676b61b73e850c6747352cb938884312edda2729d5c8cf49ece114de7673287c2e2583e084018f24e1cb801d90fba5b7b09ea928b1a0ac85fe382621c5b07a5b131350c8c673c2f9de101cd26650f23e6a14bc50243d3c1fe8b454a0d6c2c73f0cb2abee41ce2f08ea7e4c44ea74d7d48cf0337159013636ba5338220b2d26e4447d3c47e9125bb3549b29d3bdbdb8f7ac876b33e2bbcd152febf5e9069f3f88aad8e5f82ad88c912c8819fbe7605d376957523491d4483b9d1a314765a7d1e82c647c2b11e9d9535fb02073a0b362d8cdc5b6f63e7214db43d8fb5b8b9733e74bfac9f02b6f5feab15cbcb111ddd06db5ea7a9cd5685bb5eb854a266375485bc06e266c35d12a058b4ec908bbfc299d7b700450ae84d7c0657d7aa6fa55026d2b263fa561efd88ed979583b7e162ff146ddb806814774bb736abb9b58b6d6c61002d7f0ea1720b1b1ed832bfda99260db00c577d48897b69537e1a0157a6c559f5f42bfce7b0ff15f139ff1340e49a23327f5bd54c42465805c08de766023e9a4a4fc8b2ffae947c6d40433425a1e5ec0bc28521f32b3fac9a1f2d2b7179aea9b1fbd12bd42487ab5a763f9e9e95ff5ed0545e4eeb2c71d86bc56d981f60f23cc34de85a87782bc38a30c02ebebcef21d9851974f0251e736e9e516600304948f8c4a1f9bfae4687d6e41a84355bdda96883dfcc5754d7547fac2b5ebf8dc946818a6cebe607512e20a918bbf77f2be15fa487b9ee543ba4cb8ea6b1c34c66a73f11d88e3387b63b123b1b10dd765ab4d22431e6efb58b3af8e9d17793f4c4ef287aa8532fbfed3e8eca0124114f93164eacc6dea6638ee7e7063f441256e0e0896f293d893982d5c90218761ab53ce279250306787b0e01773ee45e45b248f661ef2edf963c471c02a952a1bac419a1f6a1579ddf3c66945d6d8c2aa1120a77161ff90e26de3e9eb295634d7b1dd388da60c35050941186a0c1c3bbde07e83a941a785d8227824f809393f6259269fa7a65da65161caa2507f1a229501d0f737c7bb7e34c7c0abe3e05e5781cde574338293fb710333ae5f404195e96efa663be8aa254b0b85f7a6318f602ad2b91959d832a5b3e3fbac86ed4896ab68bf841e1c5b74448a6c22374bd9415498d6d87abf1f2fb53baeb67c9598d62f1b9710bf2f98e750a770d4599256aaaefc2880759bc85fb8b69f4445fc7db2bcd0b4b3302a041d01fa916b07307c2cb793b4f23d6c9d363f1b19bf587bbe62200f7a4f5fc6fad0ba07eb6acce758c4f27e04c190dd8f81ea06a11b5020fb240f2cd579fc27f360ad7e65f9c4fa32e64304ae9555edb4b75a2690a30e70301257d4fa6610bd5279e66172cee08fe0417cefd9887797fbe6539f7df8981c101b3a16ad18c6ccb7a883254e7ae6f28fe10e00d03594dc2a31b02fa25a2e3c277572b4833de1dc79ccac4facb6f389ba7ec17a60b5ffba08ab2d671ee47f1141c6b685bb0874bd9880f0f4c5a5bdd91df85918965a9e1bf0a6b018113e1309cb65cf17dfe5aae8cc4a7776afbce5e200158793c22a7d11a817dc0728ed42720e43345cb44daba859aa66e0bbdbe31f8b25f6eb8eb2774da3221a413a9ddb49e90a656f62721e720f54c113c9adfd74aa9522cf2805f4d84267bbcae58333f5aa3161b87fc281a16f102a46fdde5e12525ab88875479d2bc75f74d352d49154bfe4e6f7a94d5921d5eedb58638e48a2852f23bddeaa394c7ba83ceab4ddb47236ca5cbccf695ef888a3775a6a63ce30e321fc8594a8b55a8ec58ee0570e679ba47045113e374ab6be5de1a34dc8cdf95c93157bd56897824fe1c01cc2f9dcebc803f11604770b15c90a4e4c7c138fb1039eae5ed8f56df29db21726cf1f9b4a6ec1844120a23789c786875b51fc37a15e59529b77260e71632f09958e50fc3dc90a00cea0540a9147358cf293f98964a76f2a565c9f5c71538c6f8d2a955d903bad638329a910f31c1ebadaec8df66d416988d8c67ec1a30725b371e7770189fb373ebe3c214b5f81132bb90028a2a2efdee65bf71ef03767788f431e03f2450303598f9049c4b25576bd506c6f686282b6f71acab1849885822f798bdae8dc7e5b172b103eed95e8b0ff0ff2c4dc25bf8e1234f15ffadbd2b45faeecdaeaf5cfaa5bf7995a1f571b3ef0dbda852ad35e22f88ee4b31c4d5f62bd938d49be01d676cfc96b01bdc00ce3264c6e5190af2757ac50fb756fbabad868f60b1876d0aaba34eba73e6af8da5d68e01cdd855bce8df20540dc9af6ddf7c514c7b33a4761ee8c607c35fc1faa42edc69f695a85cb013cf6de6a82eba8439658e0ca2ab8143521b3639c8f3e3135296f3be700c7575e28375ed5d0dab5cf5cc48f105d7d0218f4b0e64ef471281f09d687accbe6ef97aca358ee1bc49503d4db0b3aa9e394b9c1d91c3c88278fa6c1083e64c409feebc26d603e712dd07736829563d415e611688b38e7057349edceea7c6974dc2c8898f64714ce71d005c820bd5e8b4dc212a824653fcd6956c92582e02dac124afb84c703f0588c623a84c1fd4e2fff3cedf64fbbea5aa85f8be7c8b4fa6829b5834318cc35e7fb7786e732073c06a01a32fe2d4c0e2867c208059003d3ca543c275b55f063523107afa0953cbcd0eb53c0bf9e1b5831f198f0d66102944695899113fe1404181d303109697bcacd7e000999c03c893ded4256cdd9bfb300348827acfe71a5ddebf15592eb680045a290d0b77e8c23b220f0563f3c1c9b54adeb88f714c99b636d02e489c257f8900a29fbc60c40a5f86ea118ea9452b127c20dffd82ab906260235084d0eacc2214bf33a225d66dfd6746d84a33f2d540634d7df01315b382e1840e2291810b8fde0f3e38a9a2751c3298d7c618c4129dedef75bcce2a89c408a3a7224ccdd1a203ebd03d7f960e8aaf08432cd8dce1af36679fdb59b2b7b7c35c3b8ca6522d374714a12fa3cbd346ce78c722fe51b88acd3f7f03449f6716c3370bbe4a39d28f0648ef761df5f69eaabd4028ab1d1b4e7ce834c0a2dbdccce0831530627ebcc57054b99f686b205e9d8c80dd7c6a5bbafd2c33884ab01da3e721fac93c93063ce4813be15f029236d29fe904b17d8a00d6972a7f2d2dc7a636cc14941f72ccaa0f4887b2cbc421889b1728271569847eeb704ff2320ce0df8efcee407d82cee3eb4bf08c987fd62c96e086cb7dfd9da1dd6e4ea2fb439725b62d03725762fb86e140b153395451b9ffbd7b01a31973c58c454ba9a50a4aeafff2d84f8b89ec0fced41731982d8c70be2ddae9d1447c621286c85ad263838a6578894399fded44b1a201974e7e9021ea347e55d952832545d1aa4b0516428424c69b93aa9ae2a7cdac2c42c34fdb85324e51e7c729fc4cfb4e41491a59e19a25ca1d6caefe0ec219505941abbef070bbc22c7670e7b185ed61bbadbb288dddb9c10c7d13b966cfcaa67a9c0a760444ef1878d28ac918703dd62650ecc165e677592689d84f6bf6501d1f8e60e93a996666a8308301355a048bf2a83e4511d07d7c9983259039fa6dbca5725211a4129e296433b8a26b89728a3e8e0905920cbb579bb3384f18de4bb0f92f8cab2ede194e62e6eee95dc87424f8038fcd0a25ee8106d1646fad010414acdceef9f200fa4c61282ded32243d171163288ad680d6f167b47c64da826253ee45700092600ece748565202b3b9b75bfbf49cf455eeab55cb211c4e424d789df34919e922e942186a499960aa9ce3746b1f0a3c2c00610eaa07c68321f3b504b8dbf0372e240817c359a4144b7a23575d1c98e2908eb6f0242376377761d68a580c41f93141f80fe56006372510c1a45de52d5edcef20cc618276e078d2c4dca9981cc326e9148b4aed6fdf37e5d0903fa117576346ad50fc7ea83838c380c516d313d446b5e3aa5fee35f9800c0cebce3ce62f61e007cf93bff6b3e3917e7880ca61abc40a3e393142979f34d1f339684b1893b18aa24c46bad5b4e5a2948cdbc9d5632660b49e30bf08928bdb7068bc6e7fa6ee76d804b0c3e9fc89d6cf55255f3881cf42cef8c2f7b88d196049fee1da5c3fc863bafc6afd6a252c22465683e48a7c4fc1ea8d6b8b40bb807bc0b3cb452267d2b1bbd7312fe044a5991ee65294c07c1deb182eeb75802d57558572d88374930fcc5636b23492f5e0cc7511fbed245bec7e2774c7fb512badd93fac6811bc4e8f237257b0d3171f71a8c78f0e069e13d61e6f1d1413be912daee54411ccea5e9f3a5978bf0be2877c7a29cefe1d428bb1eb2bae0a5d94ea5310c1525e5c1a80b8647ba56e0eac548239b4751a778d3e2440ce5d40e9a340f52136b28e1807d423e197d633235f12f3c76a49679e5cd30c89ec779461de423b4cc27caf1b20e3cef8e0f65ef1fd9181a389fb885c9d725ccafd3204d43a528ef47f9995449066f5797924037a4278987b0d40b816c90a74b6202f7863f94a58af0c7ac646097039eec8f26c0f3de0ccaba5bfa12c06b06e94ad11a0e1860408ad22c4944acaa114ef0d108ad6f5764de38e8cfcac1c1f2794f440a8a70ae7c241c99934a521cd72f2863b26081d7dc6c13c0b2007d09e18d20fb75878428d4c6ec5b7e7a3aeab00a832cd46a244ab6042f35ea532505c900c1e951f73f24de267fb5764d5502d8332935992f5329cce4f5cb4c2a2e07a1b1a06268949b5824e5731f744f447976490789e3e2b5e68a14f0c64264074d7217d17f13f6ac6e11919fa1c310295dbacd60f53696511ca0d26555db00dc974fa30ee74620be0dc2b46ebdca67723d29c4f0efb7760c8e852c2cc948b3ec4404d9ead5628ff0105f6ae00a0b89d4c3ed632a2526570922e36212d53e9c82477805a980b398bd9d65846ae5d0ac42e4b15604b9cd0574ef8a4a7a54876414363819835a053396bc4460310278e2b60b482c06728674b95826b4efa5e750a66dc101d8a9fd47ccd45a2f615040469ab5d0fbcc9fae1913f3fb8ff14f3920865db73a37ec132b90d84a3066edeb073856ba6d70af31d6c673cf3c8fc8690c0a615f73928215b9d50df8e1e256b0026a8c380416de1e275b1da5ab63aada938dcfc5f1b05b57110cc881a9d4d715e926e966984bb17f7680b4a2158b84d7f3f5ce0c0179bfa49820fe5778ab0c13aea64c7b62ce8524590798e0ae7c3c3fcd04a7afbc327279ac107c477edec913378918aad05ff8a536518599a1ec380d82814e6970dccbac7dc6efcc9073fca6ce2219ac653252b56179a4cec9b667ee8c2392e63af52e78d3757550855f104ebc5954f22910434dc051e41122c797e7f029d761d349dc4cf70786954cab1b2d69b7403b9d3703df4d7edecb08f1aff9dff86fbe5fbcfc3b185e0d95d3c06a37a11e692c8387da69400f86fe9920ec7615d2296e62d100ca394b3765fe8aaccb34c86a58fc0df64f3071610c32d8431986945849ef693f4a4b2e05bda489fcea001ffebe93175b6c7e177513a7fdc2bca447e862c9d87dbc3577358d651fa75b7bd7df510ceb8f0d6459b76f57205b1ededcd67a305dac27682749a481d0811118c420d67c8f1c637ae59571535f9d7035c47e3323f525cd3db70f4a101a75e4b6fa308789b28e3e53fa723508d21bb1638463155f55bcd7c6554e1f5f565ae8b7a6d48d9f6ad4d3d907f6e35c18e422aa5074c02c8de8abfc3d67e7f66a0f8b0cb6f826fdec1dffc413051d956a69b76bfdfaef935b901b93e7a47b28aa407ad66fa5f1799f171f725dfa1bc35b9051a1763f3ac9406c3e67c8bce6c399c2d5fd9c4fc0c188fdb364497ff01b0b907d91e1454801eaba95a8c39c3c116f575546dd69f3978e1c278c5af7e7cc1adee1f94350ddb652327ace8799f07ab3fed95077b1f2eecc628af114877de3de28bf93c51b1aadfc1cfbde5a0849062bb321c82f972b1c57d1cff118206a9bbf2eef3e5cb3f88dd5cf6e3c8780e3289c353ad09a0120b575345d958ed7a0e1193d2efc6a49942ebf299ca548804890c02e9390758361e218947efcc819d72142f7eb397b833b0b92509e3650ee666fb3823600351e9667b6be42ec0c637e57c062687355030c9987fb3a2eec63b4877e036ec90e157a9e2849a0c02a08b371e488373be2fc60ee84f6090634aef0a01076b7d2347fbd689b2d86065613aa572c880d262629e6636b32abdc0632824e721b1b7345fe98aa2ec4e850e5e792d4a8caa1f03489a60c6e338c239da32ccdcfb98c8960fd85a36304d8a98ac64829d6a92567ded89ad7de43f9c3b05303ed66d5f0c4d9fc924926c1461e494e852fd92c870df5aa8916952b68108837b493a76b4dc16b4b0ab327266a24589139c2cddd6e0cb15c3a743d885e7355e2a860d78a12d737d40c516093dddf2ee61a8126e911b79ddf9557384c5d48ebf6a47ed97a915cfa62902ef208f3bb44af9b3419461aebfab339ab6f67aa2dd0811f8b244cf3e426cefac0ff6cd786fd49bb467682f9f638e8eb0bbc687897d3318d5a89eddcdfa2d1d0641a57dae0328e4945d8af93e180b586c9f00f5a2a6e1aa8d77524603c796cfc34a07867279fac9555bad08ce30cd5690499d67433711f98cdcd20a5737a628753e9c9501ce9074be655f0e5c0763d7617e8835748b20a4ca88772f5f6da7ea9277c95af8c846f983f47b5652833538f16fbdc9e16054b0134a081a71f59ec41f4ed88aef7c078541563353c409adad9faef186711c7c396c0f0612cde3b6ab34862a920965ec6debf85d49387fd15cb41a31646c3377da0b46db99b68972bb1bc7e340a3f59d68c297d71e8454627dd51481f11fe233bdec4cd9030a7e0ed7d205b41982f6036e6ff9ae7e400b0a742c6d233e8a8ba0ed0eb64aee63a1f844650548f6067186c7bbedfe6b71bbf65976afde5e2f69abd2e8e3e021c70942a3d35348e38c258100e90109a736c728519d5c939d6929f36c9d4587561f12c53d71e6631fc9adf0d70660a6fcfdfca80893ab8292730ead4f4e487d296a0b42334d8522881b401224d9b4472297f9369440b23a7cfa895ad491e6175a06a59987641bc66cfe687e2fd0a4bb3190b8be88d2b662bcd08c50edefcad07ab13e2726bca0d635a42d12a000204fd1679fa4f42f317d9c53b5c804ebe415ec5ac9bf105ab74ea4f5ccc53a25c948c8d79dd51323d1d4e19c8a13d7ac734d40e10136b00b37cfe706331e22c42c055f0b64e7be317834beeb91118d948a4ebc9c451c456934839f9a830eee5575a5201251959459b6bb5a2a8d4fc05231ffbd10425b2c922b0915f1b6fff20fdc0f4ce749898aeeed073e22a211e8930ec1eaa4b612696741bab6719411f46ccdf785420097efeb7c79a968f54b215a703ad92ea14a52170711d7b3e64ba6b27c33bd9702e081e8124e54736c864f5ab73a587a047f754583ea9407539e4de6692e85f4ce2cef0961cc420205a630d537023f85a24fbe54657280bbc49dad4c987d0b57d0d145c286d8d91bf9f0a4bc027afd91655819b3dceb70201817bb4a6c39409c968cf0ca4e8ce747feedbc90f01e04a049fbb5a31e48b45e2216c72b62fdb2eedc4e6d839270c702d77294ccbadc1ac37d61a545dbe86735b9954880177a106f89c14f375f6ae98e8f75fb5d2bd134edef9beabd7738d77cfdce26a3c872179c9125eb65b434d91bb0f31efe16174ecac6c2aa0f0967d32d4cd9a9a2a5c63213f95be709c1ce623f3c1f47c24b7a95b7962f2dd5b7be186117e0ddf61c29a931e6db263c18fa86982af25cb3a78b8f9585e453fb973db4a033b31ebb1a1679c37288e189c941e9fa6f8b908a83fb6a0225008f8a05d1df234535f8e022d40c4db934daad51691d49f8e794739f2c2d73655de40ad331e484d77c5cf80026747fe795cf2d09a28d6318b6240200a46786a6078232cc9730b8ae9995492732e14bc2ad135a4f17d5f59839db603f0a10db96b2fbc301eb6d620dcd78d149e48e916d42806bac6fb65b49c6f3519e9e34811c3edca9f15df78b9e8ff88962f6ff20a12ab4f43cfb2af6226c08340e6871f08cabf60db2ab9aef6f62badc81e2e7d0db5498ffcc2613fd4131e1df5e96dfbfd41d314348c8df86db6b410ee39abe0c2cb1a3ab6445d067ef212ca2fd919b95d4c598ba8ff0057c43e6d202c5db17f465f34b8a91c379453f7491c74e3ae1bbe9045942acaed76dc8dfecd599dcc4ec50b7c4f4798d2086e60e083defe9fa770acf8053219282427e229c4328680b3971447d5903be53633dd33de92e9e56947a8bbf4302cf48e973c0f3573c5def1d409708066ad465fabef58330ae5549b693cd179d55043929df2333ba36d701fae01c80d4775a8bbeee8fed64092f31bd41bd9433e8373e96002d677dbfe8f9eb9bc02a8a2f47ba5a8a2faa89342760b2f0e212674a4c74b3165f360d12de7c68ffe09d552d9f34c327d9b1dd8ef2842c50249f41546ba90ef10c3407dc8a73021b5392e52b7f4f37d0575ba16a0da6cc6734f26d7c8528a77dd9b1b20f53a5daeabdaea5e7d3b2b7ebd5dd133937fdbd0dc3e3d63f00b6d7f5e27d4e0f08ea6894e8d1102813ccad50dbe19829671073a1f9251d5158b043545050db5061a8c119891ec36cf398e984b2f4a5301a482efa70541bed53d35b2db4caf5aa4582b74bad1053213bfa5465143a8604d640d899ac9da70c06b0f9d6235173643c66abcba88f6080e484d56ef22ff7f7ca17c883407a7bf2c95343c551933dc1f5e7579ffe6aa88d1bdf8bdd2f359bfd9f26c3baaff2d22e6b9dceedc34b6b457bf38a0bd72e62a72c8aa6852cc3d0758e08646efcb71631a8e79d3ce763afd1c3e85ee005082a9f60ae40259e96aa8d5a9b5c0ff2c2dcabf1d0274ad7ffabbbcddc5f5bcab247a4a7894c02d77a68364732339c0be5f666874d404becaf02104dac2390fa628f9234edbd4b1f4351a13d5d785ee251b56d3b68fb85a0c5241040558d9b482f90d7398d6c17fd2ac2db98e9944053b2ad4ecf1a3d1790caee28bc38327f960c06024d08bd0924f01d8dee33c5865b4e298668f6aa861a7247f1c1662b3776c8f4d21ae88a68b67d6b337bffa3fc72d1263c7087abba41a6d97e45d438a4b15f54182f71db7a6639c5562f8262b50eb08cd32f7157402cd97f853cb391451eb4649c178d8a567484d73537f20f7594a9455760b6c351201e595904caeb7b46f98dc45ba36b3a2b0d5207484e2807bcb6f69a6784b96b2a3d60d489d728b8eecdda4506c5e9b745548e340375b60a51d648f19ebe0c35f16060a3ad666822d1dadb7ff092616927fd3c0d141fdf84814b2b579fb5ead257e77c66121d8c20c9bb8d7c7d4d0fd5f621fdfb8eb4b4ea45abc03f0fc3a6eb4f6e630489b6a9ab90776878e12c7cfaf371ede73d4bf3b5be17a78bb5d7ca9744244ab929033eeadc9e8662b963b111a08a397de28da38b8b7db89f67af170c13a90b2cb96c8bbb7a1544e3a1f1eb0baacbf4ddf7d1f04778347e3625ab0e7b3541442ef2124b912463a787b46ec439d7ab7dd732ef8d5dd58f3bcaf945c23d87b5181f8d4a26de7624e32bac0b8760d5b124666b1d0019ff896512c46fd61f3e9a478c07d6a38446a8f8bbc152972e93f56f2b8590f63128afec45e185725209f14a3433b74d77cc1878116c7db2882363420a2c3944328eee0a900f535b6713b468007938ffafaff364cc825da824e4a62291dd45755cb59a0b1a90bfc50211d2fd82173fa1edffb5974aec84a88e9be17eafddf89203498e9a354c673a8ee01cc0986c36b97231e49342dfabf37c92295987666434d2667105d45433405082ebb4520c483ada53bbe6f17f33ca3670e2d3f6e7fd49ae44bbdad2f2a638ec5889232a94611ab57869bc767add6e2d19e5fa33f7b8d93a13d75458377948455590b0414bac327f9fefec7ab03a906eb0786b7a06ae28d4843d4dc104dd5842e233e0da3632f0aba504f5111817db2e99de37842670fdb1050412d98b3aec9ced5c0407ce904edca5fa5bde8c1beb5f90affe067ba657847e8a14b540b7993718d10764d5339530b3276ab3b15b38838cae66740bc414de859b26c412115b78762f694fb497cbbf7c25aa45489fff9c6470d2fa7a6a63668b803226fb63bec7a1396192dbaa2ec864ae3634699575f5777c23962f9d19999ce5ad51495c13cc81477d2c4712d96f3c21c103987df10f90a804c8290da1e5fe74e902f8632cfe2a03d2c18dd151c82fede3644bbc93a0c6be424b981047a2f107edae5b970082be3fa1970622a61cd54bf53dd35b7ba06fe6b2d67160118e525fe4bbfd03787976c4293da26f83a671ada272dbc99db53037af2293c890ae3f47a848bd416eaba58710947f80afe743890be35e7d386a88c01fa034b27434735344e4618b7ac80d160750516b37c0caaa4f3cfb83e1db819279ef14fc7e70fa12fc5602524fc58cf27d544393edde6808b34f4c4025d51b5befe450d26c7959a8c46dd82fefd35014f4ec5d4f81b1d9d5db85a842357cc2ae139117b3c64971cb18a1215d99414125721aa76b9c4c1d859198a2e6251aeb5a303a6e3de2d49615eb91fae1eeb1f0a3f0a164aba0b42133ba6216e172be4f12b5db6a568500d105c94d9170cc15188f7b3ab7a5a17e67ffc60bc4792a1177451b4e308c133fa80427cd3893b21aaec18dd6d772b54c237e2bcf7e796ebab12b22767dd2700852ae3b668ae204a2678e8bdda93773109efefdabd2e408c69886864d6fe1fc0f6e8dfb5073742ac62c5a86aa3ef464623dc685ed22555b31b2fda97f8f7251c8f5acc4af748a6b5d367a265cefa5254ea5c272f129059d62a5e31dc64676c18df2a1c6d1a188f84f4875bb0caf80084c8bcd4803fcdf191b40b8e45fff94b34b07116d625949f4f9d13609554dc3f142ef05d6894e09c5130eb2795bce65dd04f11721e6f647d6922fe03153d5a36ceaeda2f122fc0475ad3b9f8b8793c92b6fe26af650d4f1d2ffed5068d48beea97a97f4d59a72b99735869caa18ebc070041de612abfe2686a81869f351103baaa4476d9239104717df853ddbfc7735b387da0affc99621b8d902bdb6a226c20616619977a5079eff548ff6d1a3990cf9c994ce720e71d44ba8e0a2ca82e070ea15e3284159193391206aa286ce8f50e064eed923a6380aeb45104b449e5102d794bf799b480d2abed7feb6aad488b76d0c27b6f39de3432e83a9c571fd35a70b528f0c1663f747e3ecfb148a6de15a4602e199cc052b6f06740619709df1480b4b3d61c3882717521dbd77955e79b9a409cf08ce9193274585997e244bda8fcfd5e1717f6e2629f5cfa3550284b67e7b3110202a21e29f58bdc163a5f6bc8bb8157dbc4f34ee95853dae56f98db6e36d9fa14d86a77f0c68c138adee34cc5dc454fc76b71950c4f02abc685ca2b3315ae4cdc6aba0166f783a48bf2c76cd5fe7ee1c13f4b7a877d92ca15357a1927a89be24080b2dca9ab9d03b686336c845e130703c2d262a5b546cba0c9041bd674d6edc4543297bc44df4c325b365f06284d7ea69a0bdec1ce2738b25c34b13e022b9feb87c576251a915b482a217e2f91c60f37ad01093533071750ec5daae721c315f97719a496d661c44813051ee21b89a540dad982b434d44a9a7d6caf9d0384b0a70707f526ccf7695c48c5fbcf74bb049781152443fc1b2e42de8b1d2801d0f8177955bba8a69f5fd44e490f15b1dc44fa349307f7c9bc261c05ae2b77c390bb3171140eab2bba16388ef056ad333f42b761b49dd9a6410b033bbb0d5f75606ff5bdc626960627e27906632650cf696cc710fb6530eba4c8be90439aefb1863e0c70ca3e8c2c14648339891596ac50bb5725c31cdeeaf35943a8aff282224ede96cb45aa03b9480f91f28d3f7d00219ee75d18385ed4a512d33b049bf10550add45dd40180c7ca8f487cc7607b712333a32f368d696d660ce126510e9473bb1094845ef8b7329a5ac132fd8141bd46c0331dd30497c9c040cedf119343db5eb704242d61d7436247b470ec1b733872429fea8a31bbe0297e074bc5bba9d8df009572fb4e627661bd569097a927e3da1e83622df545a5bfe5d2ee7125274f123ac4a66c7f4f284d6a4403d8a3eda1d23c4c66ca08ceeb63802b05e8263d86f1e3319d95970e6f6fae321bd645a40df4da871df2526f8f60871d770e160100af38cfa855bc7a22e9946bb7747afbf8af7a4918a14b6a28016984e0fff01fe7582c97484e3644ed8b45ef366b0dce807fbd1c8499501f0cba52c54584bd651276f0caebf306366bb5eb147db749c9d7b173d01f8e53243c494623f770849834121439447e05d3acf157f185ae54809dfeb18468330dcb15dbd32741c9a7c7f075f320bfbec078492b6c24e37c73e8e05255e7f33fe1766abc0622ee1671ad2c992611afd0afe5c6c73968375bf58c73aabaa0d6412c42158184d524b026ae352336d003b486c8d5cba6bf9eae15ed34a9c371114e435d57714f10efbb44d34b6350b59db89039f60ef888b41dfc0f30aebc510fd1d70754a67659b58fb74495463fe09fcb333a37e33862775c1c1ac5ab8a21c01b8ba092a582e6337113d55420fb775159dc68829201570be3becbc4dd5426dde3b6cff56992ec19871346d03d4c6183d3f51a2c208e364cb0a205fe5dcffd3e5bf782a43cd306b0165846f452acad27eb8fb83dafe7e5528b3869eaf162c2a992491544ad4834be9116d6611e7144a284bf76fecf2276ccf43f26295584b7fd22eab8074a427e240735fbbd959ff030250d7492a910362c8963e6afa71896ed1f59192ca58ace00d01ffc0dabcc88e6776823e343011501c189993ad1f6fbbbe8fe87d6496e75064f3ccab6959b960080737dc00a76fbb800af8b7b90be273f8440da1b86d8cdfd98950a53e44f7676c96fb42642bd20dcf96e3ffee022b1e28cff89d3438379c1d6273fdfeeac4ee2f782d4e780015636056eea6837af6b3ab68bb10c0be3bf02248b8b2d06fd843bea1e2c16c73518b00f91ca83eb73453fcd4df3a309b7d376ea5913c257a9b397bc3ca497c76350fcf8e2cdccc60b0707020cb0d60b52a8ccb792a1913390963a07e44a21bf61aa4fd8deb3b8d7e332d0189068fb03df2ba510e6ea11085c4a54f0caf1a861d1da61ba8d2ced71886a274a63308340f4c823970ed1c5671b0be09bcac0f7a3060d76c67aebaf149efba0869d4b211f1099256083966c078b3e9c1259792266574693ffb41f798c5eee2f993fc92c05e142eed68333b1a6773d03f2389d3c8b1cf4edf60e718aa56be1d4f7c0b9ea49ee316164809f14c3a5bac759a25860e0340d337a0568ba4555e8cc4fd3a16e0337b70ca2735dbcba79dc0fa596573d57eec1b9b643f845510821aaac7f5475a146bb83e2d50ee30378ee34cdd5d6fbcb7fe5a9a87d41118424de3574c4967a9dfaeaf361e550926705db9a4e947007cb989b6b102a371e9cecb0b9704e5436d855d95558aa4f78c3c0d6e66300fda7f3f09dd986618f81e2f77a7c25aae6863e27a837485535f81cb9927bfeac1f7b5d603a843856d5b9d6d7e96503d251c6d6e63c16d408963a3bfee67e31c5b6a8010869f9faf3456f4c2ce3ea467082f490cc0ec9ea5b3c050a7e505c43eb874e1579e4d5a5b766f1a0bf6327311d7287d1f2bc7f4d6de7bd00240d5dd929bcc3937a6757832bc64520a124cd4bf95e41acbd755bd5e7407e9cac6359d1321dc5d2c4f60833830af6a9c42a184197196a23190760c4cd674d8afa7220f2640204b6ca196579b5afc654e2d3191a19652745e55f7e154c2ce232f4d73023cb07173b9850a505b9585fbf1c1c8830779e6b8df66596b635a5de48098cbf4fde4bdf67434bbd9651c00536d81bf61a22f0992aecc7a77b204e693e86eda13296af7b3f87416aa7b54fe07c3be9fa75f2be3a8115edc6f342f288701eebc3aaea562154968fa312ee7a74303e14081128d84341165b9fc8d9f92bf5c6c9671971164c959d710b66c0f47cf14f89a07958d5155980e449f219b11a050b75c4511dc1795b17504f9b993e716481104c31f536cc9422815d2c3cbbb45f0e60866a159a1c8707fc77127a94e0b133a746a1f5de98cdb125adb3bb9f7fc74caa3ae7bf445b6cbf7bf3ec98d5df1a147e8974e1dd83ecaf9b2df56ed1e22dac92d7d592c21e4bb0e6c6f27a6facdd252cd539842d0841719b812f6adac381f7cfeb7ee55f4f0ef4a376816d71ce765476614dee504f910ded467294477fffac8a69048d59df18ba2ad7363804de820e4512df6ade469974aeac887058bbaf7a95014e0874f660326e80d13d3e9ef3aecf8d94ab27d691d25d29ad32523d12a84b3d3eaca6cef9b26ac5e86f454cc66dbf4798ce28d4ca194b905055a11d90c7fa2b182035e0223c4e8400f6996fad609feb2a1a58816359eeeceff42c9a3332ded622f16c532646b578bd6cd750386977fce4d97d1350eca1aadd2a63258e620342595a81409f522fd6999497afdaea22615b1a1fd760a61eb357d9e4452ee271b61b19bd7af0a52f1b5bf86513093246e07d5dc6b1150af8adec97ca50cbc6fd6ae44efdb6544423d8492cdd44491eb7109f51bad6731d394886082c8f8b1efc05baad60291a8649f16c35078843618878c6e9dd03803ece1923616502ea47c266f9b955cda903a690232c0d93f7a185d523d639d9e562bc50de248da30a80740453dd5e3f9fe7a509d18dadb5bd3517004f2a26be4d5141b5562dbf5ae747995dd0bb1b8d40ca952d982234f0032e2e88def826074a6287e08e750991bacb4aa1b460486adaddb6fcc7a8540e4ef0292c95e626bdc0a5bf86eb5cce68d860068e58a7c22ec2d32f85f8c714ba11dd6ea9f47d43d3aa242123bcdc46d9b9d0fc57d0f8cf55672200e5ec387d3ca1e9f113511fdcdafa212fd4ee3f5f44dd7d11005e238e7d9d2a4c1ced0311c2128017c26fc27aad36d0b309db1a4c48b48b7f10b8d2153d3c7fc4b73c75e9ca59cde72f4ee7116758b6a9836d2d0816ace1a89f8b53667b35c9f324b5f8089308c80862886efdecf3fa286818a134a0fbe53cb9ba8c09017b4ec02dd71cfdd2299c0a324a62e72b16918936d8e339ca4194ec5c8cd2ce76de83deef14710a564809514dc9cd44b1ff76ceb998f4aab474d33511dd5064446907f341e10571d51c133eab2fbe3fe60c5f996ef5ecae50752e4891c51437534efd8e9aef8acc538b85326e7b681026c381e7359f3bde0f5753030ad5ab49968dc4f7fbbde336234af76a5ce00da0c63255649d5f2e959fa9007664c16fac7eaf1888ecb0b545d9f38d188444ec2bb7765560057d4963ba13d57faf22ca5b1798361e3345f9ee3779fb397dce9ce444cbb591d36530563e557d8e0042a11b846445b98453134303423ecfb2f66744f8d153746be0994cd479a90de3466d23958cc9808caeea4e10598a33f338de22d8ba86a44c9ee95266a0546f75ca89bb7fee5495879273f34721b15d381d2d4ac03913985fd4b13ec3ee6a7bba850f32f713acfec828982e9e99289a81780287ca51dfa1be5acfec91e32ea4098ed6c015a79a0fa472aeed82edc7c0011c99e445b65e48a5bdb93060a9e467bfdda8f77400518dc3cbe321993bfd21b01ee0c3dbb09859a6065b760499ba206f22b145db9f7bf4bb3e7c26cd70b63117a3abd17a19c3e47c2c6dcbfd09a319efdd479afd107b4edc8f5fb4c0deab5c3e2dd58007e33cd01584dd4bf97ad7c66796b259ecd1fad542a3535f59645a75677600b0e98001c4ad0e64b4d87abe64d9534cf8a6033a813262039ee0d6f79eea7996298233c831af548f28c3e5bc18c9bd30095b2492fb7e1aa68fe6c1c610702ab8027d1bdff5e0b632aeab02cd2fd7e252134a62cfd5d1679ab182ca156cc86d97bb43ed0c758ef5fec1222843f758cf6eceab8703181aecc68a0982f34abb573c316f3094e10f966ad1687e408a698c1fb2b391b1057d57aca5db38b15413af96c1775819df3c87f7001e35339fb445e0cb92e104fb7fd0b33bc49e6786c31c87c21640b8a6f605c1defb212b8dd48abc7baf25db93b3229ae196945e59b3899af326f565f4714db36a8c2b1ee77395fcf415f6cfad462a63325528c36a708b630d08a2d20ab6e06c4ea9c2a32d6c899f263e9f245b3dd5d7af3aa6fcd3630d57e07afad35d2974eebcc2b4c1282f7900b775789102a53cd1a4bffb4c636bfce6beaa955bae35e16552d0a4114ef5126974ea2615aa9c478eb745edce9c9e57c680729424aeff3110dc54b374034cc2cccc8e7c07763e70dbb0794a1d39f19b27f1a3c47d08b7a9ae6b460969f513cdd8abdef1079176330b611eaef033b96c3d34e802b02acbb3125d8f9bc689c166bbd028c9b31a7491a5ca96abc932f98e6d4a16abd6e9ff9c30b4987afac0ce41728cfafaeb3d215ab92b7555680e41477cb9117418b7d95ecb7dae23a644b7103c082cc30c49bbb49206090c9de59d3fb0c9711cc6ff1b1a31b5f061c0a363bc432500191a5e11d09527f5d033ce76347e1bfdcebe5181078490a044769d7dab14869f84f925aa39e8a8ca1eb4a1e6a9f47d78d4b2b8951dc4859462d624944a1eac289f4cd17540fd4b4bd8548e51550005a8e76eb35c09b51afd17b3951a194eda57583aaf61a7c4c165cdb7c741b4f7e2430b7b9359aec89d50ba47d95d5fb3e87e8ec8c3dd23d334e290336a3acd1ca364918ccc88c636a4572ea77748a783432590e8b2982fb01688438d597623a2505e42e3372c3799d40c964f8e98a66de9aca4605b02f3695cfc993d15b2355dce3e01ce7710e5633cfb2b7069bdd01bf043cbb2994795288ec889adc0f7ddb7873cf47f583274cc29fdff2f595060400136d07da11a5bfb5df6c4e42f6f7bc7dfb45193582c93e4cee41a9ef8f4869bc62efd2a87308fc195b4059243fe0cadbb58ae1b0be14396c91e282330c50f4b60819551219cc876791c5fe1f3e144593d424fe209630f0b6ab60e9223f6818fd3309871d7db1c97187c21b4dd3a1fbf6958713ccab32901f9d770b20e51ca53012f60b9c3108158cce4559cb2d3eaa4261276375ef09d93da52c2e1833c1dab1065481bcda8ea70b0c4a103ecdc5ba13693f05e291ad11f0c79d48f5447f15814396adb4eb3ac3f437600ce35a88add2bf5a3fa86cadbe87e6aeed21b93dd08cb63b1267f049c915746f34be3b10c434b04642d809b15feb0e2d43a55574736a8e329222e57d9c7bae84830b8561318c82f7698abe152dbf5a689b2d1fa626f2ba7069e831d2ab09ee4904478a47a2eb6106ee8cd3a6db797918a5400f9cb6b07b41a18dc1b8238f9c12a9cb3cfe5a64d1dd0c8a97a74137be7f9c77e45c01e2597302cd8129ff83eaa8e97ae129698f25071146d80f1aa6607df9b8154aede7895ec22e5707b4f93f45fa41fe8b863b5b8e4483715dd896b7e589a1756da4d1877cfe3cf8ac46a6ba2494fd1561f87c4b55ccb1996e0d0e78b65eb0ece9b333a9d05effdf8d9bbb3765ce8c502c9fcf8ae2fa1241a51da172ff174c12e06edd326d16216a434f5855dc345a1dd0a7356a4409d1ec57dac27f5e046c2ea60033a6469ec4b1644189684101566c5ab62daf981f69ce9d126b84a726af196baf7544a038950d1369a6557c995acd57e3d176a1f860c9be5a732f8fe91d19b785d4dc431ab39de54cd4edba31220cc69d818517f25f1441d9d4bffd53af8c84feb074e06afbec2877a636521f8cbcd572724c78ab3ff56c320e0f9b1599e62f1dbd2efcda02d6474fc4ea0cf39df002b11108a9224f955140b246f1e52cd4aadecf5d0e0aaf1246747657982a272540bbc2fe8518c6ee6ae8aa0d4d7c986cf29cf97032fd3d86e70471b9bcf67a3e4b467727d59059ef95583898c330514e02ec2a042bda66e4f271a49252910f3510f7ac6b94a948dd6cc081649f425da57949c4edbe9369926e093978cce94d398f8b2029b6bda7d9f8d4df47bcd57709d502a6d357b2ac767dcc422a260f0477d0eadcab53eea802fdd5a624b18a6a652c4fe5431d7e4f95282f6fc2276e3ae823968ea0429f8f6fda7cb55d0ccf16902d08fd91e836715006ca112645994e81e4376634c3d8949d25f53affde76e8fdfd9fd61cc03afc9ef0020d5894882b5f4a60623e35efa9bb534580a67dc8f959a4c056955125395d4285136ab5d552c991500dc93836b57d399547dfc40c311bae3ba7357389692fe3e2887624ebbd809dd72276ace26f32cc7967386c7ee8f75aac8d61ad3c9fe68de33310f1aba0cb839ead1df49d8ce55f818ba4246fe4cdc55e19a6d7a0a0d885653b7fab1cc0f4795905fcfb5307a0d4bcf6cdc5525bc1400b6cf61a547d83fb58b0fffc236facef3844a19438ba52f4c118b7fb2ff42f405e8e95ab0543c6101788239a626d3ccdd8b3fa51c2f7391b25f0452387f1e3d2fa5eaf744bd2473d90990be116d63cf723f1e7184a11466c3e41dea9c99e19ab63bb0df4fffff31e0626a956d967b54be530df6cc0cfc30d475036189b8186818bb0252af1c5c68abea91555f0cf4ad69a4996dea325bcec096fe955652d029db8653fae6d21f3b60ef1ded6b9b8be13ebb618892c8fd4a09e8f4da32cb638ef0425b3f4bfeef23945401e4ac54084b7063fbd2df7280c73a1c2c4d4272467448bef9581a0a8bbc5e41c3d12377b279358a997e2d92468fab8d6e6237fec076727fbdfbaa8468a5c5c85bc3a67baa8d31ad6c92650e62118b144d1d11f801a59fe92149ab23df484d18c7520591f1156928fcb11bbfd7c3b5b0e3b9937a37395421ab90da88c8d336e70a93c3423f227b3249f004504b46ecebd46817fae1e67a7bd6977cbc942810c233574177af4293570e9308130289df41527d45c5757334b5c01b624bca2b736c06900523b8199ab603912029769bec7f1a6e1ac4005cb61c2643120219b01755aa409843ff9236f782c62568b68a43d66a02e55609abd413bc7deb0244e685347926b7b44f4277f41527245f237c0395f55983d296a5ad1d557cd61f1bcdcdceffeab01692ac9ede7aa6918830a0eea9b4e2f5a42f63a17bf65072e7fd477de199128a408d0c0390a378e7867950965f22633ae69a9f8e0f65599510ee79717f37d8ba5835d4eddf53269310dba5943e845d69688ed74ddb1d3f8f44051e4bf9d278471c967ff12acdf126afe414291024ef23b5bd54651598f7cc86e83e351f8a2c30c6619173c29cd2b87b83c0892e3fd8cc0f16eba442edafbcd121b1c4c333c71cc055a04c3c8db0c00b8c328142cfe0bfc25a74d296f11b834cfadc74e19ac4eaf84ebb2a2491d5dc21b42554e68331160ba5125266827883487fd78a9639831683f14c154c2e7a109ea90e782b6117fe47f5e5d93b4e76a4b32421d72947d20ab84df415ec30918899901b3d71b86159a7c1b99bca7e1290f0e9cf6214d1d64bd0bd38bf39b7d906ad30197634ecad9d9c91614ec1065f352ffdc470a68a8229b521ddcf3c2bfd6b96f47b2e08789fd033663d0c7f7710edfee200dbd7d8aed5c4f2b584f68763beeba297ebba96aad5573a2d832dd4f6768190d2e474631fc5a17caafd7d88c64c06e2dfe4105c0dcd6f50b642e08dd06cee731e56843f9df46df09b77b64edaaa3d886af79476a6982c7ae2533acb1224d8bfa73d48487e7eef52e2bcbb5d4fb5e28ebc1d3d913cc562f769be521e9a922bee5b942f2386edf0dd966c1989a42b9fc55c0ae13d7718e785f7e43eab4cded835c01381a0a7ef6c425180e415070910b96248dcc211989372c62c3ab8f92182009185f4a02f29c97d1728c3b2ef06b9bd3c6f1bc554474db6aa297ab0e4b4321db5d0b3237c491ff63bd8eff3535a8791f55003a65ab2e9eee2d9924d203200ba55321bc49a9f5ce2ab2430fe5db6310e91397228fc9044f259c7ba1fa9d9d7aefb8a2ead91d8dad5480037f8367dba650f5ac981b0c2387fe01e2a90525e28cf82720910e990c8be8b117741085d5aace01667068fe19322e1a5390b225bc385637bb00cefe31755355123176668eb58ac483eb40e6e0e720993deabdffe077b67a600e3a29b2f669740be97faa2b88323767f3b88fd6c3dfb3d1c5fae3bb2b8c415badebb066acfad434ca009b2200329accf1178a0be1135e3673e6b79ce48fbff00b4d70703dac4e268d3e255a0de5df8a50ae5a4ee2af6a28ab767dd28f1b7f4a43a1494b86f48cf1b09706064f2230088151cc72b4c79a7cace52fb5991a297259124df018dd3bf2e74f952f44485a573602b563194ec074a4c74275a126d8c3a0801e709f574da2d8d72a8a119f0dcc4435781b1792a2f5d0c0cf14ea7b21e8713506bf97a707d92ee4cc5ff2f9d2dcd1ea842e7102292fbb8fb173bcf4b8529076f3c1ac3bf6d5fd5cb36c08088018c0a3a27f6a4dd47092e1c92b24317747952e7eb8d06f21e53b976e758753352782600316364186337b3dcb1e1b36859a1ba2255d521b15b5b51c365a26d0a7a789277fa6325e78133a7d97900671227941f3e72f580b2a887f590cb9114c8daa13dcc216e8d407625872939c66de52124cc02a537779d1fc95d6bd478d559d35704f0601d04a1194746fb459c6ca17e0a854eb859464efed3da9810c7639954cdf164735a04a8c5a420f907a07ba23101de70fc296fce19154fed76835cba63fffe604cbd2bafdeccf7b0466c0ce9bce687bfb991130d7cc6aa85655251b2d372e8f5148519e92d5bc356cbaac4a10e43c0251f30b3f813bd8e128440dfbea72350a66a7d8fa7028270fe67c1db1696b320603486388c8540b64bbdb738d9e01d8df91fa9c6fbdbe1021e1ed4254adae1ace38a1f06d4ba48d0d56af370a2361daf250730d1ad80426d03c8ae7b8b2c9d5cce32a809c6223290cda414939fbf5fe6982d477246c202b453da30c41711cc197c312e08e2f0cd3c3ae312eda8ccf7a2d6d5bc7fb7266752825ce0d40429afe09586adaeb7a408c03ee8cd668089751954a51c374dbfd2b15177dd4d2a860a1cd21eb3fc767c4894ad69a8b2e9dfcab5b39f49dd5f7020b3661796be9fdb983b6101867b1b5e6904a8450c01445de3e867d688cff053ed8460195603f16f788abcd66125b634ff25f5ac42b9a850bd5d371951e623fa61c96acba2e852684477bf805ee8ec57ba3208c7bc3fd22b5f02cb90946597691f2fa0aebc1aa81e3f0bf80606387e29960d4ddca6766e3c14b7e67d5da356a4058bd4289136deb0e766c0952312360bd1231e8e7cf94244bc66764d3aa0b3fd388f0e60c0611e46fa8e5ef43eb98cf96b0b254a1f0556d1b42189f22820fb7b4af35b3c032390e2e81fb1f218885becbc6e2daed10355858e4fe8a1562bedce5cf87c819d42be0eee232a5a657ab98b5398d2eb9c8955ee3f53b30e9aeb941a2926002e20568184a788d57f42e46146628ab1163b00f2015665e40802f38c77cc76073116dd1b09df46b26ea0c3a699937eb07d96e2a9c075d34c41476b3f91ab9ef925f541c47dc2f97ed127855794223c3031450a45f0137d6697af5871120477563385b12f6f37816897431ceefb1c1ae35d716014e95e4b4446d64ced5f2672a01b9c972b588cb5a7c184f45b5fdb8ad96f576457cc559f123733628a5403bda5841ebcaa9077ba74c4eb71c15cbadc585ccc4527654e330c9b4cc9380eb79e91bd6859c78244361cf711f1be4defe6cd583d37cf03e90aa017a796a9d194c88f6fe84856ef0227bec70079f8f992595ba6ca4974ec3949bdc4d43be17c0ce3a2bbc1d579c22eabf6ab9a5a2882c3f0e63f0116cb5e5eb548108e4cb99eec4d2157ccbf0fb759418ae0ced8080dd10e138bbf7f1d20f3714e33aaa028ec251c96c7acffb26b5904b0be298837a347f389233e86504ca81dc1fa082845f04dbf973873dc42edaf7624a10930d0c71149866a3168ef22431fb36cdbfaf7f7c94aa6a89c4564c7f6bbac15d1532da872e75fe4ce5a3096a7f63bc0e428dad30f05aa174c3d5b0a25f8a6ea074b38ac00fd352995d12e1702bb7f5c37b172b75fcd737006f1f378637a61163a43e4e55376659cf6ae6e9dd99d542bae9f4285e5614fade864deca20ffe5a569c16fd95b5102dace88d829a3526be675ff8fc4bf5a34462e80ed5f4660a940be1de3d69fa5cd5ef050c61e9c91e7797cbfc1267f663a2f502320f8179914eb6643006b2db50ab773fb7642793f5606729cc5e69660b28c34483f4a1d3c040fafc7097ac255e5b5d1bc6f3c85f15756559288904ed00083401e92b2326d4afceda1c92345a0010717dfcdb320eb0fd319ff194f05b755a8b9bc75a63644cfdba75204fd0723fde1c01e5c55071a358b248321ae0ed73133a8939039dac744f88cd031c358d4b0e666d3caaf4cc2b1c54c72431285cdba38f13365167d13c4997203e08a6e4c4b8e8ce5bec513d69e497f19586db24c4f68afe9a85291cdab72a1332a4dbfe5608d2ba668c5e48b9ed215f338ed1d0e734aba0163220725fa87050f50a4a0c9e07900681e208498a8fb5159c1808bb97980bb8df49c1a629522f44780b5465059ed69e824e462f85abaae8dd9aa00d72917ba8676a31a4b4f09d2ea58fbb912f677ada191b63c0ef31fb76dcac04024ed030e0dcd9ca9d434f30550dfe83fe8aec21bf77005256c277f61d57bd08975e1ec4acf0f7c7260ab3d7851e5912d04871da1ba2cb2f66b9bea3429d7b1153bce449f9f7a684366f6f7d5614dc9677b60427b7cf0c082485a4405a9387af3899c7a892b8657c610393312475eeacf8b33376190a3bea3987eb6a514204358e61b3f782c25910b594a28d54771fcffae47533ff4d4426e96345592409c43a1fd16d918325e226c4aeb4e485ba686f0aed4d90195c9ea58e7cb6d2ecfe97fa52b9e695efe9e106508da7f3856f45320a6a95467784a7b1988b91b2530f839599678c442d3a63f8626cdd4290a4d8fb46031673e977b60c0d73a20782cf65d320285e838ed169c4fe7701e03d132dd0c0c77ba40ec199a1d7f02c98d514783815d1b12d655dbd3618977bc851dd3c9c30e82888b2a51c3ab730b635acaac35afb6e7090ceafe58f68ff6f5b8edfc94a4b580dcafa6c6abc1d52f95ec9e0762fc095d24cd181c9192493890783bfa75e06fc2214a41e5a543e634635ddcf69de034197c256778203d3f9091379b1dcff3edee161954d1ff55e1bfeb65693ad034810a25ba53f85438337d132e18241eff54d56d892df2a041ddd18194dd3933ae38ba25bfb6b1573a61d3daaa1ff1973b483e86038a2c5f940910457a412f6193c6e9eafc9282adf64fdc91f799f215373a6da7d1ed92aebd9a4bb9a9ae58c1336bcb3ab3c630b47a905f4297f759362a72cd8cb6c42b8aea5639dd97b2001a3167a24b2d5a61875bcb92d3a34d6fb5796c77923dedc2e1be0b793e50136eb2ac25c5505817a3417f1d1122efee23ed7abb9fa25ed1a0a9688984961877c402832fdd8d3a0aa1fb9fbe352e8066db05167a6d9840a867510a06d406e12a9f1dded2217bdd11f08745d8b21772df0ad38a663fce27803fd4f73c063603c8c38dcfe1d1d30d7a91309197c058dac712b1f03b8ed89c100400014b7e99ddabb49ce846db228e322d2276cd3ec590ad05b8e11689477a0380ca42ab8394b30ecb7eff6b9d531eaaabcfbeb1379ead3937bb8baea5a4f7cf259e19746be0ec4c3603fc3d7d49391c133c027ea0dc35660865621afbf147e08890ade269b10eba82d5774153321b1f8d3fd54569f482c96099bba7e2358859690f87cf90189b3df9c907ea2dc3892a994be09f53c1991d9cc8385e31c6af2cae9819029ef5b7a22ebe64be6f97a6cf61f1ac4d7d69945d0c58e8e4f0b869232e97f825d62a2c8178aedcbd533827281970155fceacf9660f5904868dad4171220236824e0c8b0d25c0453b3dd32c2545b3661a40437c7de08f824b14449c8cf456b81611b553f61fc1db45cfc0a18e30391f01f44333b02051cad14e7c51589803516ddb51994e6a99e9e8f50b2b6f26f7a8a8eea9a6fe09f1b7840cfc61fcf58ac06c6b61b5290f81e07f4b1d1a2414bbfbae2a9330906f26f102473f52fc72d54d0f639fc2463b4b0ff1e0ebca02500e12fc1935864d3b742becd82dc593273b5339639aaefffccd63f79f3feea822ac6ed5102fa11cc48ecdcbdbfcf29cd62dac2d2f7edc5e2e64decefd73e57ef2d790c86e1b0844e19e92bd34c9fdb6f0e3393a8a8efd31f1e853f97d59185c71665f31c0d474beb64adce9250b600f33488ace948c3e56fd1ade0a8c29fad1774d031e32c09b3d7ddfdc934ba8ed6e65282af66ae1eb18b2a6f8c5303bba74bccbef9df0846941b34a76acd57e538450dc36fcea23c3eae9dfcf61ad7a1b6e732ae3088c9cfae5fcea47524250da8e20c22a42c9bd0105ad933c6e39965a4b2d640b914ed92ced7cdea21f1ad722e14ca84af568f1d988327b395971e77eb8f4be11e5775707be6bb42c4c372bdfbeb38c857f33da29f7daafe1a818595d1f5383b3d6a32400213e0220bf6bf8fdb0069d70fe0c7f10c27fdd7eed9dd3b903f1f2adc888ebe9a1f0aacca6257dad0efc29530475f1b1a71c4b974518aae94c9f0b62bc0d16a3845f9b04d6b2111379f28d4c2e3ae8acb557fc0c65a8e13679d6e144e474c569a817cfc6d46593df5c483305c5cf436a8d108128aea3f6b2f164f93062bc5a76a5e839a24219e15a7042bc2bfe1d593f94507c45eb79fbf425b6c1001e2d203d5ffcbf237978f13c537a176ad7048f56158d67cb4dc753432d8f59d38a71a9349ca1db2a5b7e80e832ff506e7230ecc879c556392573479aac57ffd725557d948e3c27f53ad44a73be688d03da7a7fb1f246397406962071aaf67a8c01148dac7ec61d1dff3b82c1d2f8a7a37cfa3fc5a9dcd2cbaeb2c52ea1fd2be9a223610d814931312beb496a0dd1e857d8f7b687dfb2e906bc3fe6d4a165cd6e2d82f31c640f269cae63e4f071bd123fbfd0c4d7b8e4f5f5b5619aec6efd91bcc3c580e0373c54badb783df26e86dc1d3dedd47e3041b6dd30437f3ee31bf60d9a704b6b8065b45fabf2b694acd8c8fadf627318e813f792bdd1b4196539208a67f76ffda1a97dbf3acaae6edc22025a0577048dc0273e4be6fdcd763db16da16fb2130700380fe80784789b2d33ef94bc0dff2fd8ca7bb17c752721fad38cd8363848ab6ff3722ce1be434794683d0471742dc41aa0de8a6824c570404bcbc3517bc907fa558e0c98fb4bad5cd379a47644a5a62208da4de9d0ef7093ddeee74890ae616ca6ad9f3d5c521d28aaee5f925daa6bc808b0c0798dc27a0f03f376a7ce690498e70daa19e5df88ea3d4a88654d92cda829310f950e88dfea337f90784ed0da3db8a5446358d2de46e01a242b294678a69340c09da2df2c8a761706b3ca21ae235eab89e4c8c594ee098e652950e34ad261bb8ef480b46c66b40b3962cd5f6c4b6057bde5214fcc57693062049cbca802269d55fdcc79b5e8052e5e253d53b129c5a7d9de8a758dd07d24826209a91b4fc28952082754d199b6eeb20eb1f57688aaab7f2b1dd2986f02bc44e3b524d922f3d3e4f1250e63d873fa862ff2e6f0da72ab9d8a51407989b399c734010ec3f524f6d3f1e3b874b7a6677a3f4ec9dd4e7c343aff457f4f16b5c5220e8e45863e791114cd10ba3dbd8337fdb15b74846b362a8f55c7d45a7ca143c549d74888326310a6be3efd7959a2ba973a552873b101f02390933e3be319104c93261db9fad482a3b8ff172952e777615585e9df8d8e3e8acd8b451f0ce11e4ea6730426f85385fcf4ab1d2a3b22740a4a921a294c1e274cd3b9465beb937f4f217ee612848f506b67ca7251263671fffc2c2d778dfcde0632009e0710d24cffb1454c89222ebbb28d35728b63533ea4411fc956e8bb598864115835965a8456eb272515ab6c8c267c85a26851fa8218374b7d8026ce3d3235615d614e673c8c4c3e10a34511da47808d17c934c497b6849457fbc10e0f0b68fd6cc3993c24b4bd4c9a37ba3d93543483bf5026a34c2f6a4c17703682c7fefddcd2a6b6bb22afe65b25287a7ff08f1d16dada051eefc7b514ac9aba4d2bc84a95533490e46fba007e26dc17beabd73535ac9c89abcdc4886e5768066611988c980d187466b9a6817829bffb4b792cfd6663b3e2b64c6447b3fc37022e2fe82e9b4be5d44acf9e54654881894bd65c3419270739bffb3eec4a9acbd7a417ae84991f28148197e9abf476ddf37624135e05bfd5105dad92dc792bc5f6963dacd60f23b3f133398a1fd9a04b149fe024984df1aa50cf0adfcef6169653916b685b8181e8312d1b9c0438099677f7691c3ecdef72bbcbc4a3974f18798b62f09775a970f05055ca17150d03c28feb3cd82b506bca1022f6025771937e26758c58aaec1dfacb3279b904a93f6558cd615027696033059d67dc005c9745525ddd4e2eac06f1ea8528e7b3f2a26a892bd9847526e253701f61e79072e88e3b556b7df4217132aebb78bc894b774b20720f7eb444ff57a0298314791a1070944ffdf8dacb5eeff3161adfdcfe1ecc87105e4cfab9ed1a6afbee450b0c74a921d288a73de7341b3e50c62d15decee13836d2bba0848db028d39ad256b28e6913c89c435f493e36bc640f8859484393dbd69439711057657e688285a2ff1bda56a04a266d242115c74ecd36f1c1707dc1149f9f96339ddc6b04249bcc33740deb14c82ec1aaa76c28260437634535ab9027f52f478165863e3bb5beadf2b9bc37846783e89dd909af5244acb3554a063bf9636fba7c94088f8c9f5b8c21c12666b57d42f73bc99a48c4b21d385a31559df5a1690287b6f5cc0d560c7071ac5919b92c3b8cb65fec4f8e26ef4b1d351dcdf724ada3571189ce1673dae53e38a7e43a4c0d4a8ba570c5090834458925eacc932206fab3d786979893455f54588f0a1447362ea4d390eef67e800f892e069e675bb65856f6540d9d1278cfb5892575b4226a6cad5c2d0c746a273082bb381c3fabc509ebf7d0b0f9d6bf209f447a01ce0f002658436375dd65da440a1aee1722a42ce2d05d3558a87974326cb44190a2a2097e441c77e3657bde5a8c41c2a5a1f004aea5550e88d6b32227f94e8bc6fe001e7673b5dd65fde71d061778c0e725689747288faf387540e538f9b4f07032db971a66ac88dd887209d99e1f01238540cdfde56e4fb86177089ec1cd38bc4edb58b5ac154f1886d5a71a369cb6de8a9c8c7b1876c08386f53c3c74361b60ae495cbc8a01b574a9555e62f6ed4b3183f4cdbfd2de997a58d61d5e7a45837e6934791a2b90d1d81fa89e1d05ccb752a4ba9cdf26d6373e5282e595e7d78b4c84ff8bb39551e3eb1bf8ca9a9402403be2a377b03dc8474e467e9ec114ab116407e7ef75ad84bb65f1e0424f04aa701cef3a33412812d764a33ebd20afb18b3856b106f04885cd5986b9a793e794cd5765d8aa6153f0bf995e4e95c49af7258d2fc97ca6c8897ada7e4d9310c7cb026c61cd9cfcae2656a2ec525561281f250cc9ff674b35aea07210f30978d1d25ac7b0a6f87f8bf1c1d7ecd9024b7a57b7d446e5c6d4babba1c8971007e9d3c272098d02c6be262a286f5fbfeb5536bb8f85caa91a8696c9e0ffd6462c19fd89a5fe3bfbf783d646fe6dc597ba76ccbb697529315eefd191ec5ef8ad9c9b17db42b163b63243ce415ba84c48ed6fe26aa9c8a274a04ed281bb6eb97a86d3b05e772f8e4644f277b7ed0b349893b1753e5dcea2dc81d78b79523b3452b1dce440360b52b333a9b340a891edc8d6268204d12a96f66bcb0f389a36422172c044aa1cc077ef97215e81a0610c9b9fc3ee7c7a1c849ac2b76579a15ef5cbbe9af10e405620463e58abfbf6ad480d01db29cbc0dbb2a932580cec3057eb23f831c406991edbb9976992a4a237af97370f835807914272f4d29ca709fac11938bd97ac429979c47b1ecc1449e3c5835237bd86cbf15ce25f80f5ec788d6a439a41a67b23d753f4533dbb2fd719d4cac7912bb32179438dd66a87780fc5996c742f9d5103d0e0227c7dff6ea7a8d002bd15e280bd880f6bdddc33886cca5881c04d49142ca41644f5cd503b52a0e5fba81ade784dcc89b54dbd6e8add0bbb457b8d845b18efd4157ed466581088c462fbba71ab7040020f746812fd2e946629c7a31ca78cc1ccf39a8a7fcfa4b36817ee8c012e9fb2bf9afd5e275e07af8038ed38a87d415870b5db164234fa281cdbf9529672f3a3974ed653784ad5480432ac34b84b8e747c95e7ffdca6cb40d93a04e293407feab0a973b8c574c08b87313d2decbe388e7dddee4cca041637f07bf6d5307098e3b8fffd95bf8005ac6f6fa4d0f5b39596f5ca646df081ab8d3b8f888fe51d60ef9a0c8ac8bc91197fe462fc0c0f6395aad9ea23736b5b2e9597e4f718f7cfa4cba26be761867b6c66aea1c44aa5bd480238688b1e944e3a0784ed34df03a40fcec2f32bcfea36659dce6708801907fbd0f171c2f85fc548b1a861e3f53a061f2385f46e70ec1b316e68c24a9802cd631c7922f283b07901883ad97d83597b3cd849820ffad8dbd2c43c70dcb584d6a36d927cc88b4351daeb40a7722510c1e353b1729642da434e5a974811fd7aef093c1d43a7164d37475953ee3ce6ca583e8471a324d780b17984c0c3a2e7e6b8ac730de75be4f60dc9c055f3ea54afc7052500b4e2db0e7419269a70386d26c486dec7fcd3dd127489b54fe21477c1b6566b96147e8cd48ec5924c9fcf8f0414fd562c499e69594e59e1b4cb22fdc672caa51d3f7fccfcf3df5be77b734e663738a4e0df5badf6256f90220d578f02e591a552e832d1682db2a9e006ff79a23777baea2054bae25b5a02a9ad5df747ed49d86f5b5640e53eba6029fce5c2384db8dab7f80d9c4c72eb8d2589aad06556ce605a924076277303420dc851432c41bc535528b2b03dd78ba8b39a9f698ad5e06c695daca4411e6c7fd6b582d053902eea14c556f520eba0f5d9abe2a2253cac5dd3dd03fa6c916634a73da96f118a7510aa31fb5e0a113465beb261ec5a6edc467779c48f7ba66a6cec713bc8c3391aa8afec07cf443ac7f66ed1323bffe9854e667c82e7bb689ced20b3b64bad3bc228a264bcd861ffae838465f95971af4327596b25418244f502fb515392f1e15df92856d20bbe15f403464173cef790dd235db76e5a57f6cc61cadddd86d02da5494f56cdd4115387236ca4a6b550faa8437bac433b2c027c5544b2736b322ee3c3bed2db4088a0976391c046a31d6bca89b41ea4159dd859eca0615178ffaaa2f08a5c61cd39fe8a210408e67edf8db7017b94f5cd9864b1c20220378005c7a9dc55591a0c9ec88f46dc6255e63f1ea5d185c39d57d4c62103117fdc67bdbed4b3f62c5746c3c60e0772052178cd868e656cb55f88ed14acc8f912d526a705d456e9c86e1f3a0aaf00b6fbc1b34e76354de084e61b88e312c6719cfced9ae87c26838330d771b7d9bad6fad3127d05299a991ebabecfad7cf8feee5ceb13c1f5e65bda02b38d5825c295400d2be3080a7bdf9e408a69724d3dc1ac916e5a26a214c02ae9f875c01a1777d09debf4fd03b08d1b00453d9475fe967efbef0f62f1f7baed449821c449561dbc4b61b7b4730e9de277d68451f1f36592da47eb762a4e514f883b6bda9b3232cbb7aab276857c632fb20d9649cfaba797805e9d0ebdd70b1843d5b20f798ff385fb0aba45c7dbd1a48ab97d950eed27cb0c8228b275408320b94ed708917bc5da92467a363a65ee02cf78b99311a4faca5870012910e50dbf0e289c1a63a46f0a397034d1ce6a0980b660ece22ce6015682db12108f930d5d578b772c8ab3660e1d5a103ae61f4d18fdd25b8c4ac848434f1e6f3c60814823db536c9cf67ec369637ec5a7f097b946a35c00883d17e02ba01343beffdad5c7b650c11226255a50cfa93f18914e4ea93206ffcaf1f9b4c09cd67d2add35d8655710061efd9584cab3eda2edbca0359dc603a027116eaa7cbe1ebdb1f31224f4db89bb68218d206b40fbd1212f2e07bb344dab00ed690a04e48e8136eed831cdee36e1a32546d495565ea3d5cc27d898c18dadb31419de65b665044cfcbbb9632987518de42f57cd1355afc7d7dca0676bc3b3da221e8236d01582a2fb81f70135d06a0216ee34348cad82dc7521aff237fb6b1fa4c38c7381e512f06d45c2c2b3650f41bf8144faedc71a839f764bbf4cf3ba411225819db65016c9f7970ebf96ea6e06648072e058a2aa464be9ab4731f4e1450488fe6e5ef190ae7c166711ce93fee7481bd90ffc0f0f9a45edfa7b3bf9fca46f15ba40223b9b50054f10a70dd6f9c359909e3ae401104b7b4f8383377dbbbe19a47862e75a7f012c6e68483b46faef4e2db7921285a3d89310e6d4c820443958fbba26900e425a8abbe31479bf66d44bc85bedd3a6ff5bf9e350b03efaebaa3ea6a1c8b35f4c7741ed491fc4d6fb9caca3d314bfda5886aa737807b60213d126518312516656a4c0f1f14535198ded0b4f97bb70db5168a316319481c6660268e2f398ab9e7d6bea2c66fae5f3d39def32805c2e28694c20c504a2f4e4d2cd3415d7d3d00488452a7f5d1cc78466009518ca640e497e179a1381e1882117d8bca88edf38d88ed4a6c4677c250e39a187edd07c81305b7c99545fba6eeac78f5215ab9e39e525adba2dce1de6ca41f30f128cd0302c9ae04a01fce2775683bd8a51835bfdcd9d0ae3442ac3ed4b37b03077167291e0d8e0446bccc8ce6a390f0a326930bfa333bf2cc5f3f008128a3935d6bc14768fc9864a6d3814afd530d5ebf8f5af48399b0a14fd21b3cdb6df1badfb7fc31aef4ae6672dadbd8c3c784236d67387a837463cb9fa9d55910e6f31f258ca3aca165ffe0f4f9fd9e8640f4085f017d027d8466e782645c14216020c6c00a37779422a016cee77eef8deb9b0374c2e7e460b9a92ea96177daefb2a8e9b06df8519989386374e6b48dbf571ec704c1e7f42bcb13c6e984a46d3ae5a6f1b573dd17703c878f9000d08463148861bc71701411325461a06979f08026d39e23fe2881d6ab37a478bde799bf4c4b957b1d4fc66ef48a87c0d16a30c3add814e385cd2da5f522d307ab528be54a0be92c1e5b3e9a71986dce510e0f223a7ff9155ccfe4e7171994bd3e3f56396a77b270977fc0ded7a62d2ce8e597dc51acb1ba88ed7243d0ed230fb32040a68a06fa9e313659be5409d4172fbf201d507adfb9d33770fc5e6ee89e36de0e42b48caa4446c7aa4b25f977d12345c24f60bf36686b5c1c0dcfd043a0af0a0e56203d8cb32eb436d83288df50192a2226df8e03cad41d699d1cbc33126a6498eed5f302c57ad956559f9488452578514b095938eda50b3cd1399ea1fc871c8043e27ab0bb0ff3593224c8902e45970dc15d7b1c4606a472e26c912f0a7675eff953eb5fbcddfb7c39eb88754facbb6ea53a83b28a065d52093bf655b6c5b78fe150641e3bb794c2bc72972073d37a836a20008dda608da79c482b4a6266eccad11819e2483593e30d206197956de7ad247b402a103ed369d465058e52c8ad8d3cb22f646b4321e3b4c36dbcdc463a5121258af6efd8ec4b944e599812a6272772e2eb9beb9b1a1e34f7f0a5cac7a6d1d71a2a10beb6c5f5d16a1dca2b2ee99711457d67fcb6b2248ecc104cbd64f2888a9cc3a6ed15d058fc7981a378f194088a742b6a78319bcebea4b2443a6e535a24d76b0d6ea7571dc112b8a19f89ce0cbedd8ab24784a6dc97c77276d1c8b42170a343e0b930f837002eb80188f44fb342699eec56bc11442557216c42f70f8d62f2044ad314f999cb0f1858a826288399246b6a33b3535f1c4bdfded49a55a71e5a4d8d4f6acd1031431d3a314118554749c8ecd5b07df548410c76264c493f7bf008a76c540f74b0636c10c146c4aab6ead1937201b775d8cb3a9a8dc726c9588a7c399f528c274d258a826aac42540f3b937a119ae74431dc103019a6d0d7e339d1916beb5531c83ea3b4c536c4c867293fcb470ccadbcf32fcf25c31bbf074e2ee893a1615574a01aba55e87eee81c249641fe3f3f918583e31c1db63192400adda54572f648ec0c884e4314d9ced215f446e718f666f46669d62b5fe33bc135c5e0091f5a3c21e4cc1e36a0d0f6463d7b87da3969d0dd84227970571b3331775c4d3379aedf6605f8ba2372dd033a48bce66740badee29e77740e60734eb45e1755c8c2a6b7cf2a6218116cfc8b2413c3f23ea13d804f1ae27366cb4c001769fb026b44190ba20065eb090da2c0d01f4dc4300f0c2755b55121574c0c9862f5a1d98be3e881e655b3995b954dd191528effb5e918105d31bee33067c5ce9c13ef89df74c796c2601f6ba3ba3ae8aa21b97d585d892334148eba6d61878e0f1f1cd03e0a17d6a00b8335eb8b8a7f177d0ba1dd1cb28298bff1c1c167a645a7b6f9f0a7cecdabffc436c254d69dc9cf5a399dbedb6da43ff2dfe7934131420ced8a902e2f50eaca1a4d811f4cf39c3e5307c361434690511516a3ebc0c9bf164a46b46da825d5e1091e9f4db148dce997aacfdbd76d127dc342460c053581f9e03036eb205398fe233a5a3509e7bafae355934681efa0673cc8ce3fddab3b0f6780598f8d95fda361d926cbab0a236c70e9d27b8452d56fb588fb9e4679a49a7987045221520cf6b922eac6454a02050eed7d92b33b42a6af4ac3a91459d8e5e505edbc14cb66336388521e4362ba2b5469c49526e5c61a00cb12fe72da6a894b599836b1a397b784f2641f5b13e9509cd1ffa641bd9ca589c397341d2b416bc323c9ba57a24b60dba7d1c1ae9bf1872bfac679004f53f34619227cd1cf3936937aa3c0f28a64784a2a5465748df659d565a0cb91637a0839c0a05e600a9ec3f0d3fa2e98bec1977880df5221a3d72a66054ec69079ab18fa6d62c17b10943840ffd4bc6527de4b620e53b16656cc3208a7705ed6348839fc1f6d698ef085911f6a8d841f104a8c057c948f350ed2bd18f3b6973c8be9c23004a9e89507010375db7fe26afc62429c24d7fadd9502429be436a86fd4183d7d7901bd320427b61a6eed5c050c15dad7d37320adbf12f7cf20a152271f103401e0aa51271598c6fd3448eb7b08b63b8aedac44cea7880ba6971e5c66bc5f5e8ea88e836b93fa80783466d66ef4aa3edbc97d48d108d96a6e1c7418c7373a95bb7b36f2cd3ca42a2639220fd32572e235537fde1446a631e33c045b5f949200f131b70b9a941b54a98559230f6242495b4f24d9e440c844c0c6fdaf95fc16eb8e9622e8e1137b8b6a8c528bb04b10fceaa7b00670ed90241f5b5c602994a3a2b9879cafdf5ee87dc2774d89839977cf1069883b073a287560a1222df1baa7cdad1161489c24af2e0995492c531027f4e268201c32817aaa43265450b0bc97b3e9e23688580ddb0fc3138590c31c04d8e88b5f0ec7ebddcf0b224f5009ba29fd615f514c97c06b731ed51173738496e6dab69f5719ed81f62a0e33ff13d2944319592dd95eaafc552833a7cc84b5adbd12558f71954e7b14fd8f622e63b34622cadf8d0a00148aa2b10178abe0cbac62a9b3f6f77b08654882f0ecd21dca6524d4645647364f4dd4bdfaf6b7e816f20e1f40abf854b9c36794fb562c3d8ab48c1fa6b59fed96ffd86ed7c4508893956dbe7deb1b238a15fd7783e887676829135bfdde7c078d1170d10f4b6e32635a5e1a152154001ab1722a40e8cf53644d341e27ff42cfe50268ede4217436b1d53679a505b4b6cb8939d8dae86b7a593dfdb109518d83d15d42543424c50af22e96da17c31a6db2abddc8eadee048072132fe3931e8f097a4a95268ef6ae4b09a7eb82191ca5d23455dfac2c8fe9fd8586f5aa43b7db444b1c80ee8883c91a41f694a28d008bf6e8144e939a7a7b134b08cede2cc1c6e6ba10fc5100b5de37758ed50aceaadf7abff9f566fe658105a6914a42bedf80119a1ed6ce59f9d91a4a2096cec6e2e5dd645bb849c9cb7366ea5981e2469472b771ad33e0d6be24791042f4acaff69855c9a04798f9fbe3a8c18e0ba53fa73fa56594c2e319053d0bf20d0ba9452ab1d58ffaa1a0249ff1ce5e96e8e2df5fc69b4a40250079f3b6fd9b3a54934419811d1172e35a2ce25389d94a80502a9c6d6809e58e8b29853b39db6f54f8cc20f687efd1e49f71d88ff2bf9b1766276730cd298162e189fa7066593ba895ba3629f39ca6623982b1b9949a96155563c9f7f21f85fdf52126786dc4493daa4452b6257bb72ab1e1a6f2fecf44e7f3794a885ffcd9e86c5a639aa7d97c6ff902a77f60c1a19f497e3172ed6ee3ac683e0b78bb8a82da9da1a63a6da76f3ff42fece69a5e436580ec84b683a8145fb95ae0c33b5b384879d40635fdeb3696399b7e156d9f297e1b619b62844eaf56d9e52a8131f65ae6cd8259edd51b533bc06a6ed0babfce4048a8a7470b21185af25387716a7038c9de73c79a3f291ece0380b819fc379f6dec9b3915ff8d366bf8c88f374126f46714a86312633fb26e47407ab107a1b7d7ccffac6b7eae844e4b8354142ad9caf79c44f923a5653e0bc87871605a0fdf4dd82e6f99d065519094b78cd6594e4ec22c65084fbeaf04d953006cc5428b5d3663f608d185e733ec8040b765d254c30df121efa7428bb586ea38deaee0f6cc6a7326825392ea9de1548c7b98ccbcbbe752f9b90bb6019021524d2b4a3a027093874d6833b8668b435242a1a5cad845bba694d7368993b5e1ed3ec9f33446dba733ab1151cfb39726e12030c0911efa890d53a4ea20cfa2c61d208604fc3cbee10a47c3c0e786bf7898271916dcdab4e6f3501105859919d91038c09fcf6f23992dd7e800d3dba6739ac99c76ae532beea33f4b5efef6ba62d3fe1dd9165729d083ebdc88c266afe031f53c7d756eaf599df5433b861a72e41f5d93ec416f9f525b81d3cfc27226bea3d5bdbc139980e74f2c1fb656be30049a8bcf3f0ea112593d804201f0eea9e6fdf23375ed70ec12e418abd68b1a4d13a7110a77545a277687c7c4e9f142d662cd001264984bd09d703e31f85e778c146fa4d87956f5550d335fccc8b035b4ba1df61ac8e81cc30cebfed687b2d41daccf2e17f534c6b41e3af78e9d0b482c5b6ae701f484b3962e810837575ebaa09e540c3861456941147428c971baf155df24fe1518288703ee4b355ddfdfe1ea84f1dfb4127180ef606df95bbe427a3b130eda1ca0c4b29cc7f5e1b6919a04fa65adfd3af8f0d4a464915b5d6bb2b35ee30b7445322f7b78e7a2e0adbb38c00ba720183cd6128739d5a1cdeffc34f09ee97f27a038d69a36ebdda2e87f785146b5448a66fd5b64366f47b26223a654a5015b4f8e43d36acfba42219596a7981a3302dae58b89ca5aa7f3d76aa68ab3e7e905ad3ac875836b1d99a021fecaa67612dad18de9537ab2f442448a964d7908b51256505983b0c30ac9fe7f001bda5e161d14fbc9abdac8eb2ddfec775b79f7eb2a84586e4e21d16ff9fa335985f99dd3fa43fd63a0aa762db6c4e7039df93e34abea5f4c1f0b7c6cf9383901941c21cec44999722ced8a32fb8f3d64f9240859adaea05cb66b63c250c70365ca1f87348b44cfbb648c7842d57b0f768f8cf65a6c01d97c5aa192404b2e329689accf897dba723cb186ea60b2b9603b27d9ac67d65f57bd769df1884653368c3962f74bdf8aba50408222654d3ce980a9107bb982665a12737353736d8f67f55af7e1d192d955dc1e191ae3505ff15d86b3f2fb727bce29f85d73a2e35a91ff4c6941eb9d386eb68b175cb44cba061a7a2fb43ad0893c25a543b929283377600c34aa93d0f8b58e447ad60a9fc531e514cf5f1938603d72e130af04413075139965a8a75b02f510cf68807ac9c8a4e19d13d496092e8645ccce9f87e0201239a5798a157d7502fbabfc2e992fe564cf49746195269ba305648a30f413c86060047f363884a6ac1855e9558faf9421ad9d31368bc2d6229f4845b512ba671a064476310c8e450a24d74a2f3d8d648a1aa1a44cf3f05e54229baae57fe6db5ae27294b996584dabbbf02a0b2c1a1842f433f34fa57d9cfac7e8bd1a17e5d1c0daa15ebb2136e71e2ba95f2357569aaea5677ff9d5bd002c9bdc0fd0e42d261411f95caa7047a098e7eca9f1cbb27e980947d45e94209c2a76363d4be31b5a38368c3fd075da56c19e11c4fc43271ae0e772f97070be5f7e976ea83c424a65a6ae0c5e5bfd4fa31b34e8b61a9640d354683bbb9af9d5b471878c747367dbd586806d512ff49d79c7f7c2c47f8720ac240346789da7ebd239301ab19496c54f3087357b28961fcfda6d63071a5566f10ba2033f3015bee1cc6004f24462d3b19321178f96485006343b78a373a5d142c0db6870b5da58d57b4ccc68d78a4bb5d2f2223292fb38798dc719f8fd7b8642d33fa2f69def7c27a8e0bbbd4fe7f598fb37dd19be1da07bcde3af3334b783380f2d0401ab6d281e00618c22a199b76a1f8f3a3086a4da9941fe3a5177a9a8589f12f1231d7a6c06213c9ac1b8609a6ca2a12e81b9bf4cc3e835645686aae1ec6e52f517106e21fac9c924b3743280a298dde50762e3d00a2f914a5a8c29573148dd74cad804684251d69c1985595782099164c5043b2b915f6228e9696d49a17aa9f8475d2c4e4c18553f19629e209d044ae813b175698896e65185273f8ff4869401607a7994a166e0a9e6e0a18b21ef0a616b3427aa4cc37bf1ec15e2f37468d264dd0d4ae4ed09b6672c558b5238b4317dee534cf0d677ef7eb1462d4044e1b52e928c3bf0c4ff34093230cb247101d967c5b5622f70091e1448ef00b55b8e5a6e39463300a3da5a54a7ed24b3625601700053d89b4550b99e3d20aa6ec1a6b6adf448abc8bb60b518fa7f2d7a092844c2f8692f8eb26ba3434e39392bf4ba0be7376b191f5f603d04e6b8b2002d172fa5b780486213faa2e7e70349723216079d77149dffb6000265a052dfcb7acb04610cfe473e1b4511808330c97aefdff9da4d4f7d4ff9f1d693ee54c7bb1f014d45f19d47c147055857e9257c87441a994a311a17b2b59044d703555dd5fae38472439113d133b86aa98ab94307b1e4959a30c64fcd14656838d4b8a8ed26cb92f2fe6c4597412eff8cc2367f9b72ef57ecf37b1bf495eb79bd304e91ff9beee48dd869da0d1731aa1a54db9b12c6e6e3a09d18f51118f03ad80801db434bad269463d0250f334d4287e4ae81ec236012e688ba07046c75aca165c8c235bd14f023c9b042fe59b58b29a6a877c9c745d1f3cc59bbc308ba13957c319a57474408aeab4a95ff5b6ceffe7fdcfba140323aebfaa646e34214ded67839f582a19f5da1f9dd30f24966d61c35373fd87396f95887d6a8ea4d0362d2c29cac3ff32fb6574ebc4a7bb69e4df9d2b70e865760b3f787ff49d66ca21460f88c6e4589f6f68025af866bcf529de0f76875adec03a1968cc6d23b2de88ebb2cab315ef98e6a670a2df297cca4010e219112f20335b665d2fcfde6083a6d3e2f9eddeedf59b9ad32e88c789813cf070712201c76d344de1bf7b2f534171a9e4fb7d540a920feb92e083ca59b8540e8c5080ecb468f3e170ef1ad83268d65224570f5c72b5598ae17c8311669986223cae4fa80f58fb86f598887cf2b49d6223d31da5118c9a9882ed4bd055ba24474fd76a57c4250e339a193e47bf7fa3c7af26a03dac3ea2918c08c63bb60a6c74890ea333353d21b219996e42f1ada19c7b6355e4ffb88a6711b8f4964146a0f4c048d368566ab3bc43eb6d4ff9a7961e979feee2f7c97be11812cbefcb85c38ec2f596d24e2924e53bf883446f72a9d48315dfb625b890e4b572491cfb855b8ea7b5b3949dac95c10bdfdc8572ec9819101efd6f00ea0bcc226c919c09f984ee1ca0c42f7fabf7d265ee5070402ead84cb1962cc3c3331e087c1868d6e3b7edce6870af6781db33f5142857fed559198d9c84c487d8f272d0345bbbbefd75a0667f8509c926048ea170b33bb844eef88981162dd67f01c6c947a122dfd76cd78598209317e7ecae8dddbf045f30d26ead1a9f3d3e2b6482fba661c1f7e1f819163a34d5901e448e835b4fa9e967e93e338002e2d57853d587bb1c1e8f82ea6d3574d7cc5799359a63e60ad99d12c99741011eb6181762e1872ce205bb134b5db78a5aeb42231e86792ce05d1cf2b6729a75d99655ef0662a8b609d09c80a0acd807e83840aca7a689aebfe2a163429992b7c498edd7b922fd714645f7466d1a1ef0e9b1cfa87e73e9e85d610ca9176df618a5166527fb0501f200c21f21903e00231846c006a1b27df9d198a375059f473aeecb6f0d208c02b1b3475cf87ae81606a74f85fb307ab0a080c205f076027d3983dbc48912c518eb5ce610e0acca4b960a2e6bf9fdc89d81578512fff782f703bb6400523fd5e2ccca223e6822263b904752e36505172074e83999d72d0bd1ffed839b123ab99901755c47046381febcee69f6639783a56222f8c436d54cf5de34b2deec8a6157df47a6a46ee1c3e1b81d25f93dd6a5e4165cf79e04d4a809ae5560ba552632156569f8965a89c0557a2d4914b038a4816dc047a5d5c67cb91f8bcde915ef79e9163a091cd16baed192171dfd6789ceadc869242acfe6d7d5fa6443c1a513407232faf0f405f5d53d4e9ca3e9e39b97c42da2da544adf3a724b6c26d40a56f19a595a2dac0832ea9699d2a3edb289be3371856247082207a15f1938cbf1d182e62b73e1459fe4348f74b10c7ac231ae154e738898f3558c9aefa06fe6a576bc3e845fe99ffca536cfcb6aa176a0a9d01f8d37db5bbe8d05afd56b51e4b7e9ec60e8c544a687dc771a828a6dee9c91a17cb99143a02aa629237c381d61a396a67726c1e07dfdb93e85bff1a28bb7589fc5f56516bcf4cc80a80c0c426752dbb6ce5b00d9db389ac56ec6b3bf25248f7e6849766099db6cc2dd9625a64c2ce199883fc270af446a74d85deffcb5fb4742b8f9729c580fcef12d1d9dfb5cc9c3e2876aeb04491803732dadbb56ed6607b4cf66e71e2f181d6e1d4d512faaeb22539d772eae300d82e3529f3ddee0c9f1dcef8d62b01ee97bec35a6fdf1cfdc6329e7513c26cfb0d1c19ca5492ecdbab9e1ba548c0d6745a89a404e9e58576fa6a44a66802502e247986ca177cc1c28d89cbb9c66773b1812f191e67d4f47bfbe33f3117f1940d817a2bc18926bf3b35c67440ae9c5bcbe1f269d75bde9ff74844778512048985790cc5af12b6f3221c128d3cfe13c4f986c1184027f1e996215504f581763ba2a050e812371f365ffc77a081b9dfacb3b542c8795045c2bcb25949ce8ac0f77ee9d3a004584f2e06c2926736076fc802b9f7228ff8e11d99e63d26ac65d472ee6058b9a2aba17abb363f2702b4afbabaac5eb5600bde616f5099a6da63bc38474911910b4882eff1d2421c2e83c899f2039f2d1c46c90d57109fc8a366191b7d2c4735309225f530e780b5d5c27f362a79cc55e03745589074cbbb45cb2d5753a0de2414f0573e39ee78780c532a3dc0fa90807c783d8e980b3b7395c1f67f4ecfc808fe3ac67582a133ee4f911b657a82727cd3bf5bef0e5378972e299ad025694970ec04d073edac1a6824aada1b8cb3997a3d058aa1feda430bba34db9a43f4a6dc349704cb0e8e71336587dedf92704e89b58158759ae279f98086c7dc9eee56cdcad34718a8ee994c346db8f486306b1787ab7e25bac5bb679a0a96b9a6503580d83221683b148fcf1528c6231b1e7858a87536f62843022dadb42a0c665915ecb7f49e15dddd28af13899bafa388d30be46c34a5e45dd7ce47a9c9380d6277d1228bd02a28d30b2610266c7372f90169a0910a0aef0833198316af264e3b36fab37935c830bd7ca4b0ea8394d69948d85e3ff597f64512486736552e2f934d321cb764096b7d36adabf6888539b954e94310de27db4c02546708b770c2c73cb514dcbb20156e236d0e803f663798a3553bf4ee23028e3ee8d0e9521df0ad6feffaee374bd1a0973a2b83aaf99981195dab2e883fbfdb5e7d104ddd88bb3fec76da900076b22148676e8a98cc06f7bf874e9b81f3a03b2f85290d6e88e49a33870cbf739f043614e3c54806d4e6d5aca352d36b876a87c9fe83b6b91a587d2cd8ef0645d33211006d0944b5d92f84861a8560de3af52a3170cc1dc96305bdf362433f66d467f717c4c64e4a3bd59b12153bffad874be48914b454fa78107a3ebb4ea672eacb47394cea0882c6829a13b35ec3e00bfe648b971f76c943e5ca889f22ff7382e1287b76c01cb16fa797221552cd7744eb34d711623162f60272c521c22b41906d7607f9f0429f8e8d2cd4848007d2bb1b219f1009e69e4d176dc2b9b8a6f5928e33a651a3208c08107e882485c21eaaad0be8e69b1cb838f644d3465006baa82797f0975c2f96296d5fe5d48344003056700b72c23356633d827871fb2edfe24eecfc6efd436497daa35ef3aff7e011135a59224ab86b65fa4906cc314ce5ea29d3ae16a0b7aaed2647809e9e22ba291fd6de15fa5ad8fcde49faab4d4f69454acfa65e60a4adb7918f9fafa6f3fde453044097c90b2ec2d404237a25832272b111efaea786ca85a854db28156a69fffd4329e0c264136a4ca6e7a9703cc75b0ad3692838331bf1308b2dbb1acd9a7a8eef8e5a8c007f3b20c06c563c0524b76b2ff446d8280f9fb3ac32edcc28b65371b1f847cb1aea8ec3d8444cc34a0e4dae0a40c729592bbd7d76f01b1cd4662191aac45ac3146b10b132bdf3a31ae69d28c91205b07621dcf832cda90c9497cf733f1a938279c7dd00a67716486c65dea8a794ddc4e8f0b8b6b396e83b8c1a462e68277d648f5f4a848be65e956b05d54eec6b1f3a10e7ef74e51157dd1f784753563a6c1827ba1f03930de9ff785b0f0856f482eb2e4eb7c562d6e0f0d86c4bebb50a07abfd1d1af148dcb8b9f4ee90ecb51f9136147a3a0321035f99da55205191a2a7397d12ce9f851349962a02314294eca285918eed79922ab45c3af194b42e892ed508a33c0ddf6f2404d4dbc0061acd18b0178b5dbf07036c6b1129df2af62422ca9a611c1328e241d3a3410067d4a39ce263096e0cb3e967f021c281a1fd6128c6dd6ae16899055e060d5ee843e865d8ab738be6c952b8651dd30695899860d54cc8e472744d0802cdabbe323ea30ca7916c64ccb81fd69fd016033500fdc9125684eb1c4646c425c589eacc8d0601510b695a1dfb1bcfce54b3c1714fda567abc2012c240cf39c9837315d024009b462e9f05d232c771aa70f8e4446c9af6f5c5d6445ed656569c74652422d5071b3373991e8bac531cb67d41b63294680d8f0d6766519399090e2547741edb4aac09df3d0485d46bf9d80cb3a053e828342b14091391ab5a1b886722dbdaffa344b354f3e2424bca3728ce1228c552cc54ed0d6066046f8d93d2584cf16832382a1867e720fddbc7404e876352f7a17bdc55c1d6f0a2ede8e077acdb6b7391f5b71278f39c9718dd75d2a44370c9ace75760be6561a42ddc25d41f03fd58b244fbf7f8dbb5bdf8a59327a30b42ac46a893d6813f347ecd891a3baeae812459a9ccc791eec9f92a391f9c76256afd2ac52dc933d1eb33d860f620df2f58a0d525f6b453839ac4b27e643612134319fe2d9e55ce2387cfbe68f8798b3b603f5135fa71e356acf947bf0b7a6f76b403be0d4fdcef1bf2866b82a713212f9fb8f8fe306cca0a5e2de82d6640c3427177ec17b5fd6bca1b6c2a7d589d8398aaea81fb5d927b606495e9f43e879224eba7dbeb01edc0380512f9a08f6fc52f33562c7aa0762ed952f36c3c5bc041ebdfa7a5b8e765b5a24017069217d225a4521323e7586c9ea1779473001400acb96028fdb6ebb770f85a86a3ef053f1eecfedbf7f14a6aad29d769aafec8cd2f3906dc632e625b39bb80a853672ad557e1d2c4b55f5d8a7990c8ea32e4324602fec42f2911f92bfd56290e66577bdfa2457a051be972f6e50f23bbef99ffd2739f8ecd678a7ceb9d48e93c757598911498d4b9f52b33509efd5da2cf177f584f5501e2b9a0dd860233e52d5df435da2011acd97406ee77e2b6ab23fee7010855f0cbe475e6ac09dd5916496e46da0255ee5c59f1ebc6dbc782efe70eaed9d21eb83ab9fe697c7002ce3cc630634e868bf09917ee52c95a7e5679ddf06e85e6e02396df204dea4ac8f5155a63741592c2b77af64ea851b7703a00b952c579b498b66ba423e6275af6d5869784b456f02e0a329e035935a23dc95e3faa18669b89815d6c44d83b3b2268c3057f41bb3cb904bad9c7d3826800e8777417cbf067b1924060bdafa4f355987d1248a40c40cf9a0e7d88bb7d66b3f7c627dba0bfd82f3d684af195e8a63740e80d3a0dde10cc449f2add66cdfa11e3b566792fc4feda7269d618353b380cefbd8e5c094b617898c6392f15837bd9c32443452a4fd194421866392c471799600d3e431520cb2d09b3892b6b19b1158faa55cd8897a6fbd5e3cd3e11ccd004711bc7bdfade5f56a7bd280da9d9e416be2d4533d1351720a063759115fcebf97ebb4dcf250addf285d9ca158669f475b9b22fb3520d0f7bcdca8fd4a60f4a7088c7d5ac945fcb756952f12ed391b97243228a7efc00be3909f5b09a69f4e92f3d218942baa7eee0561035350727c751b2f4c8a3b18aba3423975c1c8cea94933d2c2c05bcac0218fd5fe14005d60a9ea333bdde82b21778439ecd207af8a9de86386292d04e7caeeb58f9e7eb43f4be6db2bcb9eb7fa88bc87bcc2e890428f5187162e4331fab2f7007d349d8aa2d34ac4682c07a5d72ba8c398d36d962d50f8cc6be82b50a0778600435d2bf3ffcda1a46a8a8f91c246af2352e11238caddb36f336f110a4b39960c6ae1cde17847d7e1b9a3cc25512fb244a74acb1ad68d1045d3bf412f03b5c76e2ddc50d1130c05daae9ac2566b9334388af3071edeb3670dd2c0072a29e80f1c6ab635a8bca89519d0de761c8ac8a9b35c98fa87309c7e1654b6e131274f61043c44dff82df4e0ae11f4d0d6c497a770c3305e6649749dc85320e945359d8122776b3462af19498b0a3c2704aee8a1d21835cc2fa4c5807bc35cf0828683c904c6bc1c06e3825613c9d9b3b54eca7900f40b788a2e627b3fcfc3013de703664ff78716276b80f08f010fd7247564e748b5a2f307c30ce96bc3242648ee7c9ebbb71fd0c5396df826927a40f94d3f5e441f97b9472382d4cb9d40ef53e25a0c7581229dfa8c9b7b8e0adbeb7e2bbc3b1956ad684e0f74db1cda48006cc5a5af942c9c4cab449a107614ee6d580de60006f9074f5f5ebf093c73db79fb4368d2abc85f6e9c652a8997d223f1502f85c92d7c8a31afce8c2196125160762f9ae38610ccf446c85f04d29d4ef69e3d3e6c76660a34b705c5eed29b89e2b49310272565218eba3dd209411c75d25393da24a6d732a6fc7d306934f207229c3514a3fc9127f9950a926eae843a85e94061b5999099df3d337d327f77d2a260d8d0a486bb46ff2a5157362255adc9cf5d9f562f45e0a04581dbe711eb9a7fa430d6ab4275ca27f07a707a943a03f04d20970208e16d38ca56975a0e363a1b7917105ba62e0e3452ad59cf7f2fc4e3cdb5c528c458c5a07bf70a36aac9b8c74976d145551c798483769c5767e9a0570b56dc93e7bc99003b3098977ead9fb326796a4286136e691df520283aed448a592bf9bc60dee0e84944f24f3d9565a2363e973871f30365a685209204d0e175da8a140dd630c66821ee12eba32b23919741d307941d126d02ae2889592fc2e07b45ac2a331ce3173d6f47e092b56d5a8e4408c62ca7729dfc6cab51d21a88775ced8fc7330b306b5bcf2ea7acc42c32f52ee97982aa6227038f0c104095c4b46d0cb6973e1241a49131ad8afa64da3ad3896f36422cdd352a4a17fb63e017509d7b4b742c38f08474c043eee95431efcfdb0f02bd44d8ecb44cb50d896fe0ce26de91db7b97d0d6c5b38d60734d71d7c5ec39b832f07034f7b22e01c9427319e9dabd4ca6a898c07ec7bfc14090444335604e7295c781a4dca0ab1e527c4b2e6e8ba22e0df4bce06bf13bc4a9b0abd0c0fa760ae57279cd2a209d8709873ae540edf5d950e46f53aba633008824c99426f0c9feefc037513bd80f061ce6e45e63a1821e9a541e22c86c83e5178566a3328f6d05eaca936630bc79312d0305ab640127b48f13961957380085aabbbdc17229a5deb2a130586b8e6a49459d4079769466b75a00f94dd087f213c147035a3a20bde7383b9e7b0fe79885bd2ab561cf8ef04f003bff53b54534dcaa96e245cec464b50a502980337db7a6af6992840e090d320679bcc8cfc5f58dbc943424b202050abb45db16c6debb9bc6bfc922fcc5586bc8220b79f26fc0007de34082e7d9616c0817d91d89454e56943c36096357b53dce22f8ae9de3b6038a8407ddd52c9d24930672713c87cd65c076e285ff2ff864e539ad4569be69bf024b7e67330080e45a388864d0c60dfc25b99bbe6d0121664b8657ef08ff86dd9c2e3b791eeddbfcdc68bf738f129110f26d74cf7293312f3f35fafdff38ebddedb1b7e6fc585d681db2a0048dc7ec0d3f2a62b08d03de12c85aa85e6d5663d9826caaaeb2cb008b4ab047b9121e1744edcaffa335469263b49216b569e072fcc4ca3d281068b142c9636936bf8a64d5b4834a5cde97df969c5857f7c1ac50c051f78858dabe6be96122cbfba62ac85d7405783615398634cd4d418061bdde1bd54cfdea05e722cd7e6455dcc65dfef101ef1b8e457e31a76f355480c45470e61d9128fa388a09fdf9cd6d57a70bfec8d63cb313b158c392e7e3571e83e6bbe23fd88b9c748cd42368c8798ae6fc80420a813ec2b6e1bce8a118009c14e47923b464a48f82f25ce417fb7f88f08598e5281a511c676a945bf02772447e80e780ffc0814b5b63a6f3162fe3f2addbcfdce63c6c6132f5a44b9b4317f9b16281626cd8f34443d0128c1727cf84b6b07c5cf5f780c54f8dec5072e8ed4b24eb0bf3c178d4c6a5f6069ce3330b6bf7b432897ac7b72cbb39fa92d083e58ef847cd0b115dc5317253e1578b6c45701460a912824bd94b852a24e0b88f0e66bcdbab9d199751619f264d9e4ea4bca215735aea875c09711bdb935eafbcc13592e06333d3f9f548deee5018751eb22ccb786bfb2fb819c4043129e04279c3947b1da5bd657965a63c57137aad5206bb221f971956b90fe00e577f27cb74f3933114570cf5784611ac50c919fce7929ff33913396234dddb870be7fee4094056b0b8c44055d0861b74e0876c52d1fab182693445713b0cc3226d6aedb14dd3faede857bd9fb5fd961aa31ae1be9968c2f8585c9a4dbc9546cb2cc7be2a191f5a8fa90f7e69695bbdcb156b75287285263e6ba28e01ae0da24c5d727df137d2f2c5be3005061560eec57f340b94b74b7707c125816ec062ad74d13b36ee05705ed78cab9ebaef7682f5c556be87c941125664b17a63a96980364cb592e8c9f5cef789d9d885f337bbb08f1624d47b474efc8ff5bba81e18c34d0e6afa2cb005eee4bc511bfc5241f5588d1b545b3d4409e03b849d7c0291d9ee358b9f59b18f9fc5b84ff27fb0fd4295b8604d9cb24aaeafff6aa71f3aa03dce2129eb0c366eb41c890d258c798e8cf5e37ad2b00d3acae7c683003ec8bd7f3ad30452dcdeda8cf971ecd500183276ea325b3a374ee4b47fe14fc588dedd455980cf6e94cad92d6bbeb1a597d7e6eee8badf08f75f381a0448329875761c9f95c7006dc2875c555340b8c7033ff03880c989cb740e3e131536ebe6a398e3adb849b2bc2733c5e8c2f493ee236873266ffff47e60f1f4ac4b2582d52e3ad0a41b83da82156ae6dae976a73f6cfa0b47402aca9975d8c6b35f1ebfee93219c2d3494ca3edcf65c8d00b70a259523132ad8100448ed5a8a84e884b2cdb70038f8bb13666a649b20d691fef9b9fa3c984e39478cba565885f184a9d2474ab3a1d858c01b0e65a336e25a35210da9a57db55c0f49fd75fe4abc075691cc0652ebce4f2c2f8ccd7ad23d3514cafaa47ec94f56139b476b6486472ae360f8e784aaab4cec5e86643822de7a51424b650f29eb078448d38aacebe7e9cbac58d068f6921d59f375763b141bf3a11138525d5fafd8e375f76163d67ce0145f957430563729a4ec1e16929efea95002bd92f7b8cc6e1a8de7156e3c6d05d2c1e7ed13586f0ace778c8e4587c7b1f71d5d607c579fd79479da63d712a28f4c6a827786e57799020abb2da667a4aa097cf4072f837690c4ed96b2af15bfdff8fd22cdb26bd48ead51de4440a88732411869fdd76b46571f859532001f98799fd8993cf870cd05549d8a6aaabe938a95b6624016b82308f025cdd69b2850c6e91631e0d7772ab0700a7dc0549f106365946638f185bc394c486eddec1f8edc14db5dfc5ffed4f01fad38eed6f5e52d640a818e4d2aa8e16558d4a02b2019e01afb752c8b8dc8883a9d1a2d1f148d79f524291e123bbf886078defc3f1e7ce1a9686513384fa258304ac23607eed34d8ea855d0acd6eec7e708b6ef18f9bdd275fbd03152211c2f577848857734a1132bc60cc2374556d8ea6a137a14128c139b67057c012679ebd52797bd3eac27a39712ed55f00a5903e2a83641e1cd5faeaba42401e7d4cd04c7fe2c9609064dbf3f4cc7c086e56f17c38c1ff831531f37573437ba204c1260387ae79ed810dca7126f542d4bd75ec8bd530ce690e25b68cbb890a1781867c491931ddbebce02066edbfd759b1e003eb1c2169a1e7c0c681612bf8e45f52bdab10922262164224a037ebc8001249096c1770ac518870540c329f0c1e32ddbd1e7c4d9c3f538f5da58bc021a9812aca718f65a3f7f790d3332f648dcea4a15bb3916813af3f8e63e2b5124ee6e718347768e572a71bd3e8ab7df05a7b096521e165f4ef883af2efe70756aba617cfff53f93f88ec977ef9992be890c2adc2bc79ebd871567900a29b87f0c3bde3354e7a956eb00be79ad36c72335df472a4d40a362b81979904dd28c2e8c3de7699b3f67a183549250cd7ce4298dbf7b059f78b19b9a8fc5ee9d0cb02fafb3b8d9ce48362069645e99ee57c6a3bb5b34021fd8273d4ae441d7eadc39b8c4c47296b0ba6e943fd4e622587880c565e8187db344c5b561ea4830e09da916c618465b07f6319ece158d01b8ea876fa9df0a44582c5ab8addfd3afe25a7f1d50e25abf6496a52223405dbf3123c97166c868eb6d2668244ab153420b255b4f4469db02e7c027474af5cb3ca523cb59080ab11202107c377083bd0889071edb0e466b2e7dc3fd1842099213a39b4b341b61e1977416961415af76a3009a1120fbd0a04c95fa03cdf29ad48450f6495fc7c3023a6e19312cf97f5ab0badf3bbde65c6eb8e877c6c6fd98dd2307935d597c73ae6818659e7cab19debf77112b9dd743c45fd46f780da544482bf2856972de19fe21dd83e8cb1492bb782fe3cb52881395afb6272cd1f9b0a67be9cd82eb57a6d9092583310efe14064c03025fe75fa5c0a01775c73d9a40ee42872f066cf87200dbb4be013ae30c29bd99874d6d9134f010e15ac20d1ecbb87d6a7ad53f2da5422811f0fcd854cff08e08b2f783405635deb385508c18fad06eaf868c649d28ea68596dd8ee4233c0e08f9a0ec20e23a8e312b94ebf659bf1e1d16969866d39e45061d0441f23e0309d2dfd48d535e643e59834e41ef2e313345ef74d244dfb0ffb7017c256d6de1e0bf1dee9e6c2edf10a88d38a3b3f488db872117c8947c504bf0766beaa2240ba4f4e7c5defd596b69033720dd2418b705195e584ac2d5a7ae66e5bbf789d2cc9678cfbc35e111094fd3028fff8e28f78c9b89f2db97d873133202d78fc878b4b263410dff64867a99ba61836542f016529a11b3431619d7b6426f9760f729f470ff62ececee4d55140474c9b9104f3892d925f87f1ace4f546b7f4f65eb072cccd290ecd9387fce99a532fa616de5dc36a02c896e6a57f9c88ac15d7a902bf8e7fcaa5433602b7b43609d4e4ce60f9c2bc4064a61d6e318d7fab241586a42bd2e0651c78345f9ebc131649203773bed57beeca8142c88e8950f0dcbc5b84bd9bec105c3545db147422b6a4bfa59d69e5f9d350c3a0825b251d1ad55f8ce56906a75ac42e32f10141f621847cc34f1c23037a9434d56f0ac790d6e3fd49d5b33a398d3e795639c416d5e028b2d54f0b28aa283368be8f07f1932270d225b2023ac2d609ed7ef5891574f2b1905b71c9322a997d39e0b88e7f017dc24a702372b960df234c105c138706cb7f07e89fb26e41ab804bf4abb4cd7cc1937042b8624a536952510ebf73bca40604f5c2eba82e0076e464261dd8f4fec63ec52e532a68a18f95bb09efc85f8b878d00c381754061251deee285ec0ffac304c93d9f9ea75122ef3da107dab9a1b57a3c49fd4a6d95a353cee9a4a5a5e92d2a481fa168ac974e9f2b33cf868ea9ef1776599b21cc45cb02df33c74d1ad88a7d1d79d0b12206417e76579e047751f23e11fdddbc252837e87eeaca4897477a55427d46b9e730e6875eac2cd1cd624e6bb12dc39b326c9996e25e4a7fb7f373373edc60b2ce87ba978fdb506937bf29d84a6e5e3425ba4929696b5bc8029aa87afbb2a95319e4daec1d8ca5d3485e4f4634a3f8243ce4896308019c8fc20ada012b689a9debd79b7cd9199db57ef77376fa301398d1328d8a6c0be37539024155e53c7d97c52c7966866998e74243bbdd24be6b3e69f9c151415cc9ce6158b5120f2dfeeea5bbbc9e205449e7a83415fc0989cf5078fe499aba664489d2be8b8832e190bb9f360174ef7e50e8328d7a823cca49b03d76496a5c52ae1fb6444c6c458d383597406939df615f630f12beafeecb358d322d8b0ba47fe51620aca3e6288cc4abc9e60cc2546f15564ec6b133120019ea2ebbf6eba2a4b9fd3c867de737741325e0aef481fae6d8ccc4ff8038797b3e34335c7e5fdfa99db4a0cfbf64e17c4c1a97d5c8af4549300305a2511739f70e55385b5907071beeba057aa3177ba28fc9341dd482a495978459ff135560045f4fa71a9cbac8a88284257217de603e691eeeca6332f1e40c5209bf33182ee2e5f99c6ec8de14929f2e42651b899066f3aba2d85b619f2eeed32e43f9b13382bcbebfcb9231c1db59071cd46acc9bccc974b826870e169c6955672e2766b71fe8b08f95795f6a2af117dcf89e56b4ddf20ec325927eee68e8c3f84efbcea5d0ac3cf94cec10a9c66f5eb1c4d86b15bc9b42c0fe0b678d48222db2dd958b0a2565cf356e5680ee5a968eb9de1b3912b765338cf0ff2cc4de995281fad5767d32e871b6f13e345537031b66e7f8470028a3261fa533dcf35c963bf4979bdcd2777db7d636471abc10d2d532c41cef1f3594e7687e5c3a69daa38424187c5a41f73e33a94c96b7090b27b167da1477806a0be62de8cd4b358fccd2289abfb460dd6e5870bcc0f792e6a58c4f6d9313166dc368d3f4c9d9153fe5d11027be4db9cd5177a6d76282d3f38b987a3724bd483cd5a610a68579071ef2e2fba3bdb0a84cf1f8b654fb182c4a876c01306d7b7244cc1e8fd7fa5ef91bf081d6140a63e562bc65e7c4cb95ff77bf2f2a984a5207e61633a41fb13bd578cb7ea9008f905d7f8a98e25436cb70b49ee119d5827c44fd4cc039aebb42a16a7f3cd085253e5cfcdcc4bab2eabfb1dba96d4834a212d44289e5e1275b3579466e4a2e0904b5d6f1a6cf59ec6391ef33f822115254855f9c57b7926cd270503818dce14ccc3a3b96e6f4e875c69679d23a69b8f47d5b861a5ab9330f6e26ce55c382021e25fedafe68728e4c73043f3ab4bf896d6dc0dfbe50801207254a8cee9ab87372786c3a53a3f03a0fef3656795123f8ef72c38d5200fdb9b4616dd381247488b716d7076c81c81f1d084cd64fc50ac18b395b35ee0d5f77dd6c7fa53c8704f1573a274b24f1c1cdcc74660a2b37d336980cdb7734f366be331e72d85167a9fd4d139fdbb1e8a35172d2cfe0bc9e7d12476d281864bdfe75c54b82ffd76fe02465794970b77044f0f7fb8d7c0897ab4be1736fe679db3ec8967818f66a62f811bf646fdcfc057921d46d66b08427cb4b19ffa0872b0273bba5af85c516ba28c3434b84be9601e25bf7b97203d2474c19b22d0545836d282bf8691b8b6102043d0c276015823f0c6859f0c67c61f823e62027918021a290c9fd9987c2ee7e49bde076d6e01b2e3152e92e2abf13324582c9491980e55e0cf2d256a26dde470b2ca7f0c950aa9090b26f742bee12ee642c9ba31659a2583558b60d33339bc9e13b75602fae5cf83d975a11d078f8ea4a2dc52f3d4b3a8a9151e33d75edd0ee7f55927e94a55a7f35aaeddf6d6534f6b281248ea6fc6632c2fd167d54660ca8f6a8a26413a71cc3bf6d1379b9627276b63fdd42f3e48d189bd04b7d309a092f0aaed3ffdc32bc778fe1aea6c79202cb2a7f13f4c09de4fd0cfcc0303f9f26887af8db10f967277c442287eaf216a4b2ed4d611a6fc7c147390ba4f13124735dff25019c9b8815d5a30c2e813363c38fccee0ac850ab7edf00adbc5fa94a16242e12ec30ff193db4e1380b954188e3693fb97d9e6efd945f5dcc3e83f89b6788f98fd1ab3b0bde855426eed757a922799c8bcca95a6b1f2386ab6f60c299db922aaf493444ffa3fba4eb29b8cd761ec4448054cf202e0b80fdd65561175e756fe38fa1f389308b7850247cf9973dcddfeb294767d1dd5dabb41e73bee77c3621ba08434335b31353a41714f05e548765e9df194a23d6c1a5d7eb2911c7469ad91d3af0c0c67f2b557123a235647ec2138a5549a8e4af3d67d55b5e5b673bc9efc372603ea4ec5bb6de6b1f83fa43ead6d4681b24b7b8681186d95a365d29290c1b2570cb70556f8b6d3c5837b760538455e96c86006f1e4ab4446360e111e8922cd12a7d76625f217da4bb4171c87e7ce7fb5012dbb4b976cf9cdece31f9dbb7d0a48bf4abc80ce361ab335da4ffe9293b21736e3b78912beb1fde8570fa72e7a8090c73cd85f2b53c7fc6b8cd41e9579fbf5b9fb6542cfcfd83991d53cdc0e57dc2a99f98c23c1a3c1ebee80f87952c54ab4f9583fbdd19009365c15f23ac4527698f82ae30181e9e7d8be2ac1f908e836de59cbb3b27e3a9fac362a0685809ad4ad923789505c611898eddd4b6bd4dce6c85dd9cbb0852b402afd14a623f656549f15963f4544932cc5978c305cdb118c530c28b41657765d0648f7484cae94dd4b8294f5ff973a70903044db81c264ad7c9cdf7114dadff6980bd269a388c4e195d3c8de21eea9b029b5c8726046e31dfb05bc8450a30614bd2d2d2580706b13084b8ad706147938f4aead14060816eb77d440d4aae95af3f41db53809c4dd46310b622be31bbe91446092fe5aa8b5081a681faeba7d6793bd2d16c643da04ed2681dd3eeef05c736d6b89febbdeba946baeadcdd145bb2a2e79c59729a21ed70b003a721866db0ab31216f743205ac37750a524cff7e69d0392b775826254d8bd710b9eee705189f51eeb9406ea7b82a2d55c8d0fe2708b033534de0d0c16537103189b0a2adb046402fbe08808e808e3656c51998ff0d317338a26bf8a6d4f8bcc0320a244d10c4f119dfbd1b21dc1b07253cf3ace0226060ec3e610a41d3284066c6b6211012b6aa93df4d797d0d9c955a7c5718d35858a0ac92ca22caa32ad8e5f0d216f1cab15a1b73ec31c7ae5adfcdec335f3c48d263c3014b7a75de5017c8ca74b8dc9d120b5b95e1eafa0912d38f846c631bc67ae93d4e6428695ca0e9ebb29cb8881ec4b4c45c27cec8df24c9f7c68a2e7b50d3d91a093172efa23d6f67e3e6daa35af10e15d943a7a8c814755f7edf66b3734d9471f2423240474a5dc83bd585951183ce9d43a9765bb74f6c0fe12affb0736b9e3b2109469201fc862d25057cd115679e1fafa47a1c13d30a414503a6e816fbb5b9e2aebae0fb6f4e7e2f3e3b73e85cb1ad1bce0f9f956a862c401a2a07bcf5d8eaec4a9babd23de6d1288b57deeaf982ed8afba562753e2b1e4b4ebf2bf4376835d685b4a1616310d57baf61e2a8796b0ec576810c254930190df32111aeeffd35a28d6814150ebbb2cbf7f6b46e8cef1d9478888cea4595bb858f196cded2055d1d97823ca7a6805f2e291576c7042285469c45c0b797b27fcdf91f8be73c1e583cbac1080506578979e8f6bc8a290c96279aebf9364dce1990e7b0276643d8f18c43583018822b797c07d9c4090e08b0b2a662b69b688d1ea9506d8ef5d1f4298fc7880a2ca57d6a4afb0ee9f9d1acbe3062993a82a043053546d6a79f06755a088526cca22f43259689d06476346ef0079e159ebbb89c08a4e321164c2b4453e0d1c2a9f8c6d442f9d3c3a0ffb0ff5eaad4502c1f9a05db567fcd7a858b4a61ae7636c88dc680ad0d635a89a020600fd9c92ceec130cdc2e519dc906747f9728bae2a3cc7bbbed0550e24dc27b2838ed260aea4f1dfece9f3a9633c939a62a4428908acb2133bfd5e6ea04dda8cf3fdc70dddbbccf1b69c871a14a4ce0523917090a0c5e8ed47abd59d472819f03a6ee5f229c233f9f5fbce93f2d7cbb0897f2fc8c4daefe7210c83c2bf15436af6fd2475877c1dfef4d51d599780dbec70706902a6d169261090dec19261e3a3decdfd397d752b48fbfe45949448e1ccfbfdee970e1820e98c2ba7ada12f2851c8d37db7cd160e55bd4b5714acbb11770745468ca3f6b6a4b4b8fe77a9fdb112da3ce444cde69b1986369c0edc42d02ef5de9068b8e68511681e7eaad513348a874849091c68ae3d1aea677d38c0de3595f6a571ee78b0ae2ec6af11316dac5c43032e45dd7433131f6a71f53c3c2a82b75b44e9047502a70ecd5173662dacb70d6b87377e845bec6a5d0d54c14610d8bb19a4192945ea7c5d762cb750bb1ff795a694aa8e16efd2dfb94e8a303540e6d094329a033c116598b791d523687063630330d29679e1db3fb2ac8e797d2ff90cd052f7164bc62e63f39bc4d553dbe147710d5564cf71f17817e6d65382442c48aa5713accca72cf6b17030181065bbc59bb104e5d4e72d52253d601c99259a8d285cf48f4fe6e0e4196f6baa1fcc24a707bb153a1acd78a1693ef2f78e055a68bc6b976794a63b986996f0a6665a4a96011336888311a077645eb37e6f70e9dd537d976aba6dd6d97cd38618d88d9b993ea5294acefeb318abb6f2ba6277ed4fef2d083fa7116c0f786eb5cc8a470bdfdc3c9ba450347b6e830b763c55be47890804e23e49e4d0a000b2c548e292eb42c8f96d6f6b211ebebda6badf2cd26d97a56d94aa0411d809aabb2f6f46bbdbb0a05b59e811eb7873dfb286dc2351a22df317debf8223f016308f0f01a86c82816a39519847d94b8ce1375be5ceccac5893bfbd52da1e997293d8de45920cad6a0cb617bad7bfa165948c33847959a7150e26e23a3e0bae3f6c0d1f0983a84e233ef59f57e8fcfdd8542316ddd23918f18bff9288a2f5f4226c27aa43fed16aa347e9baae36792d998af5d6f55aa27ae429fc7f56285e37868b39895b2898342ece29615e34b5a31b5e0adc0c4bef04bca0f0fdee60b7f9cc8483aff06e89aaf6d4cd765c2a599dc5d7a72d91ec947d71427d1c073f32decd96ece0ffa4a80f0c0509fdde0a6349ff12f5ae51eb43662dfdb14f8ad34abfa22dc96b88bfeba1f3243596dc91f27e8a8d8279059b5ddb88b44a78f676dd8b0d32f58b96c6505d93a452bdf4b06fa31ecc6a6dfbde5d91c2302c0605e9192cad0d960eb3e3fb5767b5b7cebbe91bce67a508ed067d11373dcf671f1f989d922d681638ee6e723a329c174d63ba635e29e7f9181693855983c99900d96b161b07c021d64c9fa44ee153d36137b2fb5e1ebc97a30d53e0b2719716c03c7333ed4486383e33d33b18f612acd5cff179778c740a25571f4a98ac89b77cf28a842d95fe8680031667f75277850569db561e3be78df6b102432f8df212d4937067050cc209bcee3fbe73dc4881936cb2caf7a2819c3bf47c523e15927bb298184249e4af6571cba2fb770841d5748186cd96c288594be2e71fc25c9209502f067e0f373953d56662da43e7f3488bfddefe7f4905c857d93f25914f384970668a8ee0890fdc6451192c293d1e891bfbaebfa1912986c72966df4b6c97966768352ac74844f0235c5943d38fe7e34d2bb4b6e3490e9c7fc0dd5023b6893baa07baf2f52a0f83c8a033c69a3539c27bd2340141e1e3a29487362c8a7d7ead64bdf2526e5e20b4a92de0e2cc499349643e90b020470e8e4707150c3bd2f5d90327ebe19ae9902bb030408c61f40ed7b015d14070293eb1e4bd93f7b23f95c35e06274cff206aecd913a07aeec65fe4a3a238975f5be3564e87f725439c0643ca8764c8fd2a42b04d7704c4acb8ffae9f66318d4735a799af3b5177a70e1ceef7ae14ee3f4ae7cef3c6b513701c588ac1e66899193210e7cf8b63415f5c6509e6663005ff41a3bedafb37dae01ef67ee168ec85f81b61f57224f0d47826b7dfdb2b1e2dda23d0e80bc0a546bd943ef9d009e8c2672ae08eef034527765a264e910a464dc77a8ba80f77fcfb4dd98d9421ca9f2838d4cc9080ba1b40a43bffdfca6f3fdad76b65ebd505930fdc90e07ec2df7f767451e881f8a8fe66d3c2399de6e70e9a96a6b33d6b883485dddc3c22122e3f3ff4f7473a43b0aa6c0ff91b198384a98e9c42712a6b28e420b1f6d64b9d57d5d2e286d375cc3bf499529c2641a9977e7f1145c17d36a930c75c47b53a92a82a3e43db5994ffc43afd257ff30da915382a839e43f17858b6fdc4148d6bc9d9d4fcbfaf77b80442bb6bea1d6421a9dc86dae8bdf2b4d7cfc8e19fabdc48f11b7c4554ee198759a155c102503af553306d3bdf234d5014f1b41e4d5993a54cf738e16627ef0db6d931d2da66a57f615029e223dc0956aabf7585614627f535713fab110fc1d22cb6772e49065461ab967b885c30ec368920d75f444d8123d4b4d2742ef6d5c1d993194e34b82bdf9f18f934d4cbb82556233c2c88cb8d2751f2d46a36cc561716ad475652ff399de21da27889209863e126d1b72e1222e17daad87aace226153d93c0b02573d79364aa6e0d432dfa5b741cfe9c9200502581f06ec28694bd554b8518a1d0f1838956c65ee636af7c588d08323bae9e9950caf94d80d946cba9eb949b132b2635a50a4ce855ad42ef08d1e80e18b36cfad9367d6cbe2b5c1e7a7e1d896f74396f7053b062f6354d0b16929b57a6f5b02b1948ec03804044356e274dc11302ac22cc4fb1316a720439272b318b5d6663234ef45b839847b1b3c3db9f709a11e75468115199932ad0bfc3e688daf6acd1a268b2cbffa29007aefa9ca99e33ca092e2bdb3d765c9e5b2040217a7906a7ca3006414f9567db771e5e43534e8aca35d4ed32ca870e5084fffb0b49b73cafd196c17d2ec5f0691009bd0489acdc0d67e35f20f2464ccc29a908c8e4ae2b6cf1ceee6bef72dd62a903b9e5e3053435a58fe83b3dd6f8b340f93e88bc178ed36d88b07fb75ff49ccf524c23cd3e7e8ba384a89ddb9fc51592ebe0a00760ce3a7fdb232e5617d1bf69c572edad4557c78e21419c5228830d7cd1dff1987fd26a33f44daf4e4b0eb360858ecf447ac4bd84b222434ab5b9219a30ae59b86aa1b3a38296fc2279ba993062bc37b802d807936960c2cacaedc6bd9a4b7047019df366310fc85ad024815818d69e05c46da8ec74b5b47f6907a45dc30ca29c3147fb6218b49ad91129644da0d5adddad1532269c77c8dc0c9a47a0f725a4488469779a3bdb0d1957dce0ae661eb81a75b4b10975d4a296330bec87eb8713e73111a0b06a76d21a24f4c31e50d0c16f3d2f9a57a04586cfc1205642255ff93b5501d2e554242001c84e1a4640a64c23d4620736c252445e14e3129b06ad06227f400375eb6b23391bb0714c0ad96218a55c9540a20babc3289c8ed03f6caa2825b0dce45175dff471a943f4bdaef154945c3dc5b2048d00b7c79919cdf6b9b0b2378de6c248aaa9923349ca637b853f54226dda0ca93a164175d405e11960fbf01ccdc56c14104edb60ac26b2829cc0d586a3cd835bf6a05d71660b80d3b44c82b4921e8b317507539afeb7746b0e636706cce496ecd6a63616dd6f060b5942d644839e73de18a1e0a4c638b315c629a3632d67b200e79ff472490c2915110ce2fe6a0aafbe39fd177d42992aa553d99e83e16c61acabb0174653fcf067a8e1147d1812d054c241167ec44f6fb2cee0cf65b0fcd463cf0e9dfa61877529d0d53cfdc617fab2b98350ee5582aad74e4c8107af091f2019e971732629b909414f4e8041fcd15371e86ad23d3b910361283cfee995080323135b5803cb5ab2d52e9c7b15e4871e2f5f5ec42f6da63c7a2b37478589309705b566bb2e4c14379f8d9c63738ac977eb626e4083356c4b308a9a90ce9ea57e4ed2a9380c2c2d10591944ac664d0357d8088efbe1ce22e4b4a0043d4f3bb1d63700715f41a2a347053ef909b41f0a358c6d17f1d4a7700e5196a6bbba4cc1bbc01d2657999165b8ebc4fb9ee03ae2b94e9941fd031a4cd97b0f7ecdd4bbadde07790de2285f22716d763ad056b442798082b4d2ec138386a92cfcb97349c8f0448a07727d231ebcf0ce58777eecd32698bbc697c6ba2294b28c3f6c40ad596aacb783b8b48a0360df7f46abe1e86bb90568646d1a409117746aeee550dd4c5280f949d3a78b98a9868ed889b9f8a200172deb688947928b3fad17ebaf8c1ed036792c4ac1bd886d3585c64f2223b8ac1d8b8b3d3edb4eeeb0adce3fa0003c6594a05ab2b5ce6b24266a6d048871fd23448928fb853714cbac45f48674fef28751b2111f14f3818cb9f61155963f61f6f862eecf86977463f2518e38958216d75818b2a273e301b6f3f7b883fb085f29bf35532b44a8ead3ad3395ba7c007801ab67abce484ad7e7b31604dc6bd12c699ac1f25d1ce5ffd8604215266ca2be01fe1f98f63e78594b28f0db43de03f378cdfabc9c987dbfcdd703eff8d58a80d38d49e0b5793901e07fb09cd5c3eac60ef6e1ffc105774656071a39294dcdfdd6aaf2efbbcbb8c3fab1cd1433ad9709cdda3114540fd1e1a803bbea450bda5b94a9b8981c874a5bb1bff714f77905c3cfe583c89966ca2adeb4dcbbc4cbebaf685d68b5100448ca359296782f89faabdb71967c0a0e9c38bf8f4376bf9b9045e0a34c78ad187fb6bb1c8658ef39cce97c5057d5c2988605b4dd0ca7611cea3ddfed834440bddb9bf97c01d9283586ba64fef8223e76d296d4cf3b63f18b090cdf2b659d3580c94d95bfa6dba3df8a665e141d39626591390837d8f192131818e246be87f0eb0de65369fa0dd83104feb31c76e0627a03ac606a231857b3e785d7b53009a098dd32c76875993fa87a047beae394a1c3a81e83498e495a1b9c294926aea21ca3f0977e1b7385e8042643a051ce409af938c7ad9431b23e88c32cbdf71d15083626eaa723a2cd3bd0678af19be4211990b598c6918cda2850f9a21586804c3be2ee7a9aaffc374d13916bb7ab8554475f60621f1886ebab3f147192b6d4c9f840f551a255cbb7322a4bbde471f4b10cd688ea04c2de765794f3b1f43efaeb722fc1eb975cc2a6767fc27f72ffd7e9e25a2d1baeb124a0ecde338feb729e1e517193912709b3f904eedc8521d5728b90142e0152068159e063a31bc7a7b9407d0de355b6cf60bd84d666061e6cf4727918ec3da8bafc80f4f3f11339c5afb3a5f7d874671c93d1155c381a7470802bdbd553989f049b17071793fd3136dc3ceb5d0f533ee6bdb74711f7953500eaa3ffc4dbc4e621ca96f7df9c1f3ad3b608ee04ea667033c3212e1f225f6786374d8d6cdac5c7aac3774bed828c22208c2e63e1548d8b9194d6f6f59e673d61604b148c2fdc530aa57f291fcd2c23091fece3366dbe30f4b28529dfcb6b5b5960c3a581627ac4a1b94b26914181a4c6aae309ebb0769ddc39b79a781b41707593a4800c7d4c83ee78a9c53ed3532091e42f5191532b2bb2dbee203d4b23b34e90be4c3493b3162ce1453d5839a88c11d4a8af85ac4af11f970de7837e16cc61fa9d335684f829489cf82c4ad054027be1a45ffe11188cddc0fde602718ba6b03d1f8348b197e6bde7a4e8c79fe0ba72d770d913cccee2a764df7f25d4d68e249a6721112620f6e4f4f8cc80a8df84957ab37bc34d95a21c5afa8f4256831d1d8eac15b2d28840e55060db79f82cf7ea659069ea9f7c0715492d3e72d534bc6cdd558a1a8b2921e8b089654b757b6c514d2ba62cf9785f2449c67403c977da389f80e0d6d80cc63e115b41b67da575628fd888b7eb5ed69f8b4f8cb9d7414c4b6d5ae700e3e21c587bc6ee2d5c07ede0b740b1ff24aa4da98508d02a26aa5215a640a1da26dd95f9f6d4ea480e4721fcca89eaf646c20734db6b393e1acae738509799f316eb7aa2382ba61593d883ff86e72dcf1a7b0406350512fd15029f147cf82872511e9d5f28d61afb59725f019be58438c8e747a6505dd6cb6979e2669025fcac96935dd64dabdcf53513fdc1accb26c879a09277a8c91444cdf113312b0fd24d42d114a2328ce6d3c60a622f641e0baac09dd0258f20fe56c8fdcd0cd62b4ee385c999ad6cbcf39cfe1b5e5f4e90611a2f1efb29aabccfd69ff41eec5d201158e5dedc70174a6e0bffa62c93437b258bfd19f8b51db852208f0f9f71330ecdb51d8d62c89070d48ff51755251c0212ac66121fb0a3e10465c4dcbd704afe4b03306b977795d274936c533e04687ee0307fe02ca4752ac101e5f85b87b6b75ae6710affff1e897edc1c803cd70d38291b30423cfb5865aac1b6c4f9451fed38d6e80d203b660443dbd3092e0456842d6c70ce36005e62465f1ab9344f6cb378456d79ac36aef270ce7870c4ceaa262fffe8c2c6f4dc221a0a7ac77a9db76904aa069fcf35b4621da9e47d70121a12a1b6f3a97a449150ca4399449894c60974c27879fe308150c82c83e5e7c99680797795aa7812ff0b25aa8163cfca36f09c550cae75b474f48668aeed62dc7c880fa44701e5c11df97c946f526b92834b6e4d48849d9c20b0093420e2a8c856c9d608b02cfd5d91f85004f16b1357e1ea62922445fb6282da4771ecbbbe3ef5f7a13d42f9b6b62600617fd5f8a8a459fa91df405c12b507d1b4daefb2fe112f54190458bbe632f356dbd59423f7b9e51219b20162e8b22939469e84922af68ddf94024f2d0b598b9b9d620f3a477323a1868afb36a5dfab04b35288ff7b9af5d1fb75a5897936863a466b6ab30f57d597f7e45326ab4b0aeac859822f26cbfcb6c03a3c80e56236d75a364b49f0d7ba3cbcdd757f8fcef17f21fc1a511c51d7b91d75428ab9e04aaa20f7c95177832c85f5b2945ec9794e862b6823c263d7df9d3aab1e921c52a172600980a9a48fb0fe69ac148b155ec4e026ce29cc70b767cf1f1fff373c075928cf38ca8f70b37d1fa557123ec27119f62ddca7abe422f2653eb0140766e004726471bfc0fa0a3af858196a413da0414fe243793e42d28070f447cfe4ff78d9bed9e327c91c33baf8b2cff912a0ed217dd87ca70ce77022e8af7ff12a50e0763549ccb97b4715992fe8a4b2f9d4ca8567dc6859f151132ece0d1bccfec3f40c4f22fce7086d71177dd69c24b634c3a2869b02a6065e48804b3133ec33999391c2c0878d81081a90b91a332afa7b2cb0a508e998b251607e14a2a08dba2db90e43257b3a9882c385c0cbcc04ca470776af257ad9509ca1a19231ad095a5493091ed8e90137a924f46562181aaae724a6a4e679281bde3c42b444f8a66330e674e85b3328092fb127a99bcfc76741956e0541eb13e814d585cf7fa29cef000946350f5e3eabf13678751ff430ce1a41fe208775de7f3c59e06cf83778b45ef03d80b921f255210df226557bf16c36ee9fa89fe813eacf1393a8734fedf29ac7355304adbbef8311cfd6336c40823d0541f00f40e535e748ef84097971924bcfc4eca6263ff9b5358d1fc1dfddf18f111c6e9d6cbe810bd66a61531df5bf5b3483a0522b515d320f3fabbc298a360d55515bdbbb65202b1bb3b2ce5a9fbafe9887a433a1a38068aa33c06c389ca4d147ee554215878dd0bafb044dbd1c95c76be7757c642fa80c80d663a3b94b61141b169f168ab3c1147011b101c5f05194bd8c2028ca64b9adae93d1c18d9f80969d6cb9b81b15520780931a8353c5b532a0a32e09fa6cb8e7f4a0611be1d79ef0e2e9587a6c325a92cf4d8de4d92173b256686591a340fd78d35d5dccd505fd81eb5da5633d6a3a8820c0635cdc88ee26546527e759d0325d84b048f5c50b56765ec2eb7b89269d0a95e5bab07ba6920a4bc98584e63700d5e5d54ad89bde7620fdd565472cbb66a1af1482051d8e7517eb4601c00464fe1ca1de54629a6b9ada61659c5ce380031e3b83b8f37090978359e0def961dc5980aaf423f4622d662b72a2e820971b2bfcafa10375a95be349920c4d5329aa656be3d3ef85191450ab9614af0951289824c8974527fdb99f4cc388e94dab76dc7281fad1da9822772400d0ba49c96bd12c9bc57cf1fc2a45356e9c3c8958b15b4cf63b34388c4d00660b65154fca470ae30b0f84646adedcb4e1c5085083b3cc0ffd81cabbd1b6f019311f691e9637770c7e13a8983adc9785d3641bc0c04904603c59c9b80e836f91a5eb0dcac7fe7e0c1c594c27b8888b862838e6302cc0038ca1a44198723c84ed1f9513a311f1284d6f6d03bcbf97bfd585c2786b482a471c789ca8012af2c171ce6885666f8a7679db876e26a5ad4729555eefa34c0bad76100f4f00e7ac6b3c555ee37d6a63f4b82c083b76308d2cecc8035c1c7a966a0d5d320cc67011e52aa3ad66faad1d65afa675f1ac4569873d69412d1bd85119dbf46b930e41249e5cfa0939339c59d543decadb474daf263a4c89e8f61fe0835c46239d08011e4cf914adf0d48c7b64d714284a19238b42aa52b8d92cb1d2cc4488e6e8e24fc3fa0fbeb4e9a2cb7bee37e3ed2f4731e0a6279c649021b0c8c602de9b0518b17bd04476680e600bfdc134dc7bd52fab2425e477bb0b6eb0ade2be44fc4ef5442b3b4400900f1153a92ff6aaa99d8a2a63729d6368690319a5f8a89c55297bc7f71a8412d053d0f6263cd98a3e1ed30af67141beec4dbf4c811f8588b5e5bf8d7e97f5de5d43f6e7907dee2439ba0650384dffb238ad3e8b7a674bc617746da931b2fad7ed5e254a1be8289b0d28a791bf5013160e00fb352fd2f498b11205e314a16608e6f42cdefb1ffd04023fd683291b8287797f751d027e03641654c6ae61ab18ddd75f2620c1e9240630a7eaed6c6c1097a380422ace090ce486f832d7314ed82739cab598108a94e9df81eaf59c5dc117ac45bdb211d79c03e5503f5a13b16c91e54f6220e081e743e3e37f024fe0adc6b110e2a068e89b2fa0569d549a20df7f8ab8447d1ecbcdce5f97f1509a5f9a12187d0290231192a0db5946a446b1791f78da55ad4f2df6499e025327eeb0a41e027d30a9d6a5a6e81e1894483477968540bb7b88fef8bbb77e5b111aba40458f3de9bd7f7a2d196f3c80195f51ee0a4fe3dc036de5ddf8977e9a6ea225c9ac756520398f637e2d14971ce1f7070778e5d0a72463ecf39c3f8cbb1ad5552d6582c68effcd3f714440d9fd111519762de1e988916f3c15785b36fdc27f5b569c9207691f2f8edd69bfcae546d32b502647ba794262f56fa41e93dc63a7f4a082ab071ee9e63a351084241b68750369e2806bec4c740a1af20ab1471859a2b62fc621405bb6896e73628dba953f0aed727e8e48086b9b785144a9bf8b12e17def46534a8c7aab78af61d84d2d8417fe0c176b257a0aef07016f6b48bf62527f465c7ca0badb053e4bf2281d8bb52150766252125f3132377496a67396325be01d0dbe8d2ac25cbed8eea94b88dcce8a57be5a47665b92754a99b5092f0d65bbb030c3b3081c149a131da5fdf12f0a0980da547a30857d69fe1c0d18ed51f67ccbeac37263e0d7ef98d2fed3ef4741ca9dc810551d3fb7e3ca936b0a94ffea276b9ae88c65cf6ba74f26db827b7ebec9aff776683ac34bc70b290026134f18fed5f2b6029dacf76510344d7afb34afb71e57c55e46a1fc1bf6d77cabec061c0a16f31463ab3c0e14a0e55f01875a74e9ab3b9c9e432c7e56d1ac22c532acfe29c35d71701e444f152a6fe735937d734dd72cad9fa141b68bc6f51f85e9745a6611e9d99f934da512b76fd5a1a1ac7726844466d7b861ca3dc1ba8c4c8ba9c29a15ace237a77f9496b048d40c09b45ba920bf5dd58fcbafa7fa0aa08a8e44b247718a79b3aceca6ccac38cdab20308c98cd3078f5c2685eb191caccd6dd7e21bf30ddd8189d5490108a72438e2dfe76d6f297242b9fd84e601e7cd2c6209103aa3009fc822621390931d4447d8da6c7614c152fb7bed3f9b204c0fb2ec388f0f075c2b3986ffe101c237a40d592eda1e70e5db532f66ce3f95b04d96a87d370ac4dfb4f484a434c8239bfb2e6d614897d2317605a8bfc1ff493d0d2338c87eb6e13b3da3384863bcb8f25dd83313af8d8445d8773241f9627c25cfa2442079e159e1d802803e31e5dff452248b66dcf0c60529197f94c9d814eb2d329859f2f93a4d25d6b15dc8cbd9b5397a95708c782562aaaab6b70b2ccd7011be1f1d866ddcd42264b669180e7d2b716fe34ec112dcf7df957b5f6d533f31a2fe8302c67a105f420c3394fb01b12cc186b16352a03ff0c5e34d759c51dba81022908e8a4cbb1833c9ae986e2318fa50d524e61eb6a86deb7f651a011e9990f0a2b6894c86d1d86bafb52ef697aad661771bec4ba58d689e2f483d1e34cfac1a6c723e615920342e97a18e4f0d9b48e3a0021a5027d76b7d6b05fb4411747175eccef0584b6d74954a35fcccc5fc73def0d6b1521f01d9d66cf092f445640e05d3aaa86049c356df214ce131f57a81bcc75b4c9471e8a34e4a32c0dcbd7510c49cf1708f321f083378f3fef281cc703ba61aebe7d052903dee89d83ff06ee5a5762e244f8a40a46d55a188cb24c68061067b012d6a52bc51fc59956a576050e328f4b01e2b021f631af527490060d80649d35b07d12d01d70b4ff1bc9fdb2052b070b142042b1d0d93c9691dc0e11efdc830eb83bdc6cf1c8ae17a8fb2e1e122c26d6f4da727b486d7d19d8d5388eb313c9c750d328af4702135f0adbc46ff7a4f49ce8b55c089b21d763431faa661c6a750d8ad2e14d39303c0be965a6770bbf0a8e8db14d6c35db92d8348d5e189b96c30160813077b94f6e2e9cc418deeaacf6ceb2dbdd68bec947964f76ee08ccd125ed94eac7e9adbeebc1382b81e0369c4a9c00b8d898d8e5cf93b7d04874960f2a09bada519817891531ad39d051f7f3259d0b79af64a8823e70994aa292f7e7fe97f0ee53459f4543f905213c660f5221f0bcaeee29085fdabaae590cdb075d23ef7a82c231c5a61a8e531c6dd8c6952f01b46e2e974d9e4f397da3d4ce921fc3fc9c4c759ce37bec0ddcf584a142b8bad5f1b50e13c58836ed2c710062c41364027b364ff40b707915ec782c271479d672437125289da677214308c39bda673219449808a2a7c2c5d46d342a21773ed1a9b0f06f3e14476290d6991cef3bf0acfcdb2232f154bce030df3975b3bee23ea31f0ad78de511bd454027d24d6dd8c233b1dca5403c653e8385f0474ee40ce97b17cecef6d7e8ce398f890405764e48c9e011a6e3ffef8e6500812a830738d0343d17cb62d48c8e9b7d7659b52439b85bb7488d8c7ffc2523025037539d0504d69ba0ea290516bc5740745ce66bef611604d6e9ea03da501b840e7ff2cbbe7e773e5218db4e5076711bdd04216ede91b9daf619c345391e69451126bc9b7d8c06d4b79a84cfc9b248e4eff42052cba2f568334e19f3bd9faccb132bf79a1d904f6247a26abf3fb7615e1476e5211d24cbb099f179661c3873907f14677b42be9d5d2d24d2d18637e22d20e39250b121c7885992e654917b7974a851f660bd663e6aa949047f5d24fc131694b1fd1d610a4942d1c015743217e566d7b96b7766bdd044bcbc7c6f966226eb749be64690c88376e827b5bbe6e1d7cfc71112016004e8a6405ee8f3a9c9bee902d84b8490689cbe73c8e3f385c4e4629a830d1a8bbd9261e70c7deb74d6dc845c43071695829e4d80e28336145e51c28969e1f58ee76620f21576af8cde038488835a4ca95abe79a3313280d093c9395d583e281a5fdec9323e4a5a9b1d9f12ad7267ace9721778f2df8585424fc5e50006428cfd0b69fd4e1607fb9decc7a6430c5e42e720f13dd304c88f150aa48b11782f80bbd2faf89c08898acc2edf4a2aac84f39019478402f421288bf92803b625794f3b29031984714a60520c6c4201a15bbce2cb66d2bbf75521da08fce07e5822ac708b3e95a3c05b36e4e3e8015726e0896edc7e1e468e96d014e1d159778385a07427c3d9625a2d14cc401b5bb8fb3da0ad36a8018d6517a0a23994bfc0c7be2aaec64f6d83f20f2962c1bb496e2a20f0628c7f88093474dd545a3ebf49d6ce13b450c178d4f3ad3035620d22c074d04169aca18630c61e38224d2143dd90f1b74536c89de0bb3ab270ad827d6de0dfbdc896634e7cdc9778c49607d7ef3966bcb79b9eedf886e154fd99bc7ed9130b0ce7988e9d0ded787ef18aa86ea09e072255f6b637493febb6c6169f8f4d35e968731a9fdcf8f511dd2ed4084392715a7d4f827709dfc6396b8f45ef9f9e16d39a856dc181222fdf419a300d8d47fca6263bd6f2a4c37040d2b10f575455d8a785fca615396ea862045620bfa3113fd01f8b3f08e70a3b38cf2e056d3e49fc0da3660118de65f449909be06b2bae3c68be8a4d126a8ea0d850ae3ac638102206959aae80c97bb5c81c2dc614106d231a6884a15c0808896804e904d3bd10d0db816efdfc7706f304027305ac82838bb64969f72bae844ff69963a84ca2fa997ab4db0c9f03c0373bb14f36a843a063524d0f9474af02f3fb60f25a060d43a37b988a2d7f301e17d199f472fed6b340e474fbe44d45f518709eaf766cc06a892f690f1ca8f176c48e0d52e016870afc13c241e29f66fdcdc2b3cbd1a989da49627b5d6e017216938852462b1d2530e32377468e4447160677355b131c088fe422568c65471bb99861218fa418f54cd4ddbe056febdcae92c656096ad88bff9639662fe71693f0e4a0f493900deae50ab57e86cd194466074a909020d81dad7e77e1f0993d709f0019ab19371aa849f067dc1e1909ec32e38d7d7123cf0971d996cdaddfd0a8c99261bc57c4de3369d9f9627e03bc15d99823e74841aacfba127ca3986b3f0436f39c1ab17bf5ac4b6d8882d1cbadbf2cae2d5493377896fd3209284527ae463d2e2313946ff3ef861bdd0d409afbfbd97bc9931deed315700d419169e7aba0937ccbd39ddfe18f2eeacb2657f45fa55f6359571727d25450f89af02a0c221d04c712db0daac12d7fc3dbb275cf3912ef1d8ce34919f5b33e137c0f8a9822c1cc87a85f1efcb5de281bf70e32b97463dc01c4dc03a7947cf164ebd4e47c04ccd408aefc5173b7c40e8c176cc7a73d8c2faa9cdb53d0e343fd914f11491e689ec43fdf937caaacc38b6dfb201e5f145a3f0e38f604bfdf4f7d84ebfc1fce26905885eb7878d2c7d6bb7af2dd62cc3cabcb9354ac16bde8e74253d8831e7bff9869fa3828d3f8c85cf7730964e8c11c86eb500176beb009492466a71a1f3ee4e08f90ce051677405b7940ff037d95bdfd23c1344a5a0d2c6d1983c0d22b1f8e296994d7ebfc1005f4f0fa52b29b890823362d1c756a694f63ed11379e82c8adcae7aa2a79b9f1644783cefafa69604862cb5df717af16521a1024db08d9d9a6ba5d1d995f31eb6e4beab2af21a2b7fec462ca81e602fbdb8630702629383bdce49eab2af71b4ac82de7567edfb30a50ecc4f7b4cdf263d5f0e1075fd48607d2e35fd78697097643252619d1834503431a28cc186b4eb7f2734d4addc197dacad06f734c6e6f199dce14ac2f8ba6aa95842a79b8f69ce60ae8d764d286f3b56eab5c7673720da584e179c75bf8a2aeef4b87d18e45bdf99229de80d6abad7c700f7154265c502058d259de084362d9e00760421799224df07a5567daf83c1ca9e0800239ff600c6dae82e6e2c6b65199a19326879060cf7baccbf1c0f02194e489cf274b269287654c7054437a64d0ca4e80769158146eecf925f4c065e30b83393f3f74909800eced8bce1bc9d8e68aece206e36545cdfe1ca62f8ca4c4b060ea9763eb891df0cc8923d39089e39b35130558560eb16f16aa38cb6659bdc786a816bf572be214d05811573acb26bb0b243507bd3918c0efdf29c19ea87c8052659fc3233a3cc84d60a258bf9baf28b9f3e9dd5617e0727b21af062b71a2e676c2b411ed6cd1e9a481ad0e81500b9a8a8b4d007ac8ea9b983beaf46246156d5d71727c86465ff101856965b107e865055c1523971842fad854f62182f17d4d5c76b34a0912bf6da471d7acaa7d911274809aa409a35c25eca1156d4555004f5465f72331921ed7f6a2d761749d3e4335722d48eb5b97bb5c818a2517683928ca5d244bee6ecd6249a647af3678f66b5ba501d3832e71c14f01de225d5d33a654fcc00507bd18f81810fa086d6b1430ec0eb5bc0540da343e57267f9f616e6ef9552ca1fbcb25a924d8ac4d2bd80aa4cda93456af57d49871cdba65d6a93fad90efe4c15ba2bd7662b4d856255e584807e18c2f5e6e58938add65e279b32f496cc86552b258328ccc879f679e8e231fd06e40fc02ee1bd624047192a19dcfdd9288e51f65cc61cbf8b9da8901acd35f646733ac9b5f0ef9adbb6d9c0a673c781b3d0bd0a6d409d143966147fd728394c7298714bc855520e5721163c0cfb9182bf0939d060aeb5a2e0ef977c8d5864773b8233286cfef06aa0d7843773dd6ba2857c50ee1f149895d1533287468a8187d7fcbcfbee3c14500cb9ca6ff3a53b572e5dee4dd1a6583095d9f44ed9a24f590686c220e8e89965af62d92f686c682739e049e495b6a965326e90b66139837d608f1d58df6994215d7a0cb363998fd5a2403d20fe50a2415293396606eae6efe7228b50e17c7dae0dc70bcdd74caf32c00dd035b5cee2a24208cfcf2fa0b30c98ba2f9d6d4568ea27fc6dee473722c37931d4e74ef99bedcd4faa02f17bfe25e9a3cd4ba2a7f58f33bf1a3eac3a72123db3019b14d00dea0353ad53bc30bd27210bd22e709b5f258ef4f7fd2d72eaed1c01fe15b7233ce555068f037b8561fd0c05e287d6ba1d7fc9d74674bdb578759cb125dc8983316e952afd784087aff33b7d76b4f8a3702c0a7f786cbe0cc4bc8e7ca5f2b7ac540439f77b1e354c6941cb9124e8bca509cac9a6a44d53d652fe5419a9c448f95d7e5c80535957a37d2443b43c1881825ac91408c357548fa7115e71dae8a75869871ef6b4c0b3aa30af652c0274b9d295fff4aae2427fc20c2103ee00e92a71a46b8f3859343ec28bfb89881dc71804cf49354f0b11e339ded9cf2ccc620a2fd91d86cdacd28250b17d8c00f3d1309d124e339d04d5029e69f30e8025555a1a59876b33e668b35b08722eeffab37246a73b9d11f6a8b71ab4fcf1d6e0195afc0c1607894f8c10874377620fce8f9d0ead0e452434b8bf8b1cd3858d65c257b748f8504e636ec874641381afe890ba1b896952eec355fe5a986e370cd6dd4d37feeaedd8d77481d91e0cfcec48b30d667295557e820c5da9a8e8b9300944d9ef80c771c01f5d45214d50b5d68a5426b1b0bd25239703edad5c166035384db64b43d030143f2670e3dd5ab8ae8c5dc9cd19473a22a8ea725ebb1ffd12d8168417f77a1b60da6ecec6ff929f41e9463dd7e8b73c8196d7d743932c84785adfb01ae0fbae974712f2d22a4a17f3e832683e155c84e06463b3dab8e5c0a6b9f7b75f87e91016a6f20b353c4debf4f5c2dc276df7bf9b02f01ced9e38e4b288fcce77dfbc40a20a7a6e3b29b02f6f235e1e90a0a25ed8bcb0eefd526ac0cdfed1cf3459d2fd5ab7f6eb705b59ffb74d3ce44bc9da75beeac50de98f5878e4cc27eb496819f0943524b3a0463e3dd989e58ddfed9935be61cf67a6d7c30c22a21871af049b4b58867fcde3b65ff570961c000301e5f6e4a15bf47a174cba0dcaf9ebb8bffaef638796ca37fb94c4646fb5b7db2f9ba7c9b445994987777ed5b2ffad71195b7434dee7f472b77b3b39a4e4549a8b730bea9dd1e02dcd1bd24ba633637eed7908800d304e212129ca2158a8a4f65fe498595fd804ae47028074f335e9f2f80f530f247cd552ce41c61a52886afdd12c9f4170d40571a0f76c3b5a4c107f4afdf76cc9334195436d4975341a1104dd83266829d92ab199520c185221f7d7583c14beff93679147209b0338c466614c6a6a14816b544b3a879b375821247a045d1601c142d0c5c9151b351a4f07f929fc5264cea1716e8159deb6ac503c91bcc4c97b651bc2467c5c163df503176d79fc086a8b8d127a070370fdffb642465850debc55a09012a8b05cbf2e21884261fae092ce68caa99602312f7fd4f0bfa25a56fa75a0daac18a97cb6d65f940a848982590f497d77603f4ad0c83e43af79f9f1573cf3b9a841ea55a4628c867be1a0f2baf36b59bea75787aa4070c8aca70759ced44e515e9938db1bf1d1d2ee0d1b5145ff7363ddb24b067c13f17e1ef41ae345aae87715ed5f0a8b401547887e1c5036319b09d47880808bcf892d4587d987f97bd50b650855f70aa9f038fbe4f3d3187c1135c0a5f4b6efb79d00b91b917f3a7233922e0810ac13c6ee6b5e73b95ac0cbdfcbafa4f0fa1e1a6f41594b1057fb3b8c39eacb751fe3091cec1a37d96ec85443ab1f42d86f4fb997b673d57c2a1994ec5c1b8f80e90b33e648877800246165753e73fb960e5cc90bb8cf7b268a42c9f7850e1f0166267c61972fff2ebc639adcd2071b07e8a58ddface1b2044b2a58e3a6b912a3702f55f199e7d9d07a5eb990fa1dc46b294f7ac1cfe7fd49a6857d00ba77cd210bd854343c9193b97291aa0fb37419acfb17adac9323cc8474af9da7689839b0ea70a06f23c62ab2d5bad8593086426c4fab8d82913eb45251c396430cff08862eda7d77ced49cdd57f71dc0032a06aca3ec3b142bcefd02d56e7fce13a8f2c186eff4bf35ebfd63c17b32e19d78dbf8bb7040234a613bf89efb2e55c022a600d1f392e02e71e8e383f1e4605b66949637f7c49e0e9aa4267dcc73f8d23e9519a4ae6d5d24e19593fc94a5a44e0cdc88f0c99bc47e46cc7581b57be6a2f01b1322a6756e6f6ba92b3eb9e2eb9129c347f84cc43f55b2ed8828690917da3dfbab84108bd270f293ab03e47da0689c857549ca99b5cc650e214f86e68d518d54102175b3a173c2a4d2544441d600a93c0af4d4eb5dc047d6a7c3443049c004553245def4b95392be8b95ae01762b0fdb50ee837cfd5f236ae349a13bac4350cc56608dc1c2ad3faee4f537e7265eeaa548596c395ae355c3ced86f8c8c0e011222b0d59284d392d712bd6710d5df402cc06486387df5848bd338c43880538c31686cdadfb6c528b6fa7b8ace765ad7c7be16f451209a114e40be145a5e1030cf51e5b7a93e1b3e1c30a98dd7d8adc22d081c6d139dab1adc009c5104103b1ad2983dde4e63277626453e3dbe07eb48dd1cfb9a4caedb4eb5f4ff9f712a7eeea9a6453ac46bc92f21d4ed70244322128d882e9b6590826e400d06a697f0efa89765e31322f8a37d22cb0b88b8f565682ec0a80bdcd694ad993299ef99abd3bd222c348e9d34decc80635cfea2e97be62313507fe3f66fb16c81aa2aaddd4c32066befc3c73b3e13f25280fa26732c3bdd1518a1706f4acb62fe3740d7fa6e3b4dcf7c3107f5ea9a451c08ef7c2f1223cd1d5bf8665e98484fed7c4648e19a5dd1d3d44a61a5ef8250f3a77d58bf72bf5d7d6ae7fae871ea9143012b45ee228569cc57d71afb410e18ab56f6eb9dde58e8f1c42160d13266b977ca74750787c72f29175879891d82d89d0f382e8b3e130a5df347eec67b8a7b9c1b7907f588e4f858d1d1e23079a1999a6bf305ccaef7bfc158a74a845599b51cfdeacfc9ae2471cdd06a53f514fa90be85037d5175aa252a78f0d46500ecc98fc781b3f283b1d389cb1b731291d08f04fb841cdcf3943cedfc7788d6427aff6eed53bd88fb3db7ead7e1bd6f3db80f84e2803ce90126745ffe83f197253e663ff91d9150760bf29240d7ad5bc02614ae4af5beb49b572cbb0f80dcea3c70ccae7e21122e45cbc10271343f4711ea59152d7c88900dc65deeb67e2a7bae1e673953181a66173dd7ffa934502c58e769aeb405977ed0a896467a3e3ab36d86a9e73b4f943a7641651a80aec1cf5c5b6bef7ef1c3c556528e71e372af3ca3b6ef3dc240eb211900c715315034d970faa73ce99e385834950c553d1844f92d4d5bd0e816993712462dc3475a7cb6ca774bcf77ff42f9d17673cf1275c975b3d8ecee2fc032049d2a55b973a83335f81ffb2892dcd840963f2a2cba58204bd8db43bfb679327e731796cb5cf7568e5ffbc481de803ca554267ca006d020156639631fbdcedb2a55b53aa06c278db1a4a5fbff67395625e699d204f7bd0b5c13cc8d7776cd0494324b4ae03fd740cb8b538ec0352dfbce726564930a076547952de472c0c88454418cc92d1377c5ac63648013d7eb5f8e4f8c1493935b386274636aa3d365dc3e0f33aa157c69a76680c8b80b2daba67d7ae58672c623b1614f590507b08417df97bde1423b401cf7177ecac4be164da1a01c9e2ef032792b6e80cf2e5eec4c59a5bf7a1138725d106e0d80dd10f98334f768044c90cf446ce050c7d613a36032afb97d6bde33bdb428078d163e4755ddee8a12358209f94d64dde3f408d690925fd11e0df272a5a3275f79b2c2a790434940ad5013e0213f5fe92e01bddea17ff968be41de6e6deb4e7ec841ef22c89a2ba0ecb155b1e613575aaae88c52426926178b8853f4ac052b0116351c6744b4dbd7b1c98c38f7dc64151e1cb614b478a9dd03a0f9d489468537eddd4395770292de7d600acce0d04f862bb2e8100e2f11d72dbe8d053996c4e662e1ba4b9b266a5e4ea3cbb224a36f4c13f70ef8a6d775916dd998225a28c92464da8446b8332bf7ef7394f109132c52278f50c03bc18a9b9a4dadb5319b1f9ca0864b2cf63837a780efb329da6b2dc20d880096c8e15c77314bab25f2b524a9aecc3e9e8cdcff0438fe40482316afd1c376f3e80ab60d5a0da08b4aba666c54aedadbb5e921e2080e54e6ed0109a633e332f39f6035e525b1fc454649aa65a3af916b6f10e7e2014f2a6858f3f8c88da60b261a1fb8e85cfc9bf10d63a1963f6e305de38d827f508a63320556f8535e5d784eaa864de3c00ab9943fadbfdc0f5f7dfb34215cfe3debeff5b916d16a86db5a5966e62fbb2dfdcc4d306ebc634b044a970d630065c4fac0891ec2337580b48145709333bc12cc93dc17130359886887c4e2c69683d6fd049d954dcf8a3ee2a4a4049047c8cce31a387c9a214b4a89606710e7453ea3c3ad527030de2518e919287484bcbd154287ea0d52138ce3bda7a4974f62f574480b059e9d495320fd0743549ae9e83cd988206e328edda912c8dfa99173500613296807f2f1b83048e1c8bfac488265bc73cbd3fe71dbf1db62a43e3f5dead60465464553bae4e64237c423b6d2d3bd073bf836a3dc47eaef4149107467ec204718b603b8fdc6d65ebe27efb240ed003e39fe7b5252bbf77b8e33763b0f7d3be59de48d93c11ec5735ede9185e2b5eab1786520b40109178203705af679d762970b5616ce094583559ba27070e78835c68314c270ccc5e6884547c67db93cd893203ea0aaa9e146b13addf29712e41df2d4db1ecd9be77af5456704c6e21fdb75af77a4014376912e0b1b17e8104cfa223cf0325544b8d558089447430674abaa528e9f49883836f9f7a876f51f174bf5cbf2ad5c9ff32ca3e7e46fc9f66844966147a953f5182aabfd6dd37573e51b10511b260100ae6ec586fd5e0ff5b627dd2bbebe143adf4b6bd91c633486fa8694db1a1a197bb10bc113d7f33b4c7317f3d9b1e54518d4c6edcb0cb33f91fac0ac283e9aa7d35e36b9e7199ac835a6417b932d90e098a3ee7d40bded465abd19203f178e98d439119126e306ad0c6c3294ff4998eeb11fab398a5115250d08d9419355cdf0288f7fb39ff51bc91fc00c898f8185b2434970ae2198b4e43f3da68f1d68cc48692a9d9cc563e42689339db5b1447201c8364d13e164d0a73678145f44446d51698b1540bbfa9915ab7eaeca03825b97219ed52f274739124c1d8f8fb74f04fde6a53c8b6ee6dceb68f2a2aba6dc3a177576b283c6b3b20285b9524c5718cc3f0642a8846a84d693e8a936462841e4fbc46be2e4c55a09162f7fe368fd5efd47de66edbf66a77b67f44518239a9fe49cea8397d47c3f5959e6c9e50b4945dd34e0fe77daea5f730e39228556be4a492eb1e668a82e32ce29a5fe4eb7701c568d1ab5bde0e9c2d47b7a8180ca7c272a919c380b93b6f808e7cee65ea6ef9aa6edbf27ad54b22b7df84bfd3cb4d2cd8ec9a4a3b29d3c35ab2ce7b6a79dc0e6768df80a4a1d381afb14458fb2598cfb96171fc8137e76a052b18c9e177fbe0f5ce865da0f91c77f81cfd69b28913c18d6b8d094339d91ede25e72f741ba9e42b5adfcbdc7328ec7ecbda7ba0780c1d8c43831340757a7017a6b0b074a3b408430b245a8cbec3d84065ac3781350a67c4e00cb549f3edf9d4c7b26a813d7f3e6d63b5ccd4e606e40f2a45040df7cfa13a6521109ab3f3e1de652cfeb3c8c1a045faca3b2665e491bcfcc55c02ce7127766d5ab48dd0312f49f7485dc3cf3542d7b5df430e5719b87192b3072e3ce72d3e25b07bed52484a6f747d7aefe2e00dc0cc228c82fa14ac61eb6573e8e288163dd50c059f57c81062f54a161cbf3c4488c5fa6b8052a58e8169206122d490f2ed7f16fba1e543c84115afa77c06b786cb70815b0f2ea50c9f3c24436804389aefe09ae10fc6b0ae41cf8c34f14cc4bb300f1294850e42ee19e90c712f2a3834961418c7872c883a2cfc1dc9cfee21312cb20c3f63c67f0a4d9db8ef78cc4a1637db72f9037db3665a4effc0ae21ef88ac56b9078825002507dd49b3e0faf6da0717411b0a1438e71e88fe9a5b4e07cfc5a139a84555c9eb47901013b12db4acbf62c87fe78a52bda66b2cfbf6697f859a586a25d34d0e2bda7e5f2df5ce820a34f8f1445b641f9621d3373bcd99b747b8f148d700cb7f9c17ba44d35400a6ac6916810a15ecd2877f1b49e576d18aab4037599ed15ab2a75a71a78d389b68c7586165ba970441b2df7d074c9ada9d7c397e2fb8349dbec27c033d3c64a6008745c09b6fe6990ca5f9cc8a12de3fcfbd464f99f943ac30278d58c1ecbd473150334e6c4ab243fe18a684fd0ef9b8c5cbcfeb54039f56f82f0b7c761a6742491d350f40fadaa8d783ef5a1930a57302cbdd435a8a07a79ad9f96b4e9966ffab4e57766fd1031381bf82b5f636f68fb8fb0dbd5b194491cfcb66678cecb51595df7d39e3dac72354e4ecc0a064c9a3b334437f6ee7d959c54f6706c77f15946afd73b1038bb0990bbc607423d6214573bd2f4ca9de6e8e68dc0dd0cec181d4f5fe20ed766e00de3bd371d65cbf0551e81758bf86b8c6b50f09261133ab85f8c86424e4d78296e7463f19bb7b8a4f79a70230946b0a5b7b24fef7ecd2d6c7e6f2f4736ae33b73856c40ea1f4497af21d73bdb79d6117badc13d54a46c95cfa2ddf1a54a1f3919e255683be1aa7f2ccb80b5798cefddf6dfe16b06cf7f0618810303773872dcdb9e3ddfb417527e3db69f6692ce323b6b0e1ca09dbf566196f9079ab1f76d2b76f952c6c969446df2f514d0cfceae7ca54413fccf02792203682537bf225933d6bc1259d285908df76cc80ac7777d1c7cf8dd2f836bd8e622daefc73e0b560ff32eace2be5a3c42e1adf8cc0c1824499d1c1bd3f7e1079d09725abf5b535b82e531d7f43a2e05e5066ff3b168ffd023428a0ba5782c03b8939ddfef82cbbe801080608acd4e79fc6cf5a9f9fe0bcdbeab02916056aa759fa0d3b8e7d44a1da5f3f00bfd88c19547f930c3e8045fc33c0fe5af7938436719a17fb9ff80bea9064abede0062118f791e1825a4432377c773ada4582767100350490ca716422d549f8a7baa7c9192f0c1c506b63ef749d3829941c7eb6400196af426ec273200ab82819bfb0e4ca2df81cc92a9e5795ec6f4b06a63d96b1562b7f63a57e6b2fa7f6748714c07922163b5ef362c0bb1e00ba3eb958b86552f043b96ed1dcc12ca5e5debd96f8acdd70e28ef0b91f3529c9c261711ed969faf38c79670aa0ad9dfa8e66fc0fddb0d31c134df156a5f4d73549bb65cbcb2e16a78f5e5c99bf83dbe4018ba127df069e66aa7ff3f37b1e8ffbf4962753753c7b8a55491ddb2b9f50aca0d1f469c55363d2bbbb19f397661c89811629c8a96c7682714e4409f8c035e03023a8b50ac86ec24e99046bd87238744beb81ab5defcbad20d29dee9714cf15093c66185e70fef7ac93714313fdccaa07603e7e42a182ee4f8e57a0cfd2108c6bfda5e14d004a969f96adf8ef77fb78a23c922f7c763a448b90ba8a504a1a8e718434e21f5fb2698245105313b81cc9b8536101ceb0ee8e3f3ade96f351768142664a90ec008a584d475d6867a54e436ef88c1c2f0a0d52a82390c80a941cdac07e023ef2c7812d941420548c04d05ab077d0d638b214f530e9c622b91ec0f7685464d09fb8cc34a2ebc3d18b74b2ca75117b63e1774066c07d5392f4381fd027281e6b4a83eb846fee1ab1d06108bef66073359c7f64a4a0f13241040bebcc65fcc4415f69ee773d8f938197cb46d91611e9077128bf2b3d16db389608ee3bb8489cc5235d85ce4727dd785b51b514c9ff4ce7259c3c207ee948d0603659b861219f3847e5dc083ff87ede8e0f7d75f68bc6f76445a4283e69a782a06feaa89844169da15a97ef98b6d965e576f7acfddc498277775a1e01e905bbe15095e7543ba8e7f5bca9f423785dc9217864d95d8e0603656b06a7b477a44e4716684604185687252713bb222f0b84cac4de79dace9de5e600f075eba76f91b31e5e3f305e420612d5c27f326f485788293ea6569ce900db88a2d5d6f700e734a9ec5a784a0c6c72bc46576f7359a0a47f838b8ad22a756ade8eb807f692f7a097a934316db9a203eb6b2527ef4c0c49c8d0bfbec3bb5b029bcf01074999da66828a3425f5b23a125bc6af83f7fc87719fa4d0ac34f12b275f805d3f5e4afc6a2326e6e4f1c4e01d7e5f2a45568737f8b4f32fc6e03ee5d053690b1a9fd8dce936d91b1a2c0dcc14f59c29dd34e9e92d0e255a5266ecf8edd58bc702fda514af7ca01ce22d9cbdb266e1794702daf5673f345a62d91a033adaf733d2ec0f28afce43d4aec485632a687a839cfad9469ea89a1f95eaad39b7d442df5a1235547017420a6a4f5206a900c68b4880fd699a11f24cae60ff7148351c6e56e3e96d7d7b1e90904fdcc61943c9dd807a082e9f256ba62f0ab1fa587c5d5e6991733ae61969251692dba168ff3c110150675a336b385afb41e6c09372d50a2fa8d1ea4a49c6a7f5a9ebaca48e513b63a9e2ab96f770087df891debb788544a661bc18fec45eac794da3fb54aa535c8b237ff00151f05b06b8533e10423a30ed2057842fc34e2299d6a925fb67c98fd9c8e75fd650815dc732ca4fc29d0221e7447650adf41ccc2d50824dab90f320a5b8e7eb39bd4e7c4f8e82a213854f2369366e75f47a000a904d78df1c77ca663c7e0f768f8d87b9a68d4319e62076aff65f24abee279c13055f4134a578dd7fb110cba9ffcd6bdcfd8e001c8f316de67c105abb8130285d965cf611e64510699ef49a5171d0441b6dc3ee683a021524dcf1c868903e3bcca6b6fa50bbd2c9d8b90916fc6bc3feb2ce5fef1c402b00458f164732798970f3456fa062f381ce052d44b35428d976c2bab931b46b16b38ef03f6cc1a04c21d90610197bfbb6bb10c38ad2ae5a2079fdb417d5b351aded2b58517efbc1d42af8e785fff1a71591463c5ee9ed0b6e81ea74f520eb1356fc3d8fb5134ccb9d5dc858a89d92835549d988244bcf24cc15d5fa3a1f6e98003cab0b8c1f51b324a35627077e8f288afa9b8b5fba34d495544d9b8c720a046fee9a5d839dd8b41a02a63f2d2f38554afc0a369e8b5a0334b66489ca2b46460380c324f6a9ad96d506c8d07effa8c96ecc212ab85312968ea686175da67c0638d3b9fd34f73cea78c50935a937624e0232ec85674cda7b962f047c902afcb3a4a8bbe34c7ed2865fc1204f48b2f8c8de88689501c9fdeedf01e46c74264751383876046122e195d62fc68507ccd4d1e721a212b3454978a7dfeceef71b3fe2bdb1a6ada6597b5b18ae3fb68a6da94b4d4c1f76e0677caedf7fffe043a3c181b2bd9c46e39ee96ba6abce6e2c028d6a8555b78a41e648012d90cb82d142d537cee910eaae7bcfeb0e23ccefb8506c17e1f5615910e18c039e841b85b377b03eb3d32186b824dd454ac1afa17aa537af4ef60d701e97bea68593487c3f06007977893d9d7e8c2d532e7a4c58f95eb0c0c9d4d615593732bd05a2dfe94c39e7a84bc3494058fbe469f21936de93050a6c626ff80f4ef54467d7ab0e033aa19fe8dc0a1c6ce8b9e7f73175b617ec5ae6ef024177269af1edd9a0c7b2fccef07510e7080b43ab45bd60e02239a4c581ddc5ebc59d6050787a0a3b9957ace735514e243dac989f3f5390f7ec9951748fb02e1f4b9b38ff4f6bc0a2c88d0104a55c95eae0dcc1bcfa7edd5157c2821762dfd6792d8376593c087ca4e4c4b511cc41d98b4bd0380dbd3ca2150983556300b6d0816e24d93145fddcf5e265745d7d91dbf597c93df684b90d2db37a6c9ffe45a1cc5f6615fd3188d10d06574b674a00eec5cbdca7b1e520abe1f6ea5397c1e8035cd866b66653b8eb2be3aa4956e38494a247b2b24943622af33c98a507c615bfc61b641fa133a74a9970700342e379267f927b4416ca7b63bda83fcd08578b4b25ca18d9452729d0ea8a4b9c0d095eef721625d2135195de30c714cfac927131001131ec116ac1d35a68b5048f425af1581539f2765794da06857566a5df549008ea49a2857f2262b7d16d84e0a450c9246b77a9e4faa8f560ce01628480d89754c2ec66f8e96bcf514b1966e30ddd73cff13992ef5e32a1b7c1a9baef921b422ebd431b0d3b09f8ef1c24a53eab0f7145484318fa76ba0be8911bfc149dd08f353fb022a2bdabc7d16d3b073336e3563b642fb9cf14d911d5a4a7f788a2a7572a664b90fd47162c2d308d3e1bd79ff0e32f5d213881144541c6f33aff7fc1985927111ebd97c0cf3f5d72050dcec64251017f4bdc6177edb7ac2ec4d3c72c6977a968b38c6347a2d650cdb62e9d53a667dd75e890815bbe8788af956c098ee2bcd9f31db1c10d04c580f974dc4ad9e53e954d0ead8332bf295066c89875edd7016b9d6316470ba446fc826016a245ee06c333bb8db0986c1fe42f44dae3dd556f7a3e1ac395eab4b9314d41c1600fae93713b23988edc32dc43f22fcdb43c33ea8d625fef9e0bb6568d761a8f0136103d28a8f550e9523353b0aafa1a6f8654b460d053c077ad8129754a60bf770afa644b7581b5892d4cb1d1ebb938b48f9d4583f31db7ea9cb7bc590387a3e1367e88bdb283931b3559e5ee5b5e99a16b41594d4b0c6247fa5210e0ebc419873a537cce6fe09a57b2d76fa6fbdaa4a208ff84f97eda4b7fec073fe96a50356e0fd559bcf44ebf24be3880c0b8e0092009fa3cb0df306c7f33921b9a48c4c1b30a4f8a9b414ef49bb0c140e34c4f57620b2262066ea0e81ca8ef114436ebea20c2070df4159d62ae5328c5feb3d0ddfedddbe94b728e44e47c792799e72b0b20f5ec1c0551527fb59be7831a125c20252fb40f8b9b87e48e2ec73f6bcf3608b419d832e51a0be68b810c58699dafd8f86e89caee53ddb13d163062642d34fefd84da26f02e3840440a2e1b852740ed4ca7684bd8c53f608295c8df80a238a471537bf1c9b1fa6ff0b73ad4c4c9e54b882bb294e6e2a787cb6303d67d162366078c7c084ca48ed28a6d8ea8cc3d61178814d3e99d6dffb70280f0143d987e553eb12b6850d9a5f6582bb01ceee8c46bdc92b1c70f98cc07dbaeaeefa3ad68b0aee97e97b80ae20aa6f66d5d6cc39b55471abc69570682cb1744eecba3a034a9b66e4fba8cfacaf12793ed4a3a59ae765e2e21c37208d146cbe98193461e7b2e1e75dd972e3ed1d231e07a963a3002286f1ddfc371ab1417b37b92cfa009454af9a9c9d1a58aab51a8fdc93fbed16777b1203093bab460bf06360e802c0cfcfca154cdf3e719d032c84375762e31068ddd96ba03c025f93677b591e1ae76fb5a7448b92b788143178cdec342a9e3af43b6e462a828e9457c6e0a6872b9133089eeee0043d775b836820ac53d842abd679c0b515b84671b1ef50f22a7460953618bf5c0af9b53de9c5cfaffa56b723e38a151223513480054ca186418350fb8d0b8ef8b314776003f9498cb8de3ae085f9e14609514b8f452146df46bfd4cd8b1435002f0b2e5e268e7ba24be0b7c84599df9dccc10e477710757a47bae657bce7076ba64b7e69a06bbd68f2a4b4bf14e7ed20efdf18a662e623369d74490e9a13617798670152b43a3040d481cc8a4a88d1d56a9004c77a0d0cdb54fc40cae091b4ce71142e0d3366d2eea9a7fb7d92cc5797e76eb4b1589ca3e3093c73b6c433b9fb7cf3e59c4296e20528ba89e1892c6fdb379ff2a9df0b0b4d98b0f404c6c2876e42b94a2b1f9809f93f394c0d299e096d3887528ea5c33824abe44b947b4de78d1a6b8a206e078d9b9bf8dcefd94f82eb2b17106e4e3557b110d22f4a57c1858742d23b1b09018dc2dd930d5ef8ea04307682cc5c58a8b0624503ef18a815716f7a7604953b7f602ee6ec7af77c032200efe65f8ce7a9684604ae4041ccdce0c1f3c66e70d214521a853273f37962aa30297eec121927bd27f8bc13dd4a022d4f17f45c7be943d2be31a6f38862cacc10831c0c3d9c342807c808feb5d06b5dd78d9ebd9af0f69b02b0e202fd5b6045d0d5dabd1f6ab12dad7637f86d5e49a778d94095b5d8932b9bc59895d490585e421fbff83d4992d2d76c13d9a9e783f1a7c87230945c9468821fc0be220d59e983b1aef1cc7c02708dabe24b6b65c032df981191c02f07f3fc03ecbadd7a43d21edadc576993152161a468115b6b7c737a2eff15f20e7fa43b5f93057d4b7966188e118a43ad7eaddd63f2eff5ecb2f97eea8f96918245567c24b1bdbcdf9bd3af926a165681d9ed8b1f658394a268609d4b056da58cfae811c9d2d6f59d2a5e0a33f71c2952d74958f583edb5c142f08656d3b2a9c57c01b3ab58aee26ace74c0a114ac423b8ea3550c0ead5e2c9e2fa7396c3ea69e53842a7ad845671f32aa5e8c89fda9452d11a2f427ca35c73cdf68442b75496c93fd35432692e624eccab4ed6756fee407b64d8e00675a2aff3ac3c09e153c12a45cf1f4ff996c8685914c26578fa7db64bdbfda441933f1747fe3d17bde47b85a97ea2ed53808b380c2a5d14b86ca6a247f624334ff83ceba33451eb7d6c84c3a6944d2c59ac8a0f592f88fa1af111bfbf37ce02914ccee09e36059d504031cb7c4129c0505ef25b8422c105a163fa5ccba297eda4cbe7d7e42837b210d2e3e77f2bc52e7266e7af1b5906862f9d4e1f7e0357cab17bfac71d41e94596a5ec036f633aed24d11069e10702cc209b9823734204423773d261709caee690ef53d243f096f36deeb1152df7150687051a2d168d830c9e7f1b2a27f93461e9b43730f022e3a9b6bfbc4997ddaa3e7249cebd3ec56c86cd1621912f72aaf9eedd52f6e8f7f362342dd095feff913ff11796b72887b24b29bcd32efd75ca486458bb65aa9eb81dff671571009f48e837f986955f2a8d9a534c2b15362c3607dd65f5c74fc921f86694c6d4a86b2c50be79593986f7243726e20f127a369d0531568800716a49761efce19a0c065c117f625f81661c72a866a43f7069f40422f468dc0163a0645dc783c50693e5ba637d9c9a5f256ed414ad8198fc37a67dc0fe16dd3cd9aa90d40b09669cca1890b2cfa2dee0a0f06f917d7ff567cf457f8e6e9fa12303b653e8d8ecb5158e678be76fdfbd98a3f681c71cbf3e5e0e533ff9d1c64bd31971a528dbc8a6033111c17ffac2a5790ee0d8947a513d956572c35072c11c05368a0ecd4bf2d7608bbd4586275483831fc084e84fea5cd962c1e83d180f58df6ef116bc3f19ab2b4d7bf8d08ad6ff457bdd0d040dc3d0036d7500081fdd99f4fea0928767374b867c7e627f3ee1b472e1a0e964c8f04cb64b486ab69e7733bbe31816f20cf4313c132e6be999309674ec115efb133d49a307a5d19143bd1a347a800403a6aafe029578d18712f6cc96abc90c9e67d5d28384be2adb2838453a0e784968580ccb897e42f3d06f9db359643912a292e80086978847d9a0bc40812a3f2e8ca0795a5a82d65a6405a64845904971f665dc280594e4ed73071ebdf4055edc7696c1d37ce91db37a9f2f0f9e76e9a50c73aeb8bcd4192c8817b5b730e5727af9916bd15cc5fdc7e8877b1c5ac28ca5458ad4bbb3fde187834589f80cac9d114a89591abddde7658596d3c35ca6498bcd62adbce04de33a02710f51c632055dff270ff0549a48e8a2003717c6aa2483c86f1b25cbf5845d58f9d0d206d43023924be950238409ae8161cc81e5da13d6175c60a1a0b49c1fc7ca409ec144a86d0e8b9728eb535bf4cc13f7ce17da3b5d1f3907b998a35bd07c269aedcc205689c3fff204e0cc0a1860828c25b49f2480c0fe772683a8ee9b18d1bebdde44b95eebe482e5a28ce4f153052ff388a967446ae27118f9fc2289f60a037eb6d0b87deae467b1bf189f3defca9885c20880d34ef4a0600f93ec306b1369341764891ee3d9e6eabd70060bbaa88fa5027adca3a06f9dc3bdcac1d16affbefbc8db7327067ba5bb649ea182310a08f65c6a6ab9cddbacc7251b97099582efc7cc5ef33f971fee69216430be3e270084f11d72bb77e4b5702201664ad8dfd65cf0ed53a08f760a7958d15573f3e29c800fcd949dac703274cf26a6b4ccfcf49ec262613174fb1a923b654b38d3ae46b79611dfc76f403a4cebb5e0a3d36e305d95a7f750da1e314a74f947ee26ce02269a7b5dd3f8dd6a9cd8d7c6915e52ea24bafc01b488abe0180c1980712cd5a2657e48661855b43ebca0d2df3417953ad2560d25a20c4ad344af4effd67e6ebf295ca435c15360b067019d0a392839c05ec765c29a882e06df086e059e6faf03c86c1369b626728c8eec3ea5b5685f68829968055246f99741c3cc2dccdcce7b153aee5b3795c0edfe09bba2b2c468dc97ccbdfda5eed83555c97ade7fb88b1d8f96a04b999e7b468dd3a091e51808a6166d8ef0df45d906a3627a2a2560db1379a212f159c55a033c9c721ec9f75e582eb08d4724071f6c2f65f23676f58096d899d193bc26db064eae62a2a4dc2b0d6dc1921f3c146254a4a7368352fa9f8ffecf3c4764b24620ea8de59eed5651fd3f7ed7e087b682f777ec31e5be6e55b173c2e9c32e6c30f3b042660654542a5ce640478b47a44495347169283c8b12cefde4ad03a3f86b595ec8b6d6fe00d7aef49006866d7291e6b28c7540b9ffd98989be2e36487350ecb9335a631866524242a16c37ac9e93c4f415aa7007db203eb310502840267e6e94e558a97ff66c00f01fd606e1d6ae369983e13145874fca8e09797a654d3587dcccabdc69a3df3c8da46d824a1b0812f44ecf2bcb3a3b4c1cd00cdbeea5f9a1c4f7e28690cc0ccc89d13af7404b38cfa15bf31988f206348284bffa3f7a3ed1f1dde0bf14730b733905ee98d6e5e848063fe94ab56b244beecdc916cd0ff3b6ca3f5c9b09cca9dfd0b43aadf3d695ad73006c115e49b9bb8b33e36b6a293462bd021f741813ffe1cd0cc801df515a9b1a8a46c7d67d28547c4015fc218c984d24dcb2e6a9a595793a2f3de07229738577c94cd0aff8e745c093089c0ba48870ded102c85369d4099aba458a86e44a91f5bfb14ff64583472ba95748a38fbfb65dbf438080d07ffef9f87d2d8e5d474cf270cac60b1b2fcf15ee067f7c8b19dc90fd57cf803a68281e9ce0daa646e42352fa0d95ce91fbff22f9115e68dc757325a26b107df32124436adb0043c52b29577e67b118df075ffacc4cff0994b1cadd84cdca3c6c1cfba66a7a49bab0147712091a36f9a4719b59730071f426dd200bd1a362d159e9800b96904908ca3eb2097a7ea5e3c20b7d81ba9642957b2fb1e3b11a15cd00b61b18385b6ac229e83ae531e5c1a40fbb8e5e294b04d82d6af13dd071cde6dc3e4999a733a64e895d142ad71b354c66da34488bb13d582cf32fec50883458818c3c8b60503bd016d24e5cbccdc02b1d0a4a9517617ce8d45dcde176657c76202e7e98b149e95b39feff175e09b521c717197a456d7809f282180383c2f8f90ddc72bf96b845adbd2d7497ed5eac8726da64c4b5b10bbe660b435d87007d9c18b8b99a1d15ce17e4ee043eea0c25259662904a212be59f973651736946e591c99d19e76bc78963eef161aaebdb13ca4d026a0d2842083abf5f75fafa9fd966620b79d7f360c65fefcc5e5b5274e860c2b40d6bcf2726775520c9e2af57f40adaf431d05617604527b7abf8211c68fe37e7f581141080f172e7aa608b0610a251ee7baf9a8ffd3d3a9758f03c9a19ff8abe8ab21d00d2ad72845fb81981da6322a0bbc704ea4175033798a136471ad3aba3b5f43a362bf23a1677910a673de597fcc737027e7bb865392e329a79b7e9e3a76b6cfb2818d54c01aad6419d6b7b62438175575d95e9f0d7ede8b7ae2609a27e56e097fe931bdb6d0921e8a7e8ee93d985fb3339d9204d9216d70c4dd1a2e4854723a3dd2adbbf881f1add0e54a02d852e27e046e58799fd2560e13faf84374e9b19a323a4d2bc6c5e750f929d83d946bcbe8c8f0e616e856ba619a45f78706167a39dd96c547a2acfadc912681845b2a6913453b70f82f5433e36a1a18d86901b274b6bde39b4777f15234a5e173a33758a19559da0f0a5e7d2c73bbc151f99649a513668399f1959c062f82d9e78530850ce5ce5d5b99cc594c75bbab56f91425f378721d369c2fd8c2bf66fdc0f179bdcb412dbc41a672859061545f1166a9e1c8690f4672d92e438d2095a2e6436974688d46c8fb61d2b6de575437c85cf9307efa6b9cc7e64bcd399e43a900ce130b9e63987d8584ece04ce78f7f4bdaf822a0d91b1cbc3ca5bedfd514d4f0cbecc3685eefb39884a72e1c1832b4f033ecceebcce2bdb4d60b6c0bf2f1c930b00cc6eba1cd06ca0c86464f7dd8c3d786afcd5fb50cdc4c71721dc5a59a0ee4692d49095b411ea19cd84989ddde80d4f9ee4b2da02907af65622172aeae0232a3a21bde9b36dbdc55899954850d57038933262d325c69f2cfb4c3b57c305597b2d473f67225e10c38a3ddca829bd2a805df9c137f2fdc3d06ee403872e03a7359ec3fbfb8faf405512bedbf78964adcbb4308671265e8b47e59c0716ca9203fdb06c5024346d1bc99c7569f5fac5f5a308fe873b12a128d9a83b6ada20ad537f3199f841b1df8cb91a63986eedea6819d95756ff77fa2f96f9a10c8115a152b701daf8e6b030c5c00e1ec9f8afcbd70423d5438bf847702b37dc6c95f786f591f68003b76c777ea145481a4b6f6d9bb8833309c7457fc7646caeb7be447f531671c805db154ef615c48557a9f773537e329305ce314186541906a959d59496bbf0167ae0b07e13f9afc7abc3fb96f03271d27c34eb98181f82644c3cc0479e397c377916cc3866183e7dfc91e2cfcc272804e872d3cf0a0743fdb9fc6b41a0526a82104a0ccc4</script> + <div class="hbe hbe-content"> + <div class="hbe hbe-input hbe-input-default"> + <input class="hbe hbe-input-field hbe-input-field-default" type="password" id="hbePass"> + <label class="hbe hbe-input-label hbe-input-label-default" for="hbePass"> + <span class="hbe hbe-input-label-content hbe-input-label-content-default">What is the system call number of Fork in this lab?(digital)</span> + </label> + </div> + </div> +</div> +<script data-pjax src="/lib/hbe.js"></script><link href="/css/hbe.style.css" rel="stylesheet" type="text/css"> + + + + + + + 2024-05-23T16:00:00.000Z + https://fuuzen.github.io/math/discrete-math/graph/match/ 匹配 @@ -2340,30 +2364,6 @@ l(v),&amp;\text{others}\\ 2024-05-23T16:00:00.000Z - - https://fuuzen.github.io/cs/os/ysosv2/lab5/ - YSOSv2: lab5 实验报告 - - <div class="hbe hbe-container" id="hexo-blog-encrypt" data-wpm="Oh, wrong answer. Check and try again, please~" data-whm="OOPS, these decrypted content may changed, but you can still have a look."> - <script id="hbeData" type="hbeData" data-hmacdigest="27e5fc65ab2cc7099ffb20e8d7788ddb6c04f28fca90259dc262b30be6d46c2b">c30d89177eecb6c2bd681297bac3c03f60f6add349b92ae09a93b0461a16f3ad8579c0e46ba30d7e45c4eaafe84e25352daeface85f774dd3b167eb75202e1ecac73371b260178eeb7b0705fc541d817cf7542782552d2e002a0645bea211b0a62fa00db1d5c5185646e810967510cd86567114798aa31c47727161b937bcf20c5636265b3fb3a761f8ae41e9888e34103f7426884f2a0f8b2d6ded5a415dde2931275f4bff011798754cece2fa7fff5184156078042a670116312b396969d0dc2155df655ca849d37fb87daa1cc6ebcfdaadcefa9dcc5047cefdf33c27f72ef527260cd32c23b30806b25dfca6d05f857ff228bb61ea511cad56978e35464f9aec4e450b5e30a8058cbf1a0a6606f845eb9bfd5b8ab0532132eb7e3f45c5fd7f90febafdd306e657a2d893fdaf45f2363cd2cce13f9d80670513f3b63d98694b113e5e029b1c0627aa9c6842d4bd3bbdba94341c0e373792340a525590c596b51b10f203619ef982d12f82921c73a52ad975d978967cbaf00488383024962889c057fc2acdd4d0b9eed466cb73739356cd2c21b5564a91363d0ea7435080a5c05c6f72d20202d80fb2c4ebc0dc822906ec2ee57526543d2863528dd05218a709e1171ccbc4f174c2ea716988bb72e6c540699720db1cafddca7ab148293c130530fa536ea5e9c47db70d25f198f659bee33e6efc331c5c1f1cae5dc61126cae4e65fb5727aae70f10cb80b77b035a9bcd6711461bef0fa5aff0e234e8710e98739fad56682ef557eb19626d2cffebc4c4fb3bf5074f5e8f3589df923f011913f9af411bdfa4f705623c9368965cab9ea130a3939291574b0b4dffeb2627b44149c2f38ae14dfe52ff810b8f929ae6108f13875fa1518c592106c88a6486c01f2280576a43bee3ac049a26d35ef3c1b8f22f2721632551f770fa4a8a42c0410982d83628cf926019f5a89d4ccda9376798402869810f195cff3c8f6a611d5d0843bb4d2216bd1d3b3032448956b691f45993a909cf8b078c22b34c038220443a7a82592be31bf361cf225363bc91d33ceb638fea44aecd541444e07f6cca11ab01751b213609eee6b08319ca21b78aad83a5b896e569cc0bfd3aa68ee50eabee389c4f4dcd40aeecd1ea4b3382ebff14d620368be67760eb06774418511c62e9256f32cadc1dc5444f0c5e1053b007a06e68873f0c9adb98af8dd84923d38400e475894ad80bc404459cc5b8e3886266c8827ce14feb7aac9de594175eb6b3d526b03cc5f0dbf1718bf1c2b8273e8575f7e935cf5e56489b534d650f2d75333d15b463247ad742b8ed95d3303a300b7a97c8d7e218276ab1005bf44be5d7fed59b55add0bf213fe7dfabedcaf05f88285b9196df13d5ea40aff6761ffc56e56bbd80a66e76c74ba2d1760364b53377828b7ef5fbf020991b1750f0a34360149f7890e37fc38cc742009d83c5a65752e3adbee1c134ce6889a86a6e5d7cf2906d558b4d93ee3c3d81300ab8bef487dcaa3b08e148c6f73427d1eb80d7123d260ea35680508216035d0ffaa60ccdfdaf54c01a93ab28da97928b554ba0bda988a048e2ac8f2f77e3c89ab519041e68723c8ea39bbd81236c3124b4a6ce094e6e2fecdf1f9245b2ec6a9e9f929004012e597e3bdf46ebdcd55ec60d498f6b5c53340d22259176614eeffc2a29f9b495b8c7e843bec21e90333d35b59a24dbe3cfab74a324994062993c2ae8e34b31b90780164e2a69a280bd02a158d16ad2ee832bfd6358fc5defe74acec55a4b4fce044c5d001214a2c3cd5314ff453c298062086032bbd58baf0503e349073ddcd86eb06bfab247cd46668322be4d563eb55ef6d76e211af745b0792c79b7938a31a429611f0f9634bebc11fb1a9ad55fbf821666394efe2eb4379b144668a4dff25d78ed9e3ee9027757b9e868ffbd59d688442b748ac828fb5424c69e9589fe21985e9a90ed6c287bf1391edc25eedccf2053dc8e033a83d76a6875380ab87ee7f96d4e09dd618e59b77bf14057d6fd175ecc3b855d7471f4ea721cff0ff589fa343844ca80ee9d52ba869080cd79f16f813f089706e05f1d6b876e165212a66afe40f5d802ed6006c2511339062bdf2fdd9b9a4e744babead284748806a7e6467e9c5b0fdb1971b02aaa33a998f0339e92ea3980764511917c3a4d2c4b948123b76745a12a7b3f11f0844e3521f3ff17a58e34d0724c9935c127eb15760b8c30be3bf63813417b9ad5224439fedf5fad493a335bbb1da54aa772bbc23dcf8253a57eb99687b34f5b25cea18181dfbf535afe3d4c77de8190cf44782b2217f078947eee55a5d954652dd0b01f5f47c92f699db42a10cf316ede97c7b2e7fb358c10022c3aa8b9e64ab7a1078e7c7b19a34846746e48ac7bcd28aab0376ec0d3784e43ad2a0c25742daae0931a26ecce809a4e62808b9fae573c0c344fd808a076ce3a070938272e15c749e2c45ee46c7316fa719d4fad8de2abfef9265804b3ae7ec0118c90997d28b21af33715de9930e6f428d085f25d020d4cf3725065452bbaa5123e68077b95bae5d92f66a8d6b3a82a21139833bf8e29383ad024be22c84550749cd5799c2bd2068f6d3f241d7e3085f5dbe98a24ccf29e759026c6a683ce76c10a4513a9240b7202373bc997d18843c7c929492f3dec3a0ecc05569931e34e642e36619c4bad80459e27c826070b46baaa6dd4a42d401536b354c1fcb619152ce9710e5ac6ba0ed0bd6d4fb66a4fb5884946db1bcc656fb8a6cef719cad6b82a922452f087b3f12723f1ccf2dcece86cd998189ffd50afbca73993ccd3dc14da7383c67ae05993519754006c25400e2145ddc90c526f7896a10fb867fbdbf6432338c0cbca9d5c4ba7b558a89ae4af0359a4f40b9592b2337af9736efd15d0b20913c683afdbc716deffdb600528f9f1cd639c440fab77c313b3b268cb0f16f9d8dc73d7a1c197abc5cce9537c9984dd5c52fdf3ed3dde1bd5089de0ee742179218cce4fe5e05a46bcf1a67a17c37c45aeb7aa172e30da7d5ad875628dba8ab88844a5f052d941a0248e3009d7ddcc7058d1a9916ac7045e1178d1a03100cd45dfa71a65776d35e0114954acc43514fb319c0922bbc6c737965c26bcbde658314f74fb2620aa0eedfccc74776682e9baa0e0fa4e57362fd29d413c7d849abe97397524f97a2463fe7ff84fc89fc90b273cada19bbca5bc9cffb32abcac3ab150c1d195aaa7626e88765ab5010dc587868fecdfd12b9b5c67b840791c419d71824ae2f65f4b84d95d27441e5cbf22aca7f0e04a0ab60319257c50f7adf7c6972f6673850e1a7b3a981727286bd017ec71bedadca15d72e734346c686a878598b2727fe967b6d05bb5c49eb9ac89f151eebaa266d822ffab5ca17c22362f6551901f992029e02259405b2cf9a6e4d721984d0a373392c729961a47eb7c79e2375501c6a1798d3c28b4b7503da32342f048ba000ffd77bb070da7f0b2f9e8cfe616c82f93c7740d29c451cc8cbb58b9b373e76fdbe0d49fb2ded433b8f67f043ee7011d5391a8d0b95c5b9b1fc6c22baea4b676b9b78e3db1d0291f8138852f1bedfa381b41432e1a8100723ae3e90a80809cf60888160d93a8e31553346f194a75a714a84d669b01fd25b59658cd193ea243eb4d75d947783d703a074645cba442ee675481bd7acc99bb16f08e79c2a45167a26c4976a94ef5b4e933b6bec74cc283ec7ca63b8efe758ac43fb7ad8247b0350b942aa5f8bdfce1791274baae6832f36b5185ea929a7ac74bda523282cce27319f3fe39d822ddad37cb3f83ec26c6b3a991b393558a3b205905a6bcd08e237a8de59d9e7572211005005d963b616bc121743cfdc1e8aae089ce62e5f25a11d33e25a7fd1697c847a5a2e4b5fef87aff7659b4e270f9251ddb9e9c2004df6d848b3cd7e9ad9aef54ff7f770227e79d71e27e668cd7041e90f7e6c03f18288f84fd6809994ed99d672683eb3fc88dab2e2173169972b65dc79f806652322a6d0ede27c797c40aba3e4d33deac8b2e478b79b49838ae7cb197522cc89b5f795b52d887ec9b0618ad54c279d37c8ba74b8530617b04a2b84553f478bdb94357b61e842244e15d34f4b1ee8c780067fa45300d74b06071aa1da87ccbdf698584e6a4893d25abb51528c9689a37aa19809b197040586c8d59dc10b730c603a501ae189a37fa44c78d9f8301d0805278e92d28d609ea9f8d78140166bf7ae1c946f0a4230bdf907b0eac4bd0a3331e2faa33805c3cdfbcedb2bd687bf8b6675d9486df5a1dc758737bacfefa301652e91f80b6f0419dcdfaed097e9c16b117ec5cccbe12e8f1c12f21b9eb218587921feee7efdc4720f76d219356869cd50bc8424a19fe0ce58ea75a06c8d9227ac260906509cbde466eccf210fdd42f595a2008ab286c8a9b30e0d1cc18be0df2b187db064245fe005b934e2145a3fa3e23a880646b5828fe3c10dabfca65b8298efaa413c4bd62fbc273417faedb29559b236bd74dddcdde929592a4bf7a5a0b44322d6cf792ad66a087d716b221b4d1699df3660744afb2c3c1d74d3c9d5ac079bdf2e2cace9335ac174b42d9b1e3e9ed2bb36c24455a49055b0b33597a4a8a8f2933b3ae9493a33688e54e0a0ae81dbb1d2f01b801dcee6a4ac28896a87333023348ffe16df1545559b9888f0574c1a8f78f7fede0db878a1aadfd0ac3f78a4e3330ba98a66fa0dbc9b6e24d463e7ff81fc0f987b84540060df60ebc526c30e6f0d874aaa96610bc52d2a5f045604ef63b581bb9c1c437886d808a6eb8a7f3fc7e26c77114ec8ae23b4af58e20ae49837d927e9801f8367ac5100e425ef7bea38bcd98d26fcc101506456054db599f6375868b593860a9f4f1d08ccebff9281ff59f571352f7bf054a3b6f67c75d4314b1d42104d31a18e77d7f139c8f72cf62368dc2b87d74730405fe5bddd45677935f73a9e957c87acfd2d39126b5e2f87f35c64c89a4128794d1a5d6a0c187becfeafc343c64efd5ecda3800dee6770ecb660e9dd7db4c4ee36cb31e5fc1aec55d8537089a3eefa033de2f0ddf7ff7ca77236d8ebf3a1cafac377b48fc88e8b0a55d3001c5b06597861fa1893d66b0f63b3d1f0ea79fc5dc7a54700b4c4ab052ec2383c2dc12399b6309ee039f0531841f84c5fde45501ff12cf9625a35b3c2c870d56c88eea81604c5e223908523d722331471dc13ca45085ef7a949b817d4cdbf606b73655e276c11bbdc8ed73679080ef8ab11dd699ceea5d76430ae35d83741e5479e975cacfe55f9afe9ed4a3faf5d80e3ddec57ff5bd979e38efbb9899b418a6a9c26915ff26f1d004a67d562fffc799fcd31bcc1a67434afda4aa7523c9acb1769f58ab09506d3731f47c80499fdbc3afdcbcfcfeecf7362e0c1c11f6ac894995e547555f26621aa788a71b5a424376cac60a69fb234c268b348acf06a509534526f64e8dee8df55ba1306b043de4f52cefd2e24782f72271fc5739662efc3e44b98b2cdb2b0b755925cf8ae4b2864c511607270a78783dd7f7565bae804d8bd5938e6ffb585c236a2f7ab38dd39e40386af3620ab60659b732397f7a466948ea3a757e1c44c73c6819462d79e622dda4cd7302ea8a00d4bbd645a86bdfc87062b8649d949bcd475a7b03ebdea86dcbb5848d87da73158d5bdcb085fffa67de084598fe9eb8508959c5acd9e2bd4b04c9754c3b0ea4cfcb4f75c132b154b828abfc7c454f5258f9ec5ecc474a765257da9688d78e694ab0104f9d69f848d0c6201e2d1a27125b99cb7ee92f53369767b4e743c3bd0f2f66de3a0cb2fbe1750ee90c0e80a95c441140f4a14cb38cc261e1c347e6655cb3063462eea914f18649fc20d256806a997c0efc40025a52b61a22971655cfb69c3a94d0813a07e94bf0933a4f990573acfce899b569f54f1e7868ba680d14cb0b7a1d075466f60918d2c9f73a8a5b73fb2bab04284a4ccb7e3c2ab7916b4f95605abbe9959de6a3710640feb20670ac5389fab7497fc516ee7af47828c9af31f59eb8324fe49cccae9abd08773494ddea84c5d173c1912b88f0f9722e9b11a4e93bc4621c5b35d5d2685948562444c5938debb5e6f3955d3588acb30ebf0e301490fd8d3f7b655cbae5ad5374f440f9a4d3adc4c85098699a671f2b86bf312bf6b3711b890989317d60a924251a1ac21635122e0c2a9acb4742741cc31750dd76e1423e1575c1284d16e973ab4cd0d86b3d9a8f846be58e40b1e590abe6f57fadc0e16c35edee610aecd2c27cccc96935970afbbe4ebbf4fcbdf234c7b1152ecd69d7eda0a62e7a0da197139fd1f01e2b36de6b5adbf69e299ce9b119de6d535a0c2bdc0c11f70489975f61c210fbb10bbab3966acf3cba321ae8b0ff5e53a5e87ca00e29df0814157ffdd07106fb2aa83c74c91a9f26ca59d688ad07b650892ede18e112ffe68e5457a7f0be9b6b5058a704821b77b8da717b3d2c0376511f373db6ccff3fc18be0cbfa698c113661b40670f99b54cf2f4903919e3313275f56b270787319bf2dc9d01ce95650c999a91895b7dcb5d2fe600d86d8bd99dddda7a77a4f25272cf21cc8530ee8ccc98dca6eacca291d9e976d5f552f903b780bfc6bdea24f8e419ef1865458d833071212adb6da557dd623e29c6cc756224303d14e448637e0be148d5b2d47d7b39c7b9410a43733c5c478e350469eba25c87995e19376cc7266cff278cc8f1715f8d988fb14b236e78fa46896447d43a57899afc86a5df23f1d19224b23b7ec6650267e0d53daba1bba3433ad6ce57fffc76a4265cfd6264bb2cdf53e3a562ee2364ed894c7e53ba75c28662fda66d5f1d36c58743d9952c65a4015670305370534073cfbc98c4ca78e0dda4f90213cf4634bc3c6ababd9716fe1069732a54e643f6d51a7d43f0891cae0d9be37457a4d7f7e290b51a36b0266a87efb4590bb36cb4b6dd3a2fb6418d0b3f063f5572a98748b7e17000610bdc25f2d7d26020a84ea7f37a575b73bd340c7ac2ad4617110f82396bafef669a1b6f5af6d483ec6464ef7f16bd0d9a67413a74a8ee685a5f79914aa86679e8579bd8f1d662f330b413123051ed104db09fb39155f3c5eba156c781a4cce73f69de7c44854925f79d9599aa75d94093742b45dfe3998e2baaaaf07d0093da415f078e737f76d1cdf63db71fe9adab6c7780e82933d375085e29df554f16e2928358b031450c1a968bfe8f6fccad8e096232a8170e6fdc31a9e81bee8e83d9135bfbbb6927fb8c08df39f020a38159cd881bcb992c97ac5fd8473cc4eb41f31361667f368a65d2736301416385e1b8162e69a312cdeb9c59575382f6bd59cf6579466917a385913880b5900ebe224082f640c805caad033fa7a9c540baf0f9b7c6c8138828d830f2eff29f785b0686b1ccc9f64ad6a4cbcd5e8eda297dc3f4e852c3e7a08dab85084ea31f346e89507a98bbaf2eefac354fd23c68b7b4da35af3527698b50624cbf7bbc6643f7a1a949db2bc83d7a8b9e94d5bdb97366b2889d5d89b0d7845126664fc94bdb73bbfc59a84c232d6bee5520e124b31707e840953935641c2e8dd7235ab761d62ecedea164ec5aeeaca960bffbef87d2624dfc443ed9cf8eeafd5f62dcd9ecceb8a8c2d9de668eb8fa26f59929f4528ece92273987fd3f40432594831695212f412aa32a7e8ca281f4b0dd487e07780977921d56e74d8e007089ac62f035bc22fe5c10e27d4f5c1d60bbf925ca3c71c39fb7ae3f11d513673a0f6a56e28035e2916f0f211c7bff04a94e6da52b6609e95a5cdd5c11bd64940a9203890dce219b188e00eb2e3cbcececc25417653f500ce628ae4b96436330eecd8809062e785fb1c0b11b646817a0c95cd83ff37a9d84eeece87315519f5b799672d851e7ed0da0e28d1125a23ce9735655ebcfff2126911af0127620dbf29ed7d92ad87ecaabbe34091406c2079a6ffbd78228658dfef5d681d2b9e7854bfff32e628ec3659976e69288ba3b412293e6176ea26912b540f34a766c75611a7f0bbf7e1481fabeb7c7158870036d23bf00e2cbc4573ca6ef8fe87e0c234c9fb642030499f2ed377432b8ad4867d071c50a58621bedeb6ca8d4ff192a6db6cd3ed45f55c53a97f2e215b287d08b3e54ca4b229c2aa6ad3964401d785e803b59197dec987c3b0cabaa96d0218c1a7e2d2d358d9ae6eae2b6a39f4c5fdf283dd990c44d574a01574e6220eb48f1fe3e60fc0f6e77ad6899c92ee3c5afc828b822b59c4ba1c1aba9c588b34694ca38887717138d4a67496f83718e9450004f610ceb970db7af49275d740d1d1b76f7b80b0dbf681e640ee9bf1fff48c04545e85f54ce077b69f63eb830bc73ddba0f5206c4a813248e870dc3e11cf1d8044fbe841401c2c39297c350431c6ac88f9f1d97934bb2d6f2a47cac865d21f9176e4d13e86d4f7d37e5d6b6dc599464685a50ece017bc1f52c795d796f443a7d565b159b114eb8b1929177a8dfde940115ddcd5c51c486e0a15395f3a585dccb14dd8332c99d152537c04766c2d8b199935f76fb3b4fa0aeeee2d962acda8afd0a12e68ae584d6bdafc05a8f3477d9ecfb4136511b0784fb3ac029a1166479f0da21778cea69faeffeaacae478864db709a1d7a65c5e2124272c47af69d7e039853528d555ab31ec4d6224fe9b3d74d733a2599f06f94542359d3e82087a3fd3d728394e79e4606815e8c37d2752b1840e1d78729954ea88cff10b577865a251d88a8b75499d28afbc1353052c0562887fd89c6574c55a70a5b1986b9b949d3c10a8d05d82fef4c8c597934d6ac06c0d8e6331021b7578e4303a123384e62b0267539b951a54b370532d2e0a9aa6b5f0ec1e1c0933a6ff8e34fdd8949d395854fc1d8e3e4967847c784ec5ce3aeaf5b381bfdda7c9e6a220c16104ff5d3f3fd79bc23b41baf6714acd03a9453a21be9e2c5c05dcbccf79ba752e3bc6c06b419a8d754e992bde244289b550a54552dc04477f6ebe6ce6eb175e96c7bf83e7b35dd6ea365ed9a86eb939cb2fb7dbfa8ad73617f1e5b7adcfc0e78b7aca240082b6bc11033fe9e08b763d9511e935899cf4abf79394b16f76a2502b865296d6ae0017433be67083e736f5593893d6b2672343002826ac04f57c13f79a2735aff16a8ba4eaeefa435dacea0efdf2384ba4a37ad4ae4af0452c7be8678107c9d45f2529ae6d35863d58f5eaca86f60b74a5b5f05b9530c1d4a07c9bdaa40a155cd216d2b25465e26daf8f1af3f9239386862328b52407c6d5ce08d41a99ef5f08948a7ebf282fbaf3a002ca7052fb30beeab961834f9ad994cc44383c82a6eb7d20ff83478ab292578197b2778d75a0f03f82f8ef4424c58c6d15968ae94b6d34b6cdc8affe04358b7621380a63b81e073b5b09b376354f8b0db80075902ceceebb715a7bdd6bdb98f4c037b19966f310e065c53a8dc43167ee8f75f37bcc232e902720cbfdf868d7bb78a82f72b8ca70cc395d9eef310d4f672aad8d996c2a548714e5f74d497cc6af9e764fffc5235c51f6e7baee2ce05fc3d4ba7f70ac90bc9858294b6362edf36a54f26966604ab37cee697bfe09666c687a62ab329ab4040a2ccc5a1ea7336ffd60cbae01b2e14482e7d127479786d4a7866825454160649e9673b8304cd0d58cb7f02cc9be67b80220a3ec0c37b46bbe4f21436a4c480605c23315cbb790f05e46e2cd2f498b269cc0377f7e57b8e34bb3cffcf341dbaec9c6c9c25698e3add17b4bf0692e1cf221d6affc74fc02aef529e6514150a6a405c82d4b4178d94824a61b07378cb9b83e1570e2e82de7b3d28415dd29725f7f5479d5629d5125047d33818bac98e52fdbf625b85b4ab849e719ba18d3659467d66a222551b650571dd1e33b58121b8fb6e7f532cbd15ddd3a8126c64cdc3e2cb97cc8cac626ad8781671f708ce2d8a19e02ae3050dc63755832bb375483d2c04f1d546c7bdc34f28c15ee42f6e8158344835aa4091fbb482ef1d0dfc4cf8533ae8ebb8973475b9366089a7c7531e4d21e389fb96eb6ab51fa91258864b0893caa42876398a97060e8de295cc0c2750a0ca4726cdfa555ca4102cce0a899055c3da6bfa5efbb8171f1098b597a0bd13b97f395e9d58bb7b612a72f6977a57fa48bee5cf05d459f5e9bdabe33029f4dbf091e8ce27184286fd92644a42b9e9889bba94550cc5b2096176198c5d955a4bca259f9b5877f997c83fa7537d055639d4fa0e2e5ed3ec1ee6c6b49d9dad4fc5e72e612606e64352d7e3a78257b1caaa5380069a1b602ba4c853457cb65f9528d2dd2b40bad2314201c22b72d49718fd5adb8c82cb3b4d51c3490fdde9b332218644ba0fd8854eb626984dc044cd5e82c56bc3c6de11e1e42a64afabcc79e68a60a4de779f5f70888cbdacfc8d4f512e9e6ee3119e241dedcff5b66864af8007e8de5763ce6fe05c313c2b7e629983122a31ef6f59c875096e024af8200d18568707af1b8ecf7ce15f0f827a9581f4519c53a9dbb065c357a687b1987c12b7da86d3f430447d17c2987836db160b32cbae1f34d68b8a7cb1f14b8ce56c7b12fdc0da24046bbcbe44806915408e514e4268bb5cd1b07a4b41f9888de63204b1e712a3e081e3c2f43ec0ea09bf1af14622ddbdcdf869aa2428e200262e2a4c17d73e670f4e3b0128b601587f3aed3c4a1601a89aaebbe13f66450927ee754032589d922ca63822353958a7605a09c5033690f98bfd335764ca14eac571efa8a864c903c0ffa98453ffa3494a1a43579a12129ce81b14b84522faf97f4b444ce67a4ed210e060684325fc6d4f104d0c65cf1fc9450fd021ef83cf641685a464a620e4060fd7494c55b97d7edd501622fd40fc1986eacadd975de24a89ae509419e18569dbc899a2f168d0835e4701fdd15e95fc59afc448a91a3d0f8c07c68bb2b6868a3b204bbae051a372d04feeed76c155e8929afe4d1c445fc61f2aece6dc6b0ca54ebb8c0c31415c7337c0ed247eb292fc2b1ab499c9d76db22224623b0704e4867dbd5ff67fde324f1ff2c5d089c3ab59bbeba120c1d3225520d5b83e09f18f36dacdcc973ed2dc063b7bce1c9095843857fd5913cd8d90675f0bc356f1dd5b112f89c340f7faca28b74bfecf311231671f068c5a62e542ee9572cc2929756d99b4e1aa345d8332ae946ffa4b6d2e8b9e5d930e3371e3143c8b4c709772c60e36c31ec6006225bcb8b4482d54f2a573b9066b2624fea8909245f4eab0a345deb52f54249e9a0eff6fb28cd9ef244d6653fbed93e5253e3f480077676e715de68a0017e4989b9ccc045b7129580a2b65ef2d1e9be65dc2ce94c2a5d88e188358907c113dd0aa3ba1bab54dc0546b1fde07dbd85051c2a9749561d8d2ed8d1252a7fe53f0d6b8e0806c7ad64690119179fdaf78125bb641a578d6d5bafad0864bd8dbd15c9766b0adf9f53e03cf239fa6fa00b583016edc9a2f23532cbcb094b21b487b3d034c878b5d58b2ad1377ec8fca8656f2ef05aa138e26e8cd5ada5d646befb75d341e7498e605dbf3f5961d71684947db4484a0e4dbe0618bc8d63966816041c9abe74a3c284bcd4c2a2192a10c5d6c72774e34c1dab9bb7c5ffb226c443c2274acc0b048ca15d4fcfe042cc874d03e91a5b8f6c5f8eb9235cb739fcc961b5f8bb3832c16b760e993bbc2877af1b6fff45bd52f50dce1ab7f09d6883eafb3777224b8b7e32654cdb14ade89c62736aef56b5013b0a41a8d9df8f1204ed8e2f9db62f115015bfedd8f5af0b515983f525b3b9c283659eaacb1ab8cd1ea19f46c1dabf9e6a2de9d2f56777c8ea433d0a22d90ce119e2f5721eae193b4b569f552866c6805984a948cd7427f98f365dc8877df5d5f521adc82e7a0f1623c0c7f44453b17a6d64fae0ad489703a7d2608f559b82d57a1aa42504b9fc80c55e9753d81548788c4c00c8abd64f201d41b558ba4bcbea30c63f63eecf2d3b409e392c42b8c395a30d478bb2f750e012e4c814847136492104dc30c6b5159548bb3e3fc18642e4e77945a39c5d5e47c98353b282ef436d1fac346a47b3126543e617bc886026090ba173421a310d12c6ba60af1d1fbcbfe17e11ab9875f133ecc7aebb81d5f5e0fea6b4e0eb10a499d59311353c734fdf811dbe822661cc0e2fb2040be19b0b4261441ed27a0015d9bca9bf850f5de971761c375059ce707894c19d4db61d0f131460790772c767c737def217688ef61eeb339515590d3105f3ef1b34c5469a3ddb1a0abb3c99e5adb32e9c89670e8739f119876d68a205f0c8ef9e20b56b8f0e3b191989db0e9ad24aa72e9787d8377fd2f25ed6f8bc66b1139c798121adf9f1b92e1e6b83833b46e74256eb9002f78355ae3348055f7c77d1e13b15af45f96d771695ac8098d9783ad79b3eb73f0e021a052770931bbca9b071508156259b80e9f5653ea6df41b72b67c03d4ee15bd70c263b3c91a3ae2399e6212ee2de84ca12799aaacb2303c9cefd8fff16c79cc5df448b1ef101efb193d179569c940c27dc57216cf2a0793991758ce5c82063151fca42513a1e75df82b84547363e5c1bf69eaddd3316e5eaf50afd801b7467e431830166d53095659e9af9f8c97cbba3605c23c47dc3eeb0c8f7a1b47a864c56f2a83a0b47b1cdcb6303b5755e4f7601e2c22a76432a199083211dec2877ac816222d43e4f27d1742dd0690a837d9a45df7819757fd933b9451b5af89e833ec53672d3db2810a63e7df76df6c12bfe140aa5909c36e6e6cac841452e60fafd64ae7b4bbb44ac3026d29460a36faf87279f7ff3ff019a5cb615f57aab22fa5233526875a6fc550f3696a36da0408734052d8a1b474e101a8184ad3c59a8da62f94bf5f3c31a4bd788d10be659dab94c0d34f1cc2b15e2f2238fc9220dcf3ec844b65ac2ea4637e1b94bc7f0b18dc8f32122d6a42043a1a28b357edbf00c22d72f79ef03d678e4c94ef06a352cb6ae39ae77663f47e7cf9604fd64e1aee3d63f7e3813538142d60e76b2392acc28e9b8000b02737b2148d5eaeb317148361341f36594fa47841cfc9d3aeb9c8ea110260bd876ef90d383c5f40d762273d554cca12f9c09084a3e96551c6705e573a3a577b4896190c672340b37eb4de216b95d12cadcc1c565cd3c86503c246fc561ca6c8345bc991d75257e9b89331d953a5c6d38cb6e6a7461cbacf13d46945db19de8b56c0edd3a68ea9ecf89cb201288e8fb8cdb0903578fa53f9415fbbee3687ec6edf3264bfcf749d8ce7f3365330de601e2b59c790aa9c0b5953cc46b8ea08fad1a1ceb66f9eb2c3e4edb43a21e2b6c6cf5b33508ce37f880685ef883367a4bf3a6eb609c2bca34d6969d5119ef7f345df235689f8f0301bd091ba7b19e94e3ca1b0ddef7e15566b51b06e1f4fab76e1bd91eb2cbdb9034d952ca8244e5e16ee66f48e2d6886db91be608f91a2a8234561240217de398ac1145a12d3c534c52abdb4a601f1f215887ca7eb0c0f732bb5fcfff5c630d3652dca1283f5a6093155b851a3c2e2113d3042421543cb7a73b1d295fea8fb3d7f7ff4de2384b396424189fb1835c4f6e0bb8dadda8a5df514b5d34263844cf358c74b66fe2920c9a40112e2b2fda8cab9a9e58251f7246df4a1b0266cef7ac4807b6711ebf89efb3752037676be96b3e22afcdd1cb8b47b93f0b77c3562ea24c73cebb4bcd286d062a33ae55c488734d81266fd5b8e802e50db2fe40e1dba6bfba836769457ed206202068def5fb673f2e382c774c37620a7d757dc6016a1253d76868d4cc60a68b7e4c31a30af3a1f30b0eb20566c2759852c579652f58fcb68b73b583950454600db57357931e704f3d958741955f1e018aa01c44c519f616791bf0531b54f75e5c955d79e73194e58ead21913585a4c6f454fe21be7492a79302076389b3a6ad229e11a1945af0512cb6f923def6f8c4c5c2eeb92ae79a8763020ddd4db62725cd91b20d21bd6d26ca2c0e4b00b72c62895518d4f271051ab6dcd15ec68aab069dc406d39b47bc6493e68806485154c0280ae8cb5f63b501f20886b177d112f56924644d6b6240ce6ea685a57c2e8ecfccbde27f0c36e4713b7c1d84df1b4dc0e9398e6a272ca57577009e2704ec0b55d0005242b2b4fac1d34f39d5062126f45c263ba733b2146600083002017ebe3f08a55595e54cac9dd62d163c1e32be0d34d77baf272f411f835cf9d973de4c4a10f846d2aca66c4054873acd825e3cd9ef8a59f67bf2f66f861cd48d58498598a7cdf81a7b79f3b1506ef994120652a2844a33d31570e9c2c16fa4c72f201c78a3e7ec1692b3346aefdb2f3ac6098e7acd3ad1e92cb4e51d16caae5187a00a392bc0735ccb2ca631c63d46c3c6f807055be416bd278cfc353cb466fc3a79840f0918c899eb8220fd1d0bd183d0da6aea2b403b2462a76a009be18a0e07a1044788a30558be3236da1849e94ce1dd831db80c8740fad2da491f23c204a6ec998bedfb1160a8eea61dea17edf1c3617604cace0d3ff314a4aa6dfa338158dfc106546c353e58a508961c179bf19a38a298510854d3580daf40427f53616bbbecf07954ee521ad37fca1643db9d48bcfe6a9c1fd1de0411ce8618d73245e0f9526ed702a3652898e34a92a7dc08272148964c7679f7b8233ae983aedf429a207b82ad67c8c585444602b1865fb1f58578b7b20485c0784dff94181d41c94a371f263f5e7c837eb2b17972d74c2b4597318e07f37f1b4b5bfb1594e63974048981613b80e97927518359e79faba85281548788c2a19532a6d89873863553699053d43c2a315ff3b5e1e7c1674d98de2c7a946a71109b9659eabce48911b4e17972f13ac754acbd9c8fb6630d2d56f6f97df8edc91bf89a6b2b5b5d95a4d8828a4d8bcaef6b6180934dd7c3c2fc93cbbd6217b6f5a3b2b19aedd8d269f42040ac9bd89c740ac144aed0841142b1ab5522b541559c58b7ccbd140a4246d951d51442e4ea0cadafa5f9c9acd4bc76a8bb56e190439b2fe8bcc61b574a740cfb309a6836a7d71e400db24eb9109afa077f9721935a580b939fc652fe9e0c84bc5cf00e9c145993443b72db5f3aeeecee2cd709f0de97b472cea80c236aa109dd95763ef7fe9b69a5474c0a45a89b87f39cad9f37d17102c7125d3c40d0f1491c7abe3427841efb38d40b358e227776d4f90bf2d3eab95e2c6fbce7e89a1ac037aadc8ece592632c39fc845c327eb55a410aadf761f5d77e2ff2e8c090672f3cb42f60e8dae2cca5d6f98b79e9085d225e468da4168341872669af79ab76e0f322590c1edcbcb5593122ad7ac4147d78417a935c1d090252bce9b08a3f127bc2a65a39498c9aa49b0ad99ea7074dd039cead3761d9a33dd758e47f3bcb41d38914373d382f9ab2af1018b4cae5ae2e8f9f82c681f614e32c84647893976dc4f920fb4b1e062c47eb84c3efa029f9096bb4ec7c4788c7bf3ae1d6279cc1d19dab8a85cf486103fe7e91b3ff081eb87148869586cfc3bcfca0d2f004179e784120e6238007f7e33619fdc3a3e422155b2419c994e5305817ef2136e5b8ed4cda3718d5044c428d4b5dd6e222f50b980541b6637a693e87b28481f2a26deca43c6061ae8792dc0f10299f2f84d1a1e5c07f4b0d82a3758bcee37144833cd6d13565d61b54ee36234386c092b8ca70e68348dd6672e0cb579c14ae4c61f06ea5c73733a8b8b4ec91211e98c53ab4c6cc93b0ce34909f7a056ce8d584486344ab759d7845240b9b70a4f5cbd01f04d90344aa2826a39067049d460b719e9252609eeebd59744755aa2bdba28fdc577e770ae6eaac50eac277d16b8721355065cde0574087b8f2f25a1d0983262304832ad46330a4bc1a992d4509217ce6c5787a02efebb59d82f1df9ea87b31742b66f38753b692fca314254f6faa1078455d0b56befdd3cec998af6e3f85f13d4771e9f49197a98b4f8e1dfe022e0a9c9e27e5ce8511a8667dfff18b0b9ac7a4b1b24f6dc04b54d6ab5273e62ccc160326210dfa600c82ccc1d13cecf609c02d80fb31e36f86110187d484a044e13bf7311189eaa7ae5dba143bf251b197d595f1e25fc864d9e0ed8ab8872de3619ed0078ac883a6c546a4cd5ea175c014430f78218b7342381de996f56ca9abe2091195c41bef709a477ebad4b54e73fd59994d197a60a1fbc6803a6608764d318751aa15306aea6abb762fbb6a4a1ccc586d06afff9a2f1e8f193a4f95925af9b39b57bb857c9671f4f7a94d139d0df45b5e40110dc49bc4679ab84bcd4c279bcf3ebbe67de2bb72a1276b50095a7e74f4c07bb466418fae281efa94032aa7acb9f7fed932f87dd05daf3972557a392d2d22280ceb7e5f69100d67af2e8a6a7fc7f70e7a9fc5b12a7cd52d99ffe3db9aae557bd6bfbe828fcc6b46cbafcc19f2bca1e7671c8e5d4378939d585bbf0940edd0343ec6ce11c89eba05301103c9cf1560b31abecdb7c99771190fb4d1407dbe852977b4ea1921db40cbb643fabaa0bbc6302f1ea7da98ef39bf1dbdd61aa69f1f7040aa6be74dc46d11914d3b6c25eb3e967859dbcc0eeb1b8a0aef124d3206fc19d68ae3ec683a6296416ece8c3ef78f74478277da839e2db6d9d603a034b9921ab9334523c70c84b46059af09358e5ff3c20c212c4abd2507088db05c78e62b48ec534b49bd7e95ac29048bc45755cc3b9207392b13559253eaa68c068eab043f8d101e4d270c7240c50a8269decd5bda44c7957b3dfab210059f3db0a6eff5abe7bcc3bad47ede528d465af79ee44b5e6e06be67c8034004e9aa4cc0916fe600825faabfc740c6c1eee28f7718a2ede58a877d3bc5f32c63856901f2790efa69fec0fea99de6c64ba09bd38494a88859c77cb80699c60524b9517009e54be44fa10d861c275ec88dc376d9e3c10dc3713a5e9c09e57888548370ddd91bdaa66104e08a58ff3256286c914091fb016fbcad89c2bb6fe7c9fede03dcb7289a10e7196fff056aff30f875bde60fffcb5d7b4330aa2c4aadb536101d68b6a7bbfb22c82f13f23c92bc286c27d5063cdef5b488a1bac0f5a12e6858e4d2e2cd3b1f0bf8410576ce86ea12253cbbb86a771104e876f44f0f1a0389716d55df907e93d5775898fb623821ec9bab2e1b30f9dfb4f9990c8be466d80cd40e16157767bcc438096c35da8c93e90f31749be2d6e5d85c82c75f1f389cbcd3a217296912faa32d45dff00fc507d7f17f48636e9e92845693c70449ca2956d6c62b8f7766ec08b048af642b94fbc14d550c984ca7279e9f0cc40717eab97256b8bd8e4d36f029d27082b6d9f0a8a426c79e996af0a57de30265e5314b058a12146441e89ea285090d78c57e8302d0a88801b7dfff182275cf70aefa7e8f1f79fa9b66dc765d507ac3af690107944d572075387bfbe7b850d1d5a8d7d780d8194a762385fcc971509ad76986adf0b432425784be1f44443a13f293d46603ef6793b1082b54fc985245516af2e6533d24d10a4c3b21a2ef97abf1db0ab91756932f503bc24951b2279fd971edea3be00da2ffb5bc5cc83a6b65435dfca81c431aff6dc4fb5759ac6b6ec597b8a082b50402c32af27518531681f5be8e33a51d11f5596d521c1f01c0c4ca6039f2b009ce958a5193528ba41815108c5b6980c6507c62e91c4938f6ba1cd84db252322c6ef39df2ec7470b2df71544b677b008b75f3b9cea300c864fe146b40786e2be2941b7b60d47595baf9c605c0094bdde69a5f9d4eeadad726abec23c60d3b4721a96180d4a9a7fbe414c4c7898f7c0ecee7182ede7052c80438d816eb4561da06776f074fb430560fcec0a97c6c296306935007f324ab04b438ea493de43f474ae2617980ba3b4dfece01dbf8c98fdf0ab00bf917bb6326847b9aaad1d2a8985c3aaee659addae62b10132adadb2f63acc59c276a3b3375a60f272f1f0851fbca4b0e091a8968b050267c2771a62eadfe9bf690436bf0a3526690c801a73fdd53435e154bf9897977219427a658fba18a5cfc2b8c5fce31fb10cb2782e5952ed4319c7b9ff0a34a4d30664800b668afcb1f02bf42688b842893ad499a1261377e1e51e56970b68b867acfb4fe044d35d8c0ee553a1e641c57e24d6d4b420e7ceb2129332a774baba0a8a8d47aa04eefe03cb70030b50e59ada6bc0c3f2e0a92b2d19528a0d57bbb784c0e445413c35d31613c9d41128f397bae335af6e9c6fe2e9534fbc66b36508e7a664e69197fb60215f6a6278a1fdc6bf395af70fcacc60f7133677b57d130fa8666e1fdb884af8f9c815ab0dbbeeac8cc36ed49f6854b8b4c14ad0d0e8acdafa9be6dc09d4394754482d93e73ce701011f6e644885af8d35d2fdac7ac508be405fc0b8ae57984dbd69617b237c2885e1be45629d5c02bc6ab8cb587e9c85e660ae2d2afee0e3bc4c5f28e4857896644dabc84d9df7e8285fc23450f99be6f553dc65a2498163834e60b7fb0c5909f8e39f5bba1b4d8fb541b021eb1f8fe2da51e4171fff8f84e38824ec390d1c9534f2f45fe7c7b3b9e12e517b25c9202c0b10aed4e0a0db265e5c4c2d9157461c0d69ec153172e94bfacefb5904f0f8fffa679918a7286bcbc9122ed47e528be774b3e13e54b985879e4224283906464bf330075fc56293edebc91b94c6c7e5321db4a3a788f97d4172c685cb4b37438fce1170e95d0f1c3bab89a6662a80c214715ad2151699c3b39e47eed63a73b8a245c7dc12f8dad4ab4744d9e87d0d1f9a8645e62ba475342cc9766f134d840c12907a45ef8e490f0a20db62928fb64c600e174b4f305d41330fcc4cb7b370091d1ba454cc505f85c80b33991b931c6ad88eded8f905bc9efbdfa1545f6408825024860ebd5bb218cf35bfc36fac19203133192aad1cb99ba2ff3f2cfe1e921504c73fceb04aeec9252d96a026fdcb5d29a5b7eaff8863bbf8ebbb10d39ebdf82d8b41aa029bf75333854c1d1db46e38fb3db93921e7b56f18bb007d468b3377b468f6a9566b44228044f41ee90831f3214b9c72d49a87afd901064bf8ad00d2f9284ef0cfaeb3bd7a0d7904b8d02e553f70d5df4bfe3043146679a978be3a09df3e36fe2451aa855028f45a53cd49b644f2d2f74f2a709524c36754385e946118b57e7512e80542b165517a032c323a2d6c58cc592d1b1f38820295e35cbd28c3b5c3c1168c5d2c803210563c23513cc97170db9a3254ba445e32a245e25b1db41541f7314daaf46ae0f55c3746aa782bb87274bfb5e2687a919b013ee10c246507db49169da3783217fead0dacd35c1edd1db72c852bf74067da675c73ec2fb7875404a222170bf20a80ae63dc6bea93e18424abc6c46c4577d7e5d5dabededa641129d076ea48dcb9604346945fd856b69189310adbfaf0ee39aba96865d19629fe703d752b6a9f04fac6887cd10273ee6987f02ee84829ebf52f0f7df490abb82fa56e9417cd180d0e83bad13eede4bbbe42fd03d96985e9092f90657778783c56d21c3012bf50327bcc9db3420c4653250983e91bb617e1c0a961bd7f67dd7b88cffcdd0875dcd51a1727ac7c5787046f25e82b3f8494ad3bac0ee32c33cf3d9655367da2ebe151f83d820064f8453229c9718ead49b1709409c14a56cfb55917a2e787e2787ff383ae256c86b9c318861ecd1636491c8bdad4b78ae8f1cc3faa47ff092b66d59c24b448893df52fc409bf9fb82eef536eee0908d5b5652dcd7f29ca3e0082fb4523932abbbfc0445a7dbdcf83d9ef37c0df5d0b339387a958f5db64e74217f6808257325377db0320dcbd786649f0da2d2643a691215784cf058c0fc9cd5b0032912ce29f7c0abf00dc676adad8b611f52b8bcc2a68d48f5bfadaf667cfc335841c5eedf3fb4c9b2f47113c78779087b8010f736f2c619d3b2761f198ff6da867e09be17acd872473612e476bef16b78728910dea14c52bca704ca2dd18a737931eda29b5c3f92e86a355a33231f5aaaad60a9d9adcf4058e57509d1ad494d3f2cd6a87d331f6b39ccd50fd17f6dba95f5f156a5e409f9f666f998a89b1cf74ff31fb22baac21cb74b6fdd374063b0dfbfa8b9b99e154ff211f487b14db137482177c89f24f217b22733e68052b6859aad471ccaadf0058f19b947b8edaec3f4d0dfd4188445a7b06ee396545d9f1973ba5ed0ec78b345c9b682b0df16169d9e8f7410e633724574714c8de47c55e379f1cf8a7b4904908c5a31b1b158cedbac2d17a6c16d126730c35bdf2ed122d2c164c8fba8bf469f003d2d335d4831507db80c2af1aa8290d26f5507fb4b6e12a76bfdff848daca132d0dd6a55d2442e88d4b3057d885f6d113168c5912aa8d1c522ea7323a6719129b90f85b39ae45555666623e7f266a1f8deb4347e231e7a972b961b95352fade14ff4120b6f580f5c7362e1881558aa73eaa918ba1c0d585c815d8f6ff5088d1fe4a75582040e2a3571d664b9f9df18bae4d4ae4408f2827c399031da80cf88404d9f499ea11404b3963259d60ae441b003018391f9bfa2d4f9ec67e08608f2abd934de3371ce80638f429bc4b4d0678f52bbd4d541d2454f667941c23d3cff4c92738976ed5e0d22913e6a8fecb65281e19978a752b0e1ae5c927de41f1b73069f79820a3f83d2dc7ffa13520213bdc66cc7884bb9118906f0efe87e22a878d1ce2c1fdecd68e25a5cd424ddc79cff69b552d1102d14ae25130be53ce243c6b663790ee83ceee91c2157cd2911c032879356f7cb40d8ff67851f6dace62b11a000b8c3279c7c7eeb7fe3766bc2ad180b571ec3da364ff87d4bab9ae34a1cb3172d788824eeeaf711e6f9199ce3518193984a81fc069d678e47a6dcb04697f4563069022fbac2d021bbfd93803a2547777adff9cb591778e9f933e8ca000143bff6cd35bd1b4790a3489585755f645bcad91faf38768b005439156823a3fad038b85f882fbe19d27fae9e17060b1e1b724168db2cc774f340c2507ad40fc1722f74833a98f4c0664c4c8a7c65f057af54e68a36e20cfe085cb5771439e9ca09240e5890090fa6566fac17ff55bbe4f732928d8493b04c9175e6be84201510d1362a2d1d3a4ce82c6e8b4aae86615ffd30aef6d869eb724de0926a3ea9c5b2445f879496f1d13f0026922bca9e19b2fda74b7ed4f553f8a1f52096617ad7daf57cca7cf20b76231b3e6540d0597d7564b6344d615cf5aa43af3bd993faec621d864a20c90bfa3a3d88cb44c14560f1e577da0ec78f99f1ff245371f2da4c6f2b419c3efb683788f9fc7851aa1948d495c45b9c2c0ce86943789856975181d2b08ccbc4526c9fc125def3aa4493024dcfa2606968c40df3857e6b4c97b9842fc2104a1cb6552710d292df443a23ce5ff78fa1c51b81cea4624e973a54a977904e90351eaee33058ca5bd09c74ddfc6f1b3e68aa48787e60d410aaf2ef4833918bd1a5d1fb3fb755c4b7221f94332789ceaa9577fb6c5c189fd07d111e16a9e99563f3bb1fb7e9cb0120b474027899da220cc2cd84e710101238290407ce25f737901a219edd916f6cd7a1d4d068340aeac8647f958c8ae54d43b4c00e10877af8cdcad39381e630d431c21982bdd18c09e5a3b5f2d94091f160667fc61e056c113ac77730d8d46ee0169911d60b0be95ce94502e92dcf19848ce50808a6c264be2ea08e200eb0aa064709d501af47696ff283c4d15a65393673bf7e000ebe45e7ffd76f8b375f70cef4c4bf04ed36d1a7ac3980d79bbf762ad2e96790895fa2264b10afba3e72c1aefa6597e3295494bd86f9df4ab772e7fb74919617ca0a769b798081b3a1bc529311f4fbff55a275f23e30607f43c3e647dfe1bd3a0069a107431d48412c6e533d4b369f6e231679904beea780e99edad1350cfad986c27a6cbbf6565653a8b00c2e793ae6a1e2c14712f410b60099f816996af9f7e89ad7273202c9f34b879cf052c9dfd79f1ced00fd6b7ef67c20e6e1a67eec5b7f5ee31dc6b8cd20c05059335d6e9d095d4adf8fc10c7ccf4040e1492d3ce89896b05f6c178d19f9f387df2b214b9dd67a722e4c6403d657b3c9f45cea7e49ed382147341b78a7961936d9a152ac7578aa03ced2f34edb87db4ad266e28bde1ecd3e66f331674ca840d66ebf13bcc6887a03f662f4fb908232b8964ae5029febed548cb2b683b624bf2d5cc85db43f23afc9c6a2d1dffdcb2efcad4fe1eb2c500ea7d0c18d2ceac74ebc1a384ce2ed309557e62993b192475f17e7d75e153fd9b5e352b802c019c896c657f6e343a5148746db1257e18f133a7ff561f06c6cb1e2cdc2a52eb0e12af51b65908a18161672c4d536f8d79090e5db08e37cdc536df48077d46a13aca38720a5a408ce1ff06db48103b69e84538135651cfa89cf29d258a7446926b9acc7a4521feffb4800727af21ea266a9e8446b3f3919b75066103d80079581651f93e3000b53d38321651837a31f22649d75210cf5f378e845c099b694a9397143aa822a66c39e23ad9198b43a866a7e992b12964169c3724ed4e14659c60bfbc99fec941dec4ca4ad4ffab0788a15bd6913c7ebef8f0d900035c6c0d3edc3e12e655c79b8a9ba5b7b2bd66720e7f60f9bf26e42620cf6adebe14aadda8652c869d824bf2ea848719340521b6060e400524efc5bc4e7672ddc6c3396d012324f9de1103985fb2d1f33577047b1e27d6f70644e460f4f37c59fa42a368e96812bb32b495427afc7aa205548454c9222b9d71bc373a59c6a428355821b082a8dc89d7781ee1779f26817413c079b5760327dea32a47d2ed6733484d1dac46ba9faeeb231eeb553e0be1b5c224182f30a0c7bd6625bfd34659914e537838ba49c080d706acfe5c316a91689849f402fec0ca586382ac7d818b75eb3ac26a90d4d4ecc3645b0b820472ddab832c09a2190bf7c6df3d7a4b5b3b3049658d84095a95125dcf07b27fbe665f06d8b65804b4151eb7ab7402f43b47d5a6e69b10c68de70f0ba43c4c9f96ee9419976289ba0b8efd4adae2c7268009817b98d780859c5dc942cad5d4d22155da7c28f50b43e4b01f9aa1b306ef4cc73bd3cd4abf9f40e6c0c5dd2121dfbef1775b129a765a63d4ba6b9a7d2ed2ce80fe772830abbf36b176834c3cc0a8c8a786d41dd50f45d535e5615c05ae72bd46a8a648d4599dd69ad9eb785a3c4837da1f286dfbe4566587bdfa431a28e665fa8cda4dc0812fbedaa49dab87e8e1bf1f1507c4092fa6a95cf92f4121d4eb4f280eee4497ba627f9a53186d1c1e7dee70537e63bd4144abe09aee9d8cf95fe6ac45ddab8c95889fe6511b9a5ebbc0eb495b8829261ec460bd6284d09099aa96ea0e62ac53c96058461c7deb6dbe6015e517f3b1cf11b0c9ccc509e806c2969fb12653e36cb81b3f93a94b72beb7b20bc00a1d4f0e8e2c32538c8c31233d1d02d4159b21393a80b3d3176a1084b6a83f452ec7c3557b38cfc405e8ba1d3c0f1be1266c1935ce9660a593e57ac5f8528a2666d8e5ee8ca060918882de62b1ccfd41de85a6d380b707bd0943cc0fab146ce6a4e5e5dabddf4cddca39a70366c776ae65b02c3f8221bdd24a77b875330468c92cd3a67819551a5967cf10e2b96f8c1ac38ed3b8e9092dc419d5f09107f28f80d40bced57e1b7941ba4b6dee79be7531b74a56046e16586104b5b7702647bba0ca322232a41dc33830c31f4b8188f25cd0f48e37b58aa93e02d96b6bc98af35d7ef46887926cddcd642112b4721cb78ddc36c2542ba558e1b3f3a7a6fcef3e0d0d760e5f1d09df54620997483fd71e3e3daf32a5196f66afbb95c9d139cf88486e14c0a1045664c9a30eee09a5b0d010636a545a8e8d5225eca76a11e139ec5eddc7413741501da702a89fa91ee9a0437e72a599256831105524dc9fdaed501f9825c346ec1533af97f833dcc70e3eb97202f856c54f098bf02d44715e5590a3292089f7199ecc9a44598bf63444b74363a917cc0f02d934140f17e927ffabdf4ce945d58a22b5b1dc72bd0b546671984251a0ee368d5c324f88245e92897c3b1c0c6de4e22a2c4292d838e8b9e329d12c339d478a818bc72a9f1f90acabcd238c1dadf870b6a470bb95214fc6dd5961d19677f0f6a31d2e2cce65953fe069954498102520e8f622cdb9bc5eecbac903fef1cf85fe05e4c173d8810aadac97bfbcdebf48383c9044f3dd93b95800f39974706f46da6f0fbd2f52cacc6a668ae47b0a07e9dab1133457e30519cfa4164596a6c83937c3ee9c4954401f628e19068985074fca6d89c8e9ec3d10d8ea4be2ac66e2ba5d3c3078c163e61234c3a82fe3834e5e4f691532b2b056f88cd12794e40883b1fccdb075d9de1604d0417da2c3949035131d00c14a5df67951a9c221bfb3b58ef361170163d3012a5c928318ae06f3838f10bce966cc88c28c21ee951f8b41fa8ac6222d3ff95487f4bbc4c4b0116f2965e5d64f78b97c33f23905b1266cd1e6e1859b7ddabeb439c3a1d5d06fbe69f1df52d8744098bf45b5eeb8f844b6f1b5cc4442551d7fc8ba94461d752af50c7e119842c198d634635f16e72c7c0eb76a2296e89a37e93613ab1eb691d77bd4d73cc3fd0cd27ff906c9a86fbecb4ad72c23584284b819fefe1fb4c17209df66f5b9f9d6779c3ee312864a9738e4137691433e742e8e10fcdcc5345a4c56b96f25a71553cbe7dd9ffbdfc6b6210daccb2876488f2a1f22ace3091a3e529cabfadf6316c67a8d73ce7d836b5f6f39db51f70f95639bda81527e37c06b467ade33dd2dce18a392f305e71aa92deb1d5abd47d060144c4d3feaf30a2cfc6255a274742c2a31a93bde4b32709f0e6135003599654c620f51878702e5a7320f0cc269f09bbf216436cad0c9d95201b12b1dda5a1de3495b8a7ca35d12e310777b12eb5de38156260d311b0b93280666f38acd97adf5e152ea0943fb74cff762e1ed3dbc69d8a99e3b5c452eee1f35c1bd9e157dbae0638081feb03f0577d9f8cfed9fa43c97d1145fa0cef8406cc5dcc0e6b1bd3963a4cdf4222cbc371cb530a183f637dc2b223e04aa05569573fe072854afdf6f751fd3d10bca99bbd17ab81e46c2bd580af597c28a053794a4f07a9274c1621524c6f016123143bfe545878ff35db7984eb683147dde84c2747ea0fe9dfcdf2a03a4adb24d5f08759fbf094a625c03d0975c43409853f7d26dcc3874216524492e2562189bf25ba551979242293b918db78ec64c974f83543284e10894af676543eee04fd3211bcd61e23a36bff5cef8d012c06e9a9e007f1a9c8054c4030f3915049dcab9ac7712f73e990d7c863493443cea881e67a562b033c9e5b9386df25aca04fe264ae5a1dfa72af3bd25e656292c2c9fa3e61bba87301185da378ba8da5eb50c4edf74195fea5864456bde4d26250377337874a9810b0dece8758359ed7313dc5a121d6b6cbff1f2453f73a27fb8b6543fb474819d396e487df2884e3b14e83fa7be1b43fc8ac262a478a6f840e34b9ca5164e191c1d09b33cf808a68a027bb50c5754c4c7b153ae7ed1103d2a5eb03fcd99b305f08c012a30ddaebd2b2645d4823fc82c898e848ff440b356617546235ab81cdcee7b0231db7122b2540cdc17ea2db9984c958e95c3ec29839c2f80e5e5fc0c821f9d56c7499cc15d510d792c704479d1f496811409a60b5f0ff7204f84a35d9746a3359949665b3d7f7e3f5c242a3d7ccad3386e2421225a54ce92e2aa8173f58437bdad13544710a4d576bf660ecf777a85774a10c9a31bf105b26ac007e8504ea872a58fc1da8f056a907dca70f344824b19bef0d185859389fd3cb4b3f6e0a57d8821f2571b4d2b17d432c798b92fee3fcfa78811355b421b560d9942c4cdda65e871544fb8ae98ac9c420889e08ee128b1d3aef12a0646360b02b686bccbeaebf2801c3de8853251a966487ace9aeb6b6a0a4f98dfe740f1a8476a4dcdfd4a51a98634431fdd9ece1338aae95c2f599a5290034a736c1ba836bd44bb14b8e0c65d3104a7117ede2b446bb2d885daceae82850c8edfcb5522bae79d1a20d9ee1088d2d26064b306cbda03420172def2dbf00055861a4af075b9c0f864dc293d73498ee4e959fd9841c41fde6c0d40dcf61f16f67dfe007407c1a56c4230ffdb6a313cca507a9195a77a0504e11448772e42b2d33caf9d440fd4c9975d5d3a9adaa1beb5f17619fc3026b6107c72bbc27d9bd1af1620eed0e7d9a3339c25491bc6c58c5bfb598e5ad7784bc1e40833998345e4a68578c445ef85ad7fbefe7680e224971ea211b6439f45c31e93bfa46efc28b0520477a13bde98a59d90138b16f212b2d444ade3ffaed234542380e6e957ffbd4bb1c9bd5b09237f4cd37651030746da28e2ac88643e9352a6fae275ba12f8a4a94b92f454418888913db84f63f108b8ddc07af57526733ae026bfc86ad7399fb202e7bf164305744e8997b4e98d2fb63c1487192517c4caa86fc619d9e84a66422abe3e6f445093c52399a6959ad1d9902bfb9b69458e41f0d20fe7f0abe4a5ba958c3208bf4d60318b3c134af2dab2954e1a704aec536133a22825f70ce2011be082e623c5d1daf086baba128139ae7de1e9dbe4d5b30190b09afba601046d39e7cdf9c00230eb78de1bf75e6ca5b9a3b7a8d1b11d8ee7bab3d7ea69a97757618099e8b6d8662cb3f91fe4164e8262ebbda5846c1736f96bc6aa36a9e18c253026b7d11b16c16cba1a5f72f6d026937e3dac1f5f3b55061cc9e864ce0f5e20749ac9a1ad0bd83a55b22c4ed33f2b0db47af21a5a7d53df4b0681a52cb6a8ecb0424794344392507cc81416e1df4434e1bf6a21d782f61ead1cd1ccecd7d221bcc7c2cff738236e40b799bfd9b39a9ac7368a6def370aff8f92f86440538a47101927e322ef088e31f8a2e02355d679a406ee970571822c4a2799a485407b0bad4c6735e646b7e3fbf3792554ac87e86940664f062752d920959e16c01ee053443f5e05d29cc186d5d1b0b84fdcbf7870aa7f5a5045bd0311104f5ea958105bf8582757ca56a65366fd1942e2718a58dc1f513915b38fa153cd43843f9bd0e9c8f2ec0a1f631060cbd32a39f4694f1fcc5ccca41c06beff8444ca2c08747e2be96776446ced89770431b30876b91fdc59ebf71dfc50e48eef32e72818bd1ceb7a6e36de989a624eca6de8ba0e20f0d3a9e1ce3bce220b71ebbfcb08668dcec0e7f2b4fd7f641dbb06d7d66d50a7dca52ec308759a54440291c8b1a10376ea373c24589cdd224dd935ffea304cf85eb8cad21ba7a3e25c58edbd05c4e0f554e5ed5eca142d71da1e6ab86f4e8c476b2d26c72e6ed1e0ea988caff787b9e0a1fc0a02ede53c013aeb90dd6975f52e45ac705a134156aa69b4765689a32b8f287114da66db687c0d65a4a5e5d668c231ff48a79b3074b3d405635e73aceea97cf0a2d5b130b41f9d4f12e14b527718d445dc724a5203f21d4f965282520803701009aa9f8ad871774a33f2a4ee627905b21e4b8e253b960e722f56527d29a801994be7d3a6aa078cb20b615e1fd65df2559f768079c263b54f86974177089644d11bcfcb918cd808da64e7a4dd2b597bff34d3580de1d7f8d6792056e204ca30531b5fefaf82b15419546f9a48237dc7508635f8f7c8bc8dc238299325629902b1cadf23bf9ae9ba7543993babdb9e1dadf2137c3e10229e895184d06712393d4d344b3a188446876060536bd49d754e34ac41f78a240b869c392c8fa13d84f202576ff13d062133717c3b8b6f71410966e69781a0ede8ea761e4cf21b21d47bf1f4a8899a5607b229d79f3dd4d296714f004a03946242970e4c4c475bdb52da2f5cef3938a5c393823a0d6a9fe7d5507c4fe07f68294b98cfb63bc056da6382a2c6f2689c16a071c5af6ac36c442e91d49c41c2fd44beeec7aa7faec310f980026fa0dd67a55734c8d7e87674d673aee0231add4e9caa5ea8516a19aec9ddb90d00c8b4512e7ae55ef22c05d17486ceae3e021db6c44465790315d765afbe1fe685992f7e7edd3ea23cb600dfe9b86b1d3f5e63a82eaa0dcbad73e35f8400ab6540eb2391a8e3d49e64dc3b4bd1be18c78655a89a183f24c8609acb7e25dc8ad3668daebf9c48c920992c3e1cda2d98d4873d46a0038710a1a16b7cf9050ed6ba4ece79b35eb9bfe077cdcf2f2e24a9bb64410d55b2397af30f6cda318bae3566ff3d4e45cef965d2e4137353e3289942fa5273f9654dc62ebaa298ec2e6aecdbbc20833caa3696e87d5676105f54d07c4ac12068d00d886a1adb6e73f8bd1413e990c68bf24db5a99cbdd6d84756cc2bb3efad7e72513fa32a7d904fec503ce17557a30b09fbaa558dd8e7ac44bf05e6d0c6daa30f4406b4819422e29bcc572131aff1aeade714a4f5c229f75c9ce251f890921479644247184619124b077bffdfb06af0829bd7740b6693a2d211a45f39833cf7007365db0bad2b62025ebd0d3d43ab967a3c1f9ecaf4da6b2d1b31f3f044337aaaf150c9c8a281fc71b92b59d4b860cdf06b3aad13d563480b479f7458319b9341689edd19215cd29627e1968a6112a0ac7ade7b333ce60488325ddad8d347e1af1c1251a32e12beb361f705dfec378cf8d454cdbbc05767ace7a39e7f6fe8f1ddcc513255c32412ccebc790a4ea73b6c289aa9a6beae19c463d7b199ad956277f59603b216a96924778f26a38cf3c965410e2d7c33f07bc4fd37b5c84b1ee118eaa724f3b1ced8a79a83d787da77703b9795c714d6047111b1bcba36a90e0c0fe8c6cdfeb151d5377567b333a27b75458d9b2f9e6d33f3600abdbd5cfca730883982493ec7f91dfe67f2dda83df9d4c46695f2f50da3058fdf72e052148808d56e2f1e7446b808f08d861ef5c470a328b0cc473b939966b8f2f514fde481eeeeebecf1f67b84c2b06a197aabb0ce7d9049b1fae1043c9354015d46fc0f7ccb64c1d8955d5280727d1110436b633aca1854ac614c30fbfd7798daaed86b2242415b554917b51ba98e6b67c1e6a09908095e6ccaa960007aca46b7c6a37f9231b6d8a4a715325e87afeef6ec8e8ce8330944ce2df64dd1cf7fdfb994e78ecab7e6e5f05b6a5bf899dea59a273b91ff358e359e3735db535223a779780a6021ae37ce68f05c4f48af1279acadb6d9d3563c76f07a1b62a13f148124209708633bc38175604463dff32fa6fc89d5cf31957756c7a6fb00eb08113a1a941c0909d634902527d396b88749b4bcb120185183efc23b2bd7092588ba55062de1cd7d295501a3605cc96613f9bef2730aa61c21c2ae869c8288af86ac3b7a4d0def2e259e82dd8a8268266d73b4afa7fb89b4162025a8e2926ccfa4143d6da5abb77f8129573162ec92e836ec027d67e1be6bcd2aaf81060ddb934388dbb9de2a5e298f822525d41024e3b43fbb86b2498b206508d6dbf163631fdb145122f375c71ee678c4c2a37337392fe46fec8bd483b879ccdb6f8537d36654001a52d9b34758798919ca2e285e8354d55734e1b050c89c09c3945da195c6272310cb1d51e5cee97c9a94a61df8569b69702fd19d31aa13e95122a760c55b3f64fe11ed7b98005704bbcda940edf4eab09063d04bfbfb4fac60738f0ddf721f8ed29009d090b8c57dbb530d9997efaf2731983260bbe055ce57e374c88ca7803705e1580459218fde5001ac8d64fe0dad67f4dcf59d422d357ddd45d8d24eb7a092d6300112e6b75c1b25331a108a6d27d5ce23bb0b2fe0737f2407e16bd20aeb13b82e7569244816584cae6cb853094f22a997f8f95660306ed385bbb2424bcce9540af37ad76bf9740a058d398177327f6cc075b5728d72b6266dba485e73c4b1824bc86ea3208fa3541ce60d4b815b8ed64bf4f0fa9609f8a4f610f7abbcb64be56852de9ae9174c5661bc0e1fef70953b5179115a7ab768dc2e802759e348b8f91b4829793990d25f14e0ccbf07ef23c490a4adae6dcbab24922cfc8b080a8fee6f9ac96ece2b0a302a981ce25566482f82959afdf0469cc798a24aa558ca657e2b677e05e0606fa3620a8d0782bcc72b1c973bcdaa3e594bb8a6f4c0b77662fc9dc98f4b410a937ab081d1ef5d46601ba897c77d401c2ac8b0d9546bb57a36be56323c60c7ff1da6e1b73e3276f9dc02f8b6641b999f58f311093f547348d0f367f95039be5d6b13f534b65dd2a170f173030579d572f8639359db9607144f03ce0d93afa1e6726eab117887dd40a3ae6efda3e20d4123bc7785cb6f5df6fccc5bbe21c64d1ebbe80b1bcca78e87ac19d22694567fe01dd94585c47505899d3de9226f3c44cbb899ccb71b086626d4671eda9e1dc4942e619061c455eabc424405235ed2725b56e3a52d3fb5662ec351262b7301c8dd800243422d112d003f89a01b2180adb5addf2935e723d792ee224d6c1a55c84d716124a0d81feefdf79db6bec8dd41f262d3c07229228a5aa65aa830455c91d52eb8f334ae4afddddf9220a6367d31f4b65006939c80f81a87cddb75c388a485654f90d203587971fdd1cbeaeb0e6d8a7798fac434398d32650b3152568ac70c34e430e77a2107ca8d637f2da72a2f0afe156fea5105be895bcf2b0c3e725b860929d83c849227ec385c92360da9c32242dc7934e8587be3bcd96913892b9277602722eb147cc4c585824bb341927a7bb991667807cb5e1d8fbe2036a99084ec5c6077168233c245a3c193ab22f8fe0184e18b5cc8c44566267b0b69ef6821c545b0893cfc327b65124c41df8ad656235f5b1985b5db7ec7ccf47f3dbacfaa9886d048f190d8bc5f513f0c47dbfe85d6ceab11100eb54b3c2a310b2d69366770888c1666e8ab0622ba3d618cb5554f9f6c3ce1ff05ed1e979cfd23d8cf0b32e710f9279053669358dc442f0016773a4ff9268ca031ccf2b708039c988d75dbce9d37fc02fdb4410e1e813c86fb1cfbf394c7b2bf74d3dc0b2ba6ab44a7e5655e49ca999aa2d0be8fb097b934371b1d223942b004f87a2f95410e615ac45d7ca64b641982eeeaf6c5c81625281cae69d777a8a59cdb982a37ae0ac53f8719609a377b880c115d9ab947997b5fd6ac17f80173ae10ef1ca749f7b24195a6d37d499ea51252ab5c55910c0c0bcdf6051e69867981fed959006570d29bcb16b9dcbea78c47eaa9a4347808852e4ab9187379c099100fa68bb5cbbf999f5f4250fe770d3e2bf5e6a86726b0b13fdd0ab84ed23acea66caf7d98dedf66ddd08cdea2103391b5b86da502c470a0a481199fc1d448867c8d48c6f5c5660b4d8fefed288f5bd8a07c81206da7b3138b26221921e3e594a65d99daa1e796bb4c0d757a4327d1d71f5fa3bf50740ad25248bc5d3441d7d067fbd5600a7586786d332882837ba17d4ef70099f514fbf6278af75f508a42c50a82f46cdc1e35caf5f47bc0bbe191de8f0096ef5857c106a3fc90a86f71fce07912d358747faa16e6d31d2ae34f0f6a8291374f59a9d977411af46a1fb8c12f7c2be9f42608d5f8c3f7fb8e53280c293bdeba21070c8b74c52d16b5b8988dd85044363ae465bf878fb86cae58633f2dfc8724856fd1c4d710467ad3ba8e5c5c0a744dc2dfcfd513e3d74a330df8b203faf36a683d9910fb240e207a5b531dc9eac5c67317bb1cd1fdb044a69374160dcfc6ce55f74a91195fecf3caa437c7f49ba6256670657c74522a35949786d74d648a0d58f9c435c5574e42eb1299d731ee6c77e43ad86860abddb77064669af7795af3183685f2b4ea32b1df4fb83ec5d2c4baf6d6d875bf216d0e55b7590e3dbce0df5a1d7fe539b9b10be371fbba7ceeb59de54a04c7463956621a6bd89acab66fc8d4d04bcf3de083faacb204154770149d3903d168dbff662f48139e85e1c4ad04287d1835067cddba18b693b3c35218df1bf01ec1004d5b8456578f412c6a064714a9f6916de61bf9c4e5e963065093a4ca6cc96a13d499903442e9355fcb313d054712d8770dd2a27dc803f4cee8a0f26e60376170a8ca146a9636cd7b9c098990fe22e87863b3ab8dcba234daa7aa7134e6e54427e903ec78cd987a79c65d4357a5a62d5f27578f06b3cf5eb8003d515126a0f4c4bd01ed4f77b200793b0bfb537c720a00bff150979a42b3640880b8629651ee2437e55fa9d33b84a770d58139ffe5fd7b4a1a0b09ee7402f0baeacb537808e0a48661c7cd3b492efad5d7756d9dac9cc8093f4d0d9934ce4ef8df570f7c96ed12d2437837f7a24401d6f7b3c2c159c68472d7347d95431cc51cc651a5a1705d722be721bb4e74ae54d6ec905b1d9f1ba2a7be112f2c85239540639dbb124f436c8a00bb41d17dd5f565ca339047757e2e6ae0ccc251b9ce1669ab55033f73d16bfc5ca4465017706d3f5516745493d9fc5758cbbb1484569572a7d7349b5fa54ee2d7b9403b371c1686f784fe36dbbcf6389a4d3f65beb3d9be0a83bd23ac19f819a05e038e91032c587b2e4d36e4dd07f99917e8ac681f0c6d92c70bc8af412a68801f22c1e4dafa43ff630b4763f683a28a147383bf3bf9c441b7dc4d65566bab97c6b7f4c8a83775af14bc695b8f6c6bebabfa0896b267f6008e972f9a91f30beaead5550ad56b3ff08ce4852d56dcfcf4fa522b51790b0ec81f36d26d7fb4e3293fc293432b47103d638de8b95f5c86c15115d200dcd079de6e2abda89d062c83f3f31f43105d1499c1b938352a890a2cff660db12459959816bd95b6b5768939cf9098c350b1e67f671e0bba4e382b579b6234494f16b87d54859a239e245e57813a5adc0302b5569f825e4bb9fd7256efe72ac178632072f2715e41f8fb67a40ec068cd74a7ec1a73f6ee91be8033a16119bb663b9af7af3bc09ef9570065715872baf79c0519788e6391ba9464c5c68d5301194f1abe87e09cd5fe0f9a1c2ed5daf3328db0636da95c042933bf6d6625368c2913d39efdf45a1bb1ebeea3238f60e4b42b8fa6087d4108e9f08db5c8caf404d996fa850ed86719e3b9261ccf3a25eb9e324eb7a563f87c169e9f7616aa3413e7f3afeb45a973964a406da6a4b98070aa1402d7a92942826f01344aa90f3ca3b0e60c4d7fe56a1bc95d9f49c17d86de0f1d130c71673c2ef17a17fd052840c49318b12c6d5bc6b719fc958feaecf32113d19817ec064c7f5e3004f563f72f19dc3d4b53fcff22d126bf64b788d212b5866286ca202f5feb806ff6b98b6ac59ff8a96c8ce59ca2508edfb1a832da74b3cc5fd1f8297162347085bb97068f8f97cbefda6b6347066ecf1cc9a329f2ad0a535176a240063d7fd0d52965165ff6c865d14d7a7b93f473a7a890e2947a1458d04bef693c3ec7457650f5b1302a65bd568b36e1b2cc0f43a5f620e4eeeb5649f6db1b13e28dc7a04c3ae344446599afd77155bc9c155b7a771ea5b8976f68a60a842813231c1dc21b72de49d1334a1969fa81d1155c7786b8a5d04036c8697caf0f4fb5a5c0c9c960bb8ea07ec4551d9622ee3ed12291b64a8cfaf9963e27256cc59d0117728012d38b8ced7c09304434afc2d5d931598286ebf85063c3632ec05924e29a69b1e6204c48afa783a99318be799cd341daa246c335dae01e2a266f01231e3b9deab7d7e45c7699c887851b4253e95135666c1bb43bc4b2e95d2ad7f6a1daf6edb127255f8ebeed21f17b4f24e65c5c413a3dfbc6691ea7784a000ae85bd45ea46f65772bd85705340a91dd8d6789177d2fb767b61ced31a7326ae054e9fcf8b166c863b494f58a63c89be1961c93ec559dc9760fa87eab778091aaae4390d00626e90e002634485349958f08dac2781a337aa75a9e7fd17224140fdf49bb19e3efce118bf002934a3ab75f7ff67e9956d91f55cfaa5b4b08caebf9f589599b37b1d0afb3f22679d311deb99fd7dc934307d97f325a6e44f9964cf8efefb9290a3c2cc4e0af8090396baeae48dfe58037a75308e93cae499cd41c9745b6ad032c04936816f4973fc8adbd3d913c3a7b3538dde0733442d059011e5fc93fd22b9003fbcbda7781f6dbb2cc402ab76ebf0ac0bf9bed216df18b028ac129fbb4fba212b4a101c3456245adacf3febef19b38483717cc5e813005960c78eb49bd37961d8f74cda4f7bb5bf5a0189408466e06d18ff34745e1d1501df6b99aed59ff1e21513c83f7d0c82796c34aca01115d75b45fbd699bba68de59f4d90b0bf95178b9296cceb7ec127d5f6d84f24abecfd8adf8905c7e1f7a3074d1e81b4fb7f0f35ca00f1017cee34a0d87212b3f09b8f5c08eab9bbd2c34f43b6998bab5232b9a18fa2fcfd9fc1c7fcac6ca7a5f7a5ebfeeb7829e2ebd7b0fc20879fd10699136ddf567d38d075d0797afca5c19149d98d2d248d53a491f4224cfcbec20a15dd4509bf36d4ee98b31c1baa6c2e8e6f9246c9edd315f540ad16234e368524a4a47a906744760623bd99bbb36255ee3880f39b42cc80d4a427d99d31b6dc2e94e9a11c52d513308a5f958096872cc6d5a7e3b1d81f8d013c1ce370bdf754ec82fe6c04761f329a6e4934595fe4ad8cb3eccdefed7f4a6dd207d088729a798ad0a3be7240fe492a3656fb477e9e11c6225923e18e6fc5b8bdce002fdf84e96ad7f8e14417c2efe46306678ab6221435283a4ca3cf2ba010002089d034feb6d69cf7affe2f94b4f54b5c5e0a56a7bb4e239f39e8dba3f540930d5194f6cfc21c661aa067d0b9696ab1375d468b3cd2ee97a200ced5047cf10f25ef48800adb118a057bec722f48b40f069ee4d874fa0eb2d7708f33c2f358b70df31982178756a10403dba7c73922bca3fdb98ffc3242dcbdbba7f5e2af71c5122763e773209bfc167285dc12134c909762d2db92442d18a552aa225c9b160d9d008d86c8083a9d2ed899c4d93e594c3882cce3b20bc7f8e4d2030e11e697dd9c3bbf6990c5b622c5bb6b5935be67ff8954718409395a46ab18120edb1f69107a26d6026daa9263f5ea95d0f038ce6937315a964267e06f28009694c65383c7ddc10abdab9a68f057046a4cc42b1d65662e126eabe49b55e3a4b59ea4f3264e2980f7d0108d1d58b3fdf4ce5a75747289a5054ba644864eca57b1537c9cc9f246b866f3e3c7b98eae7374410a2a908160288bdec5b8b3027374aa4effa93bddc8ed47df03a0c231d809c66b2947346f54bcc5caa12d37d235b1a86d2315b894e210cbda07405d288decde1ca78d7e59c16b8a548caa31d575352c5d773c6ed7fc59fba5280249f83e63c4d4e18dc48492f5c1288a5c2d1401ed2d8543559dbffb4a56c6003073173052d665474cf03ec0e14a5be7d948acf96ec25dc85103b29ff0f60b6a46266a443ff6d8ea8fa377e344081025efb8bd8e0e9b18c7bdd84216d1bd5c12bf00165e33b3949e384a1725457399606fd74c2c11eb9aaf110e7ab80509a1b10dad17169216bd5fbe8db2ffe42985411ea0dc98159ead1326a75f03f9b69c588f72a48527480540dd79217fb67417fcfc440b2f16d46b1d53d5c1f7fe789c600ea655e851bc8b3b344ceaaec4eb70fa90613b638e8b759349b3321526e6532c96dc9ff4c7d0fd69b4036af06e1d18f80a512124881544ebfedc82a1df6fab31047e675c6692bfe32685ef20f0196e0fac454e12df2ae5e93cae111303d896c05d90003495392cf3a529d1318b2cf8633c11bdf5b1d26706b1643cb7e3e869fbaf25c8cb4fb430ca3c228ba47babf65d87e779c392d101dbebe334920203b13840017ca661140e163d968f2546568e72725423d30e748b41d99f9b5fa83647cf61dd9559c24d7e86c3f3d81b60238cd7071b38908f3a4a172f3aae2c8d25908ce70d7d2222faa2ddc393568a020ed0d5342d3230adcae0e6e2b1313bd3c08c98daca039f7eb7977e751f3e669598f4ede34fa0184891030c729f16e667792a1d65c871afd294bba0138df0d2db04d93e46fa67a14053aed2a0a4b54a9735077d0a4b63512e2ca0fc6ac5594b1ee4a7007c4b4a97f1f10aee7bcf1e016ac8d3d1f2c4a8a63894087dd825c1ab8c6f9dd5a88aee3058d538693bd5aba58f374d42212c1dbc482b6a9c588bfe758b12630b4e928f34b2bfb96fea4a5b095763542c5cf8d9f56fc699ef4400e0dbc5ee3a9e5ac15b27e1daba245f546c72560fc9c55bf9b895466635673f5c3057009a0b7fd96b29007c1e7dfe3ed2bc21a19a81d1946c14f0bbcbf08fa014d16e8838033344d1bf5ff3f14311348b07d6579786d0be9f11b7186db3f993c4a7c29b3bb2ad8f5283f79084901d6555b2875a705afcbbc77918562e3bc686f04359b84962a509cd058f8604c909188d81e17748f1c6d90b9569d593640fae61e1e0acb71610f5b888513687e589c1e42f0f41e4cba1afdb3a127d562756d5d1630ca66fcc093603cea34bb140264014fae2d62e689a5bf2704f30d48d434aa1a56407d3da0c2b3e4b158b9007495f1106d16a4aae5dede6c45348a1195d62f9f35350b30a24464a21a6d6b0da051ecf3c50beffc79d1429f6d9638a0702a33a58a1b5b296ba28059007e15e9ca5bee4883df232abe5f952037561d3a674e3f13d9887c2a9c2057f01490410fb01d42b75d6dc74274efd63164c445e01636485011f7dd84db0d03e2e8e75fa61c7b61501fc8d1f1de9df139baf2fa1f98802963933add4abed6047331f4638bb97d0a899afe9e2c8f136c979068aa79e64a978396cce417a695d97c021452e98a1167f052d89821ef135bd994c9da496e171d2db324f1f0082ead9533b8f070ac43f6bc0b35c5336c0d9cd97d2bb99b98d985da4715f2864764926a92106009b6d97e57d83b06619496e6f127110696b4e646ca28206f5e8c66e31867272de36a688e415d42a48b11e1b71ce491183140bf2fd44a0008261abc257b2fbba62495b2aa205d86969446a4dcd150c168d1d7f12de97c670e373005275d75f18a0821a574022986711231829f62f2be21631e2a751a0f5b9a7cb7a132126ca7837ee96c5850885cdc769484ed9e3c2080ed4a6f883ae01c3e1d61aaeddfc326ee582b59ba146d165ca6b0bd9028ef01c648691f931f1120c9698af2b08294e1979d656db108a19f7ddb437d8ff04be0e42f4fefecf27c149b29782696ac26b6e2a53c26d22689ba5963b5681decb735c1f32eb5432eea620963a03daea9bc29565dfca114bbf29c5450631e212743a8a973402e82af9941497a09763625176c64a345c65b1aa128eb775d83fd23f5948ddc7d5c8523ea954c775be11cf63a0f1e7d3b7db751d709929d6302205983b85d9c04b50a51dc27a5367d8a84208d12f21680c1828511ef275230e744ece16f83291b1618bff0d06a566d264bc9eeed065f38ab34ea3aba65c514f3d0f243e1d170dcf35c4cf3c0d1bf47acd253169ee0142ed39c8639a1c85f463a7f2a1be86cfb3a16b44cefd23038b5a52fe4ec05b6647f2b953661afb0178bb38e657c3f70bb758310a6e1a3b55c539aa2eaa7aaa9569a195580d8b308142b23928c9fb271d6fdef685ed777b6783c7ba467f5d3a3c72db3a91a524952808ccc8eda52460e19192a5c722f9ba4fd3cf106bc6b7d3690dbf7285615575370d0d8e92690b3a1af9c12e34f247ef72eab0c6caca4bfcdce40627191cc7f113fe1432c018e6f016504a5e30c2cf623e89a06e34789a10db91944e7d76cfdb799c125ca9d7d2f05c1e8329002aa5e33a0b9081af2e12392327bacafeeec09b6fe5afe7ed3d567207fbe3a25fdfebf9a8c29aea7c12f01ed2e1fb219291b4adbd9f529ab78871738a6e2197dfe6eb825bc5f52218d7e5d93aad5e54b7c876b32baf0a322640d0fb5cf042feea8c1fd216082fa500639dbbd1b033d3d586ae09efa16aa69631c921404a6adb97e3892a25425538ab7ea036be11531c2543f80cefc0305807582e73fd0d604141f433c524fca46269deaa919a6e94ad08f2b199e7d0fd01fbe2d53e6ce5cecee68fc8e570c838fabea476edb9cb310567c2f9f0dde9559dbe86cb115f14446072ce901f6fb5cec4d4e72a7c302ee4310592316719289d722718553c23d19c2639807fe1315d7ccdce350514fd28f63ab7bc541b4685698808b989d73572e20741ba0590070b3bfb4ce5e5f1a0804532d923d06d073b5d0838d5420c9f7ee577c8a6d7744843ce69f82ac2821ce542ba67e24a8963488d4d48dc2bf1386285d9f13b87eda6d80709d125ac86f1cdcc182f3721617600890e0f6bbe6483f4c90372705ec3b1b0d251adc2ba4c0a222d2fc703791d0a9a3284bf1b45a641abf44486d190cbe3cd29ef63488d7ff26bc7026d52b79e459e71de373d0b70532c4f20683d8677898a835d0859951a561dee9633e757e08b86ed4532681d982dcb1ba1b337bad7b23b1c27eca45ca29464a574539e9f966d493b99a4e4d4d291dd8ee27d15f3b6df0b80b5b521b5b6b2a3f325f05f8904d6a6c4b896fd01940de108d2579ec6cac9a57a466c6e515fbbd86bd6543c71498c22ab582d0355cd9b8c0a23babe66a0f76c70c0d201be1c47a97bd082c1d28aa9fd5b08f90a8ee067630c1925da0f2b7a4fa42d33b2e787c20ac2ade35cfbb8530e5e0154650a24000f8bb5c74e693b34d81e4e0d6ca2884d07844e8aebe92fa44e1a336ca2864653c14eec4f1393718529951b4a8cff69829ba63c8fea426cd511c3513419e362628af15617fd3ffc43a6109c123bea5d196f1cb5ac18b6fe673f5f098beafa5a59c9cea8bb89427cafed1ce7ad13c3fdfa89bc2343da260b0dedcfa6750e117171ebf9f6931efe0f3e2a65a5a27c6315240a83ead59384658bdefdaea2b756df32d196a2aa858a1c4449e9b87fb3b8d100f0af98787154f2ed9b4305930899e9e6ff9c2d09630946ca5466dcae376343b77312af4db586e696079553f0c39c74facc2b7ac319f97d6912cf7299321d186471892b23791ebb13ffc1b43a94aa5705a6fdd290a65209a49298fd6565c5f181b18a6b81f5b98b572eae164d57e589b4942f9cd0b4299df1bfebc47d7ad94d10826c4e46800a595cf588ca50ef2ed70e617e6924965fae9522e8f1c541b96c234361efb58841b2c68b0cbd1eb04e7b1cff75d71b098ba6e1b6f11b3309c8864fd44cb639f4fb9f3fc7c8f4753fe31b245b866b61ad8a4d5a390674fe1b807e30ddb33c0c4da7fb74dc2face874f5a08ccfc6c688da157cc31ec0dc353e4e4e4a4b3b19744997d00bda7cd959cec07470247bd799642b8ebdc3fb9d5e65d807bf0eef9ca338284e7af6256b8f0cfacf793956f3972873e4ad40021f8d9281c902e6a7eb14b3f491b716e7f011415a0b261b705073ea7f7b192cd71c5170f6164cf0b3677e75b9bd7c71dd186fb7420ca13a39e8646aa91f42f88ae2d86b5fcc48bb6ae45c1b94a5cadf1a002c2bd7aba257ac2458f36909e7e1e95c190e8a86f63ed57399ab18eacf91350559c33916d2abc596f5860cc79f5bfc4d65ebead08143d3bc8faede6368bf0f7f57670b12158642afc77b0e2cd0d88604adbe97e7fbcc2216bdcf2b6e4be62ce7ad15a551ad307d49e050a8b8de634b7cacd525e5ce20fc4bd0f7861039ba55525e3c6df9ba0b07046cd5f8430f60130460958b0769c2625d7bbda34b1c4ac3195d6a892055b724b562b4820d8acc1768e08325051c50f8a6e0bcfcd3933b72af67bfad703db775b33966de581e892885d566328714560e3d8dfee47ac8dc9a6ef0a9547c961fc3e0424f9f0a53af280837b3fdecd7efef4a9ea74568c416d7949ba6c0d2be7574d280a4e193bb70ea7758613274a77e67dff316ef72a46b8b2805dc42702fd344dfa2b09495b64ae029a2758ee9b695759e9866a8117392ee29991f6cf83c7b951a1cfda7721c5988fdfa2425e7e4ebf0b1ba702a587d29830e43ac39b12bd1e8a5932dc0c7981af2e36d7a55076b67e0a9f64bf98fd05e5e8eef4dcc2dc0176c6b97a800124d7bc7d411e91919a836ee081844c66318723f4fc64debe8798cd66be42894c5ff022ea5977ad9249cf9e556f3b857a5508f41d6e3a2bfdeeb564b5bc76ce1e1025ceb8a0bd62c6fc94c1ad281e2f38b59ffcce5b60263e5e613adad51c4b6c528a5b5a9d4448f0f4837be73428922eb0077f661618fd33c5692641730a94304952da52c2419c9cd8e1ff8ea25bc476fb1ce9145c8b71358ed447668e13d02dcec8c8cc86dfa38a27c87cc643dce43800915108ebbfb78f17a2363f1a67b28a0100ef49bc9d57337f6b90a24465160bbbd2fc91ec8097585d2efd99ac0f7b5454ab014ab272742a372120e63c0062ed2f9fb1745c61cc702fa08cc1b428dd2f4d3081b042b075897f66e0275453a8accf927c405fa35728ac75e42ea4841721d7219e8739b69c351a6ae6c1a77131ac957854e56d6b513dd1068f0cea7282eac274a8fb5ec4fb6108c7fa50242005bb0ed03721d8687e664c93d9a37aa64b64a01e8710626fda681b3f3ca1fedb172fe54ca12ba44a2602a527eaaff3255d2fd826a5c6e93b4a82fd8c6ff95f9028119c82252518b8c23aa77deeb5be877084ab887b1a118a1082a225acea52cf56384616043bd953b4ff4edc213218388a9a570f5521e6e3ddf5b155c58879cb64e9bce1a96b825d614b0986c821c9cc51995325ff9793f418e69d8ff0084d300a52fa57b524c497486eee3ad9517f1ef845300aadadc4f9f865e1a9f1a9c848547673d44318f4fc85ce34b9fb1b4d3e662a8f8206f8434a6da9b231dabdc4dd8e57a9b929aeaefcdaf3683272f1102f514d39f58dc218958f19f3e2b905a8d48b3096ebb0c496527c02517f58d6d5c99b2140975925ed9fd1d64a3290ce44876e2bcdeae4aee32a34740c68479e019cbb67383a929df6d17af9aa0a74458f3658fec1fac749050582657f798a27fc7e09442023664472f575f3270e378a682bc7f11c8baaa552a5c8b6e795f37e6a45ec0b2617f5b6f1ab3b2db3a9b2a209c401c5e8c14b133fbd5d87542aa949bf3e2b17938249a5805e64436b73a37315f95a90ad9154de74979ac2a417d4fbb7683a98a02543961c900eaeb8d924e80b431d93f81e93fb210eb67796c553f1002a3df31613728677235e4d3f259b71b0adaa38e44853272d87f2de8a2522a5c0b957fbc819957530a70a9c61d0cd0304b8c4e1f0a22a80ce8443440e7dce257721ecdba06372e53fc6b4ec0db39b4729d572f47143d87a42b95e85876aba8f6d98931d12c08400aa8d692a76e16153b0c65018c38e5d906c46551b55e6a236b94808ad23b82c908a850622830e4c429c268c5b4e3fd107a9ba95ccc5aaa5ba3788dc343ea22017fbf4f4ac04e28887d6a4aa1fce40c49787b5ddcc099b64d831520a9c1d733893c34a19e0cc0766a01c6c7f81e0ea506983a6d342980f6d60999968fa8ca14fa63b43139f2b44027a82031cb8440f2f69bfd0d461853257a624f6bb184429ba760f4e70b5d681f99d856a270f784888ba4bc9fc218809c72e37ca38f50145e5c0d0a059862f34ca03fec98aeeea797896cb8f6631af069d42be2bb197c8126ec82aa3cd596b72f59b1692319d2112d9cb618f6434031a9630479283a7cdcbe896c5259ec9abc359d46caf50cf9353a24b47afbb7e34a78ea93c697d3fff0f416332a21ec64f2f0775fd75cd1212f6a325b19fa275d364a83d9a9521737c194d3183ad3e55ad63f90510c1c1edded7bc0cad6432c3daab2d61d91c72e7d8dfe84449f5864f5526e5b30f62bafca6f661fa99b309cc67ca33f7c93e7759f8a7cee04689f0b883d01b1bb5025553dcddfef5967f21621c136870fb11e32fe1cff78990f73a3e5dfcaeb73ede71656ff120d4244ee47588a600c9382c32feea6520deeb5ef4a206b535496577098b2103ddbd81921e67570b857bf4c53aa7cc309aa48a289b3d14cbb3928b90633f71cc7842826a9f62f6a995baf9423c8e90d986fe30208b99ab516692bd9c6992ce4fdf73b1d9084929ce4948f28304ef6d76d40993952956c726ce914f9bc3689b1c519a6351bbb914e2033736a323a8aff13006d73ad8ee217f8cec24e0c802b39d6ff67479486fd2a7a33925a277a309f1b1af944fa8c5d8ecaf8408d53416b32d8b65be8d84ef16b27266ad56d8190ba25db44ddefda6b0b00e84bee916691ce2f290ba7dd0e32c25d72c7bac795395197d8b7338791572b75114c2be9b2128df2f172e05efe52b16dd6af6ec73335654f2836fd0e5a7679682bc0e60dfa6c2dc3b835c990c5ef945839649ea1f32cc55fea8543ba61e0d0de6c14c729f5c96b7ddbc82d74185efb86f770fd25b3efe9100cdbd7ece1eb4eb322156ef334c156f49e058700ccb985ecb072eeefe5adf4fd22b2e93716337d5316b1f21165a88ffa23a6312a4d2b45cacf460308787ffee349548f3821315e1ff5042ff832c57cae5cda12ed1659dce02387d5133818ab73a6198f4fb62252eb1430e39fe002191d15f5ba9e8110ff0d70d5baefca08c994f13b8177df49b13849173b854574335da7ca8c23a41a8bb1303ef83943deecfd12deb6c968370d575ead2d5aba8402f030f5a099312217f9c88f69dc3d4e667504f5be0bfbcacd54daa9686efc2cca01bf3aa53307ec3dd52d55990a6a618898c43ee4f4f57749b91a548bd82eecfe98e087c9173eb601afc5f0e5ee28183f0e08d4d750163635bbb0d3d83fa53363b9760f0dd84d70e05214df90565b93bb56856cd71f3f261ef7bfaa1d3c07648d73b068ca72281bb194809feac2817a6128102761d1055c69cbcc98c1e680c39ffbd706453a9e801e3e0408f491889e7032b55a6425e182b7c4036e2444c9c6a897b2a06a415cb30077e818af5b70767bd6bf171fdd7b7ac1a2ba5fb4b94750f9185939c59e935c3acb46537f56d40b35d23705b718c938e8297478bdb93a79242a93a2e6b3b84af11a9981080194b662107c687f14aae0462ff05f88ff4c9745ea82df4ccb7340ecb32853c423afa192700d7ec7004730052c259ec5e6f5c5861fc625ef9168384cf0d59e4d027ea1305c3d1251aacf7b8582958ba38a3631d9be668639b80a0437b364e3fd2014e39760b29cebcae3bdac129f0b8a0d18dc81018cc0c7b0ebf79a9f9dcab1db09337946ef38ed3a497c9a9eae9398f1754c9b00e41641e47170ec05cb174abcae7a23cfb6cd1c899f028874f0b71a4e34bdb4dde795acdc8c817ddddf45cd8e9d5f71f2575089eed1c17e2f21f7ec039d35387fd7ffc493f88e7b094562d5014a83d8b432efd8b1974aa7592be5506dbc521b67e9afca4a01bd06047710725e54877ce147c5c0bc86d9d0d01a85498a72ee82d5c3a80d5be7c8ddb462930993998384b0a04bde2596e6e87937a5237ea7a945f56fda5220367030fe6810956e8dfecaede81a6c1b48606efde6ff8392ac5228fd1e60af0ea8c549aa73359cae9a34b6f771a95e260f9f90b31cee8c2b0ccdbef11c1490f8f8573f6c1c4b49ec6c6b5b36c92ed933f3dd53c6a3fa0332377a0ceb800eeb1c14972a68da129918871b9dd7731a7ca094429c27f2a461372028df5710773c125568f2c35f179cce3b789b1229bab4c129ee4ec9cc7fc9ae1caf55ec1fe8a3c2c85fb91a73067e9193880a1e39eb4bb18a59ac1b44b90d8adb71f1afcc39cbac2b0017c0fd7558a5ffc8329477a5a27c13c337ab402a3f759fdf8b97317e1704c4459a4c273076566ebec7988b341f963024fe1d0fe4d1a9f5adcda466ac9a242ed8d2923f161d46100b3867ff1b10528bc07f4e47eea845d36e479aa446599ff701fd3a129bf24f32470067efcc5a1ecae53415eb9fcc5c06c948c99acc7f296c7ade4d96a841e484c186ddcd8662cd78147451bfc9e005e51d7c3e4a4b671f9c5f4baa0ff42fc5cc82a035fa596dd61144779a99a12310a25edd4980ecfdbf55f942be5a12ddd0506ddd9fd15ca2b92ff632efada851df71f62f6136b322a98c6c0f7911b74f47c6ee9c7b52d4e77cb68e17e118ce0c8148649f055beb1860aad243356a3be952e9b8a678671c92a8e66975f13736607a8c6b26395d6f20bb65e34d8f9cd58a05deb64d0d3aab0a3c0bfc9d10285c9767767381f714ce4ebc8f688143f90759ed5bd9cf4941c2e373a53e07a1f50ed89f0d14f74fe5eebf52187906d4479bdb6e8f288cbe6ba1c2a566e1fc1b42283044170e46c9dabd204e2a0581eb528ab9fb2cb05f7bc9e3f56beb05d8929ca7a482571fe5f4eb50915fb1fd3717771284b9979c2f917b8342937ec40606285385292b125007bf9eb5df381eeea84dcae913a0037dc4256a7f8a71d6bbd623e793b8c3ba9f2db73330591fc9ffb3d57b1e69f02510adddfd52af675a846097e7f22ba9edb10cc5e809203ce1414ca583013520ad7d966ab22c908b31024cfbed3b83d6d2f32dadd778fe86b74d466ba6542aad1202c2031fd61021c62774502744b47516bac5a7e26ff6f450c2d1be3c14e25379063c5e7be14f33c9d039eb4c014c2164cf57d55b6162da32a29b72d0af394c7dcd7716ecc2cf71283658a0337d731b88f6e5a09b624cdca48532e2bd820c50f23cf886d3ae32af80f229120097091b0f8de5b9c0174a87ede3537a6882faabb3e5781cd3d3b02998ebe3d399e11779872d8c4be9e07c3b6112dcc0122ffc9d2a004ead5d21a5e23acbc82ea14afea545be57f70abacb9a9151a2bd088fb1a21604767f80d5dc286a776709e354e739e2c7637b98dabb804a786eae11249659bc6182b7b0a67ae8f5ee6d8eae9e7c95c67b77a800476c9bc326710bf2b1fd747824107bbf566d20b8b02ddb3de4f03679548a50841e39885c27a15cfa6010ff53a4fd2e7eac038b06ad03ed32dafd5271b94c83c02bca4d723deab2418eb70785bbf5c62cba51367d8d0581da64e8326c618d7d6593bcb77270a4efe30e66568cd6ddce09da1ea71f89003d4b6aeec0f3d657041ca85da60d8f17019aeb44a8a8dc6d024accff0a95cd70328aecfb61023b7967c20a30e3b3f620762b58174e742506e0e1b1240bcbabac0ae7e93f73a7c6f1f969fc25abd22afbe055436828b56ed709327332801de6a9d4311cd28452ea417dd2dc298305abf853b6dd4c854c1a690fd407b45fad5eb4b421b0922aae69984dbb0488622463b03c40ac605fdfb914a998f8e4c64fe7b4e981b2c5a8e6144d8e3e962854aedb9595152c530a95b97ace2b340a7c9c636f614a8db066cd0d710f2aff86f2c203e7bb83d1c57de8020d4936ef10b9d53e89835a0886102a70b5f6dc7a539e69acabb027c681521f823cff642e2c8c1dde208a8572c1a48e03ca93758b937cd750a484c253b810cb44b849eb99ba2e5e1699dc0b07f6086bd0a8a56bf82b1074ecc05964fb67ecfd2223e2cbbc90f6ac39dcc9580b0d15bb732477358405d39a0a73aed10a160f9da41fda76b48353b674b8c42aec94d3bacbffe305677ffc5e8d8e9c6b6470b32c32eca33f5201b5373444f3a1b2d2ca7444552568f0a63027235941e4927ad864efc7d33f753cebf4184f41924454987ed7c84fa03c26487a7a4f1c7d4043f960c41074540fdf83206cff1fdac0984770880bc7bb76985ec3fb36a79c75b579e55bac76f16053d22f87d1f029b11970cef76a2f8256780ab6f32e289b24fa1568a4cbbbd611204c656bb56a7fba05d183d9b3d6123524a000b1e2c3159c8854c75946876253a72a864005c97dcfb2ca7218e04c3985ee4d95cdc8a36b14e65510aef5f6419c5744fee5d3f6110f241f1df649c650fb43b01b1c8f440598add81e9758549c76fb7e7d2ea0d9f89f01d2b3cf5ff702681b13428f43ae46eafcda71580bcb36593a4c7b7cc9e6a1a9a8116f8c81493b2007a26205d92b3a0388b4d015e8e1c07aeb1a8daaad3da4e7aecc6a445ba7d60b0b25a3b8005f2b221a4aee638bce9b71d7d9056194bc57a6b82dca913bce1c6a67076c2ca9ea9f603365092ecc1a272cc96d7debd7603b00f8b690028caa4cb6d6e79149341b4bd316a450e148b9746780e261b1478338bafd55b775a1521daf5430c19d7ddb54f1644989e05947eecefc8b920242f029250cc8257e57eaf8dca7fd7c5f5ad0d3d27b5938e0da17a01c8e8b5e8ee6859b57f288ee89e29a9f5c5b58011b6132cd9142d2ab1f40311cfccca2b895d0901e8d3a5bca9e5b0e900787b08958a4896d1ca2e44db987db3f414c8c2d87c3ca693f9e7de53a467398805d33ea9cd621900262972c536dcf4e6bf66f56fee83c8809e7e0a61420e1ebbd9d4a64d907c31da07e6d6eb6b19222901c96e60e408bc574396c1630869df8c01f4165cc60fa8d4d9dd5f649b3fb3318b0bdf437870c24a72c03bf8f74ae3066fb866ac8aa2071fd3a301487962ce192fedaf90187be8e4c9fe0fd5cd5164543f757fd28c94737ccaeda5b21b1698dbb5d77cbf594ce4fb5d3b88748ff6161a30c1084c68ab4f14c25b83b0b444ea77f094f123dfb518b4e0b67f4bdb7803825e44c8cbbcb66d013d9ab3d8b56ea5de4661f1794b7718d54933129b1e1e49be1572faf5b545b5e16c3212e2f82854327c0a25050dafef3cd55e1939b0ba4c5c59808d153e5e3ae75f0b6a1f3a65864099f22c72a3a664bc66abb1c5b825bab0abda76830521fc65aa54f887b5b585e1b95aa702ec9d5549cd6a64d6cd7e27a2ca8cbe76dd1fb5b5d8a45ed31b7cd76d3ad9daabba56e4e0d513b749ef2c00a42c4d0257794fff7b39c46c567c4be97673ecd37f466a570b37ab5745ef27d4f5a918f32b321460a51f1060da399df00e2a09c4e0264c4070dbdad1aaf84217149114c7c65c7bc115b3eb7d3144341807afd8a5407edb65ef0639fec96ac4187799c14e960111b7c046d84e6c79c543434420a8e3e9f15a7c672d8da655690ec281554f3eb2fb2cded00ea0a5c5f3431505fcfdfcc0281d34487b32590a1676a35fe0ca112e69d190c152f4d9838ec7f5ec6e4493ad1e348f7784a17ea8c1ba5a7897cf51eed143b58c84546b71b71916a52528bbeaeac381661b1cdd7f9abaf221085d43b915951780698c5fd2fcb40e2323fd06fcc9115c6f319082ec214f3c09055fb3af916677b4b232227661f2806d95709166b836a30a6cf23c6ae08b27020e08abcc0e2fbd9fb57b66a2f74659aa110330f67e486433fb08fdfaeb5dc32e4515adbe5e91f429a504e77483ad1b7a3ffd82ce3ca89678de0e34fdacf1523716633f1414af0332fe1e350d7434239410e3fcc59ef55c4ee07d65644437666e299f7c9b8de8d31bee09fafba19a1b3e5e499f0423d5ce585280ce745f0f4fd97177f5955f508983bc9ed310f4c1bc81da8a0ad8533c58f5c7b1694340706cd2ee5ef40c3f39b9ccc4adcde5f84a7f34d1db10be6e686ce33f502020e153831ef5af685b6af485255c5f6fb883d1e5f5d0a784d88d981c94c6e814c18dddb912173865226ffdf0ba5b5246da784af13651a62a894035b5fbf5515b3329ddef70729c14cf43842004916b599d43ed1c64b79908a2aede2d94ac4e101c8824033f62e92fb64bf0e1489f890f0900a5bc59c6f07a7c427f9fac94cdea070a5a144f4a8284b4858566db98e540d0113826a824765c5ff15f8b153d81dd0804b8cf0602203a870c0db766ce8c7b9a0b01567ce4da1b307d20e50d16c9f3718ef6d44637111f944b15e25aa1c4006fcf96826e7c44af1e88f14b67879285f6762eb4f70d0086640cd88a6c285f2c5db3f32ca8fdbe9c1edbec70fca4fdf50f3c6d3aeece196390f856b6c17f9453823120382ec5d47b2b343c45c5e153bf78482db975921120f3803902d9680aaa3af4a007cca997c53a718522e017b8970a2c3e4df0aedcaaccbb1d2995f4ee4c917b63e8160537a1decd1255e0b4682c1cc9a421478fba3e95f1d61878b52435ffb7031d31d292c0f53d108ff0bd83c7069f49d69b2bebbdc0644bbf85de41c5365615ae6f144cedbc20f2aac1e35c03408322b2492aade903c34c48b28984bf43ff72ccd0c8ebe8b641b9c8c7e277e4efe180fdd4336d2d8f48245d9ae8d161bdc03c406e7f9ec05c841aae12bd144f41cee97e30d8a2ec2049c6aa043a575f48a7f4385ff79001be940ca2a48683cfc6251bec924b3992fe2f4fba793918a66c1521a5a8633d569467bed022a83585ab1695afa1da505c90832f05729619770278f65fce26209c212625341b20a94f6530a42d86f4de8ce4b5dfecf94eb56197fa1875c31df49ff7d120395afb973d4af68db0edbbe6fdac279d5a33fd3788de07f7e8e42159943a7d9dfe8b3df438088f2de1b45238e09dd2bc7b57d51d604ad7621291b4004992b3b721bbb9dcd861b87141668a475d7cab6f1cabb915a8ab01c48b79592e99413891aecaa59b8dbce78cfe3e0b6c23e46ba1a005094d8e852b1bc697546d785b2cf777c93d1cfd8f4d85c0d0e047a077d336ba7ab43f94a5985165c9b5e4c8f2fed8071ed1831d38a4483f1dddde0d0e11a1dc1e14616f47ae1690d193e967d0679e4fb347f88af6919942e97af6b589c6767f2b90cb41564e0528a3babce8540d92f0cbb7a9bc5c917695a3fff758e112d68b0e24f03f5b66eeba9bace8489f3cc0b8bc4a970f637b8729325cee20498b1460139162d03c64f12b7521eecdcccda0965ce3d9ffd8107530033b04c66719aa416f2b6199fae7502508d90aa7132c65256a53d29e94ef29b1ba27dbbaec20401e4e728e0330d1c8a578bccbf6d40731c4b2ab71dd2e66b8d06b0b546e98e50cfb7a7dd473bf0b488f2e7accdce01dfd12974d47706b8718f87c9f371254b5e1ff076e2b7eab773722198820633a8a73865f7f2e7f85978ac2e08abbbe8637f80525fc5e0c72235e007721cbcb37d24171ed5646e59fc0b5f25ef4aa0f244d0f01267cc3a7cbd2b0e25a89cab981fc2fa283bc98a78b87539afd4a96f209808a253b3ec2b19ac89e049db56837820dbbb6ccc8a37744535d9f5eb41ba508519b8fd0eb8187b406c40e1a5300edfb5967214d337d351e9da00a38dfe075e58e20a4d0ced140f72de900c763fbde4ecc0b7f3cef43317df725771c607a4e947ade789d454861fb08785efa581ba7af4737480d83f294e445130ecb7fd345d8d941faaaafeab88fc68ac76e2fa25c50b1ff999a8b5184bd3bf8215e5c6b426a4b7a12332214f80242c10a28f219743fe2e87c31d6dfcb2548bd4a0d345acafd2916e1a84ee2cfed4e8538e3a1070c74678324c3ed86d35d04e77cf42d51bb44d477e0da1b04412676f27a711f8c210a4690bb92864b8a1f47d8d843c0fd8c1610dc286c5603f435895af68f8e3ccbb10a05447e064cb5e64153510189115512d6556eca8cfedface16684122a56b3e897368360b65fe41e58aed9bc625046f739b816219446c338c27246a6f62aa24c393678c39ff53f45fa6a609dfb221b46c5ac667768f263c2ff883ad7c4b051418ce4a64700eff32416f882aa71b58123368a703d1698385f41496e17858b336cbc3669447db8e072f870368819a83ca990e246e58dfea891c141917bede55ca3c460490ab54afe6f38ee3aa5a6e2229c63f77fb7336826ac7bd84bb7bc587887d2ff35e8db2710b52570cc368bbdfc2764008cafdba9f1025213611cebec0c3004c6524d79e2c54fca12540e9722ea0f2a1301397dc50418580322676ffbd0d479ca75cdeb28f105f62a86c1c973bfe4cdbd79dcae8abb76f61510ba07e5afc87959f51342c8f2cbecd31b6a2c51e4a23015d94139b9c17720eaa7f90313c3a3719f8d06736cd1a3cecedd4b9e866c6b93184db1bab5ec1d6a8dd680f0025b6318896dfb632c0885bcd82d34b7d1232ca93e0b08a558daef08c65ccd3a8f306eb78f8743fa14ccbe1b1c6c4c81c1d3acb40a20b7ee7a3cfa09459024c839a78dedd0a45e76ac508327eb02261a5b49999ffba09ba7b9db72c6503176c1748867ce3b38eb28296e43220b03892722ace3b965fde13c8bc9c04ea038ed253716e078366f6b3664f2ac11e6ab92bb39b65ecf6a885fe7c77bce6f0d9015d72056933b64456dd14d338b15d10b62650075cb4690a32da89ade6f48035eb12ab8df6db7346c123673190887aee56f5a2d7859d172ddb956eb244cc5468cec9324817498d7a25f86719c226cfd429a361f0d93006e716228e7086286a42e8b8ff2c429cabc7043b0a0f292868bacc54b25e756b968be10bc01671fe5b54dc477f2f3f97595b3895028dbb4fd04f02e24db5b98e1c844a0901ee0a752272b79b1098d20957370dd531c9d5838eb7902e080075d3f0dbfe47f7b2862ec8f807197ba07f36e31096469f689524944b68c7756270664d4387abde7b3e32ad943c8c5c4e9a84300a4300fccd83df9d58455f8da6ae54478f2ddf4be91fa743f0f562b6281a02b3f8b2092280886534ff727659d30bb2bf2b5451141a4cb0ed8607f38a468bef806d39a9a91e91b21eb67cb3695553aeea07ca155078a602d0bb9393ce4f633e2e2526353feb43fbc91570e241c359046fad23627f64d54f324df50f56abb6eaf3aa6e1cc7c1886d11d75cb75f3b84ab3f898c6a068cb01d3911f19741442a2b02c01044ead1bb294b5f56022954de7e0fa995162df721200402f50a51a76c6e71a7269cec907b2bd66e664825a0858c20a295dee464aa996865033cb020a395facf140aaa2f96c90fdb71776db7754019bbd36fb024784e4e240ffe787350d75e6b0f06c05828a2ed14bb2588dce025a6fab913bcccef863f3b4add29966cef082859da60f03cc9e468b863d51cb7ea774c8c15a261230f9e9bf852263879e76787f82280a9739014cdf6d1c53e35c7b6bd998ac2647538a6f4bd372884256a5cc6eb6ee43641bced83214a96ebbf31a4080232d3dc31caf8035cdd81481e05a1b4af10495efa9374b1aceb3e0a9076ab1141b5954342bb0fcc88ee25395c03d413c169757773f3b43b0ad49ddfe021827fcafb2ecc8b3d45649f2c97fb5d107f4f3332ec1547b0a63075a8c9e76cf3af7348a69f5e0eee8ecccf5336bc77aebd06800c801e9e9da957393103db04a7ed3d6317aacfd222e8442b7b2a37254cfd1e1c5cb3cc37d601c58c6811d6c16263118182ff5170b2b29a028370be08d612ac5ad7bf221a408e29c7ba5b2dbf82a2a027fcc37f5faea142af383ea5190881dab915640a2a24d90e0f6a8d6e485d3b7e27d68c31434adf88484631dac936013da6562a3260b11fd8e2ec69448bbd6e4ae84834370717a4d77abaf0cc71a07f034049f68fc297f3c750069d5e7b231d3c4355cf0dedfe422dd9f8793acf0644d6d6786e85e5c64e431d7c56c5d36cef21dda288a08d871215b71d7bc91563610ef5b79a1320cabf3c56eb4c49acc30f9c254a1be93e7ecbabff3fa09d3ab7758c672bd381e807f11d1007a2f7d15d3c7adae330c25f0b88252cb1fc64e8d2d69aa6e9553b4923aa5399d1436e9bdcb1fba1184c8ac991b693933a798e272d9b65bdc38124e71e47677b7f522a10c8c7812c4f0b8090500539e9cc72c6e915453f8a209d846751311d9159fcce60edf7f2b9cc5a62ee47676117c96323ea3c8047067a86c96d823276ed5c868f41198206373f049e871a7d3847feca5ac58b5560a94f423a7a61547697b5bd4ff5bba3c85f2500920267ed2c631ff46fb9df26e9c4e75c1652fa98f240f9d37c801c2ad242e0cd5a801defae3fd3c8b3604c751cf1e1aeb917ba13873cd501d77bc8ee037ab9a8555699712dde9c137d1cde6c939e0a1e04312ece31b316da0149c04d45106305eb12da5453c18cba0a6bb9886c02f363a2fa9cb41c70a267b21cf013b03dee9656651a04d52697be2b4759f12cc7b4c976bf0ac98a67f213cc973596b1c80181f652a22a1ce5163b74854dd5167878a8d2a068c53c3156128bf1477c43750e9d40c3c4e7fd6e7d4b2c3a1cdac6d9b67d8e50946fc435b5457bb2d446e6711ef7760d44e5ae7ada14ad636b0c966a19032b5d5e8ab44a9a4d451577b9b1ab596590f53b8a17be6f4b679aca19666da99431b2c3664eb2b12c06b6ad7913cfb11de184275a55fba881e1c8595fbcfa8b5bac5eff67bb237f2908527ffdda2ca61536fb2dc1f9da62213761622628958ee74ddd656aec3c0e14e1af3d36d45e608db5b77b67cbef803be1c57b369560f1df0fea03eb96d5c0809619b06328f32b8a35b798a1c9009366aaba8f97384f6fadc0538549e493aef702f30d8ccc27a77ec1482604db09dfbb7d48e367661f1598d3401047be91c69d4b11f14c20667b0bb2988ba231a3bbc2cfcbe7c985b2a8a52172ef2bf5f2e91e71cba8af14ca42986d22a9359afea2edf2b25a7b731a5cfede62310fd09de8ea154667bcb903d965be26c37bccdaff8d520b281daa2c44322615a0ebc29774a99f6db60ce6a3741965e42317b2d1efff03104cd94bbca15a05f6a83a9cc63e50a0b092e8b43105a74bac1a11af1629a9b4e13479bf6600671aa4c3c0d88dc453e8155fee5b2c49958b8f7fa2814494e28400f2bda5b40eda7f38bb88bc673fc79e4daa41e2a249f974e8a14f9a44823c8f688f237ab544b0cf7b546268f31d41f3a21faf99f5724142eac71c42ff30173e01fccc11e48632d3b7936fca017d61e68e0f70eebdb576e8e70538cad4d5bf0afa74223acddaa51189ccd0d84a313c4f313ecbd689e8866ac542746298b3499a96ffe452b7b8935949e12cd48bdf654e868f6e24bc142571c15adda0b48b0ceb88ffa443e7cb7b933245702ef45200fb3665b05101a4157835fba3d9ded2f4ba8c2ac145da7d56180edfec7c45f56515f0446f879177e8241b5b699bb9d4087fdd9d470a7273c104b297a70929c6c2a4990b9a44041c2a3f18a4b05e0cb01f3d8c2fef4a98566b8c327c2b6fb854619bc26baf354a905f989f764fba0be06771dd7bf22509e00e7993c74308016f5ff9adf229ad21b114b45c1ad4db35b616f85f19ea294819c9fe391a6dd02969bc71597c7b761d90c3f1a3dc5ba8714d619a38e1f9029c929f974aa9e835943d62c972e3c636b635d2d972977564ece3f7fb1c713d2ea17cfe755fb478c4a11458ccf11d09caf3f76567b98bd7ef54df84da9db34318dc8991a129e94ff5dd2637a14ac8dafa922cadfbf2b8b624ba11f026305e180f6414e0b07c3a475fc7026dadc8d71d89db473eeb94e6e92e96cd6ee689d04076817a4d2c44336bd33be9d59215b9b6d4c24fa0fd5fa60c36d26618c66342d9307577881c357690e79d2f5085b2ce675b77dd0875f0f2abca95f14b906c58fa3a1fbb0e6cd001b148cecf48d8d46ffe833e71bca59ea35d5bfc2eb4c3f7f6774aa8984edd42178aed46c94c47ef10a38ce30951c442f8f70c69244908cabc3f8c3146faed2cf27481ca55a2523986feb6a2a798153b64aa2bde846db81ecb0d927809200e9d1c7e0ce93195906a3df5bd02c5f3591e570fa9f9a26cdccf81868eb5957c611e4c43ee099926a3342b199ac1092611051c6e807dca39489ddab2d041fb3ab0038842d159d0c90242c29a629c51a4bea51b2ac478ac420524459615b6e7509ded4c571607a780053e2db6020c34522788a38d5c79b881b9f1a563ebd68916c9e0842eba64fc1851f98aa7b37ddb6071da81fe443bbd3c05cfe90b18c4539d57c79001600637b9ecfa59f9df5593042c872eae10c0764cc7a59abe1e94d5652eaf26be5393b596fa1122717bdc71cb0a437f061d43cd0902243d48620a3ff6f5fdad9536086198b56731fa919c9f215c393866dc6225ba163894676491e2b807cc2818022c3bbbd1922e2993beb38aab8391ee9a90b690ad892f3d936f01891705fe6162692ad7775402f9f308ad3f7314ab86142468c24a101b3742e2dd220ec8175bee8023853f8cc1f7fb1c1d978e6cb06fea5d2690fe95d1396a512e871fbf19299e33ce6c62de8b0274743b0e60a0779828a127ae7d920035cd866095a9f8bb70f5b6aa8ab603ede0ba80beabc08da2142031e21a66d2991dff6917fbdc3bf86068e2b070470be8c3ed3002a33c34e11b6f4ff17a454e72d41a148777c962e1e74dd124ba410b7cba395c709680a3fbe5e9eb2dff02142d277e25cec2242085925481fe185c6ca80a34ed2b56cd5a4a989a62bb3a5fafdd33405a7b4ba7d69362fdeac06bc5adc64aea2278715c8acfd2b45cdc6a433fe4359ea10b6e7b445767bfb0115ea4121a78513613822383d03049db73eeeda00d12a43a5dadf152f3630f0e944eed88bd1c44d256e53ec66c65039fbd0c555f74fd77370c89752eeea6595cd0ca58c171097889f4572fdf9918954182a3349cec71459c231ead91df2bff5437a96e11b156197eae301bfbd7fadd5103596bcd882d6ff795fc01687b34bd0fb4ab2d5787c7823163adc56f87f84e8dd2bca1b43b8bc6956fd596782950b407c905665abf93651ee803cca2eee9f2ee1063b32e7d0af217ed9e19216fe19e4de911cc2c7827ce6e58773a172081e4266d0f4b25f745d1ee7f36e4dd3b753b6257900e3c4dc90f516024422af5153a5f9925dfa9c1dfe8f63a6dd58fcc748a5ba09c24a120a433209c239d2507ad59bbd3633dfdc22210921719dc61d586075347c3ff6da30ffdc2fe785b6ddc97c94e30a32b1272cc54435b81a0468dc83d783f4900470d55855e5063596eecf5f85d51b32f929332b6e816f02aa716221d1080301a6d0005aa2b072a2d8ffa3097f1d2c41036af17852c6b4d2d00facc74ec638670b2ba02cb404411235493e16f3d9b1c3b7360426abaec3c7e01292192635f3e41bef4bd57600167dc9ec7ce7850bfbbf0ee37dd407ea9cbd5994bfb72ea8ca4add472ca5bcf00cfab1e1451bb73fecf8ce612ca289851c4f0dc72a108ce26be44148c09340f6cfd5ed714724ba56837583b721933aab543513434de0d94ed02a48da499c09864a39c4456b9d931f4a7b1c8211af4edf5514f2223f7565ef67f4e32b4f9de3ab3e1a9c1e75b90f6c3f2851db1b8782da291187f48db09e010aa28ffba06dbdc1ad0bf45d579264632520c9339d65da681891ccae4ebb43bf525a928e666e99af3327df65db876951a19a9c87b65694c9d0f2b1cc39b6e55c4accbbe9d381f5f9ccade15692aa63956a403a8634eb7037acccbeb8465bfd269c5cb0c63d676183b0b3f141c94597a13b061e1a924981078301302a514a1e1104a53df6086cb629a91fd66d6252133427a8bcbd686ecbf0adfdc209bb6109ffcd7bcbb11f34f08cf7059844fbd7aec31e14232de664ea2215996ef389a7d8d480cf23b9593366abc05b961aa77292a9268a25a05124d323b7bfbc09670d6fed29a4eedb858a8652bb76fde4e1f43060a912c933398447c274a1e84bfe939c20ba3657ea54a39e2065792cbf17427949cce0cf9a5887ca9584ab78162b911c98b961c5b1102ab016f07407e2c00264534584f65c688b04f50e062095234f0fb41688a9fd05fcdcaf027543a2fca07760eda4ebd0325959461bc29a04b22c4f334ea313d8d423cb2d9e310df899b5b6ce36cb9665e7f8d93bda3377ae35cf1da84c665a1691593aab4c6c7aa669fb876166cd1130037b0fbf0a3f9d7810ea7092c1dfe109f5ede766f04fa40308824e953b40b490d779f22e909f5cc2ecbc80f7649021cbadb1a0fa7622219f2ad14f80c827227ea7da268a654a13a7f213d34f8bfbb644e70fa3151783b74a069128ea4721677c23e376b170297e9b1a130347225d09788d47d7d10f45d6d5588a70b9e32ccfe9e498918f0c1e71771309b1085a25917f5ebea1fbc052f56d4beec0dfeb74fc47286a3dcadf2e41da719b96f4e5bb0268166c50ead08ffd7e5120e633f3a97a2fa30f8244a4a66fde98e0d0950a0992fa23896b4e77f06244ea89449cfeb977b718f2686a2e0f563b9471ec09794f7860366feb7ece035a0acff5a2b051d9ba38b6f2e7373fc5b544d73ac9810fe1dd5bc61a79749b29a5119433d094e02cddcef2a30e527f12307c02ea07cf4ac107047231f18562c0f7834cc71eeb12027cc5ae7e34c2dcc9088d04dfbe840771421b7a5bc3ac91250faf8412b28d395e3e3dbb7ac959d9924afb4d319ac5281ab6c6fa5ea7c2f8644953f0583e3669c283468076d69edd5e6c28fc9ba1cb58c90000ea52e7ea85c39624e0f1793b329d1d1ed72471ac81f2731516251f32fcdaf58237fa43023ca2df1b4b1deef91b8961c0b186564a9bdc344fb1a96d517d629df647ecb5993f2186cb8e32859fb3ce3a218a7259243dc9e9361668b647b0f0afde6aef8e51da7724ae2eca2046f8e5169977972571d2063c979b6bbfa5695092cbd3b43bb9755c253c970a0316a6186eb8e60d53f1d7bf6a6d43bea5df6868170813ae6bf1896b84decbecff60a8bd4231d7de16f63f6f63a7f3fa1faea7572d54b5af67b62c3fa21d1d9850ec7a2e59a8c398e720c639ca4fce73b43c049466c7d0454ac663a8b62d2f24b26e6205ee55a591223914c58efebbf83ce52f07e156e3f798f100a6eadb61b7199adace968976fe1903bd010cb7a1ccc371a5036b441a3527693df298e92f761956c92515cf958739e9de30b0757fbfc02d60f17bea0aaa077b48812f2de51846e167a6bc322a3baa9131b966c1ddd00ae7a44d0503530daf0dd397c1663c141d82567dbdf038774bd4a34818daca962a159b480f30ccc033e336d4eeb19728ecf225c0278d6bc01e2185bef20677c4661264aca401e25689c429475fa3c645004e7129cb0c013cce201636cc2b10a8843620e5e8d24e498ef4d8b7b1080da844c60f7363fe31f4179d2c4d18f14a963292ecc6eac076ebbfa819a894a2d1cc312f320fe36275a955933d17c4e5f4a6c2ddf2bfe904901753ecc097468f9ba015351362a68558ecf88f762c8a00084d6b45be66f4c86daa71a29d0f6aa1d4adf6e04b36694febefc0a5321524369af4128a5179a00500479bbf07eacc59add65e72284e0623acf058a90fd360a07e9d79a4f6fb52ce234651977bf16dabdd2ecf1202af14a246f10a62474bbbe9937ba45a5ef384ba263d50ee6675ba8197f618f20940e53f1714dbe0c9833e28647853a23e3c5e4e4343113391bf9f6e45ee7bf9c46942fd19db7fbacfb005d90d5da626f0ce770b4e5c84a6920e17accda85fec07d496e34db4c5cfaab70dd2b6084080167ff074e783d248f74ae927ce1e75d22b5326624d3ddf43ae1900dee171bff3312503bfcb465a1c55fd95831b20c95684726ee6cadde8a9b7bc7d3d70b59b8f23fb7a1d9498550d4f2fddf3ad7009097cacfbc2276744b6c80d1e6b00b1b5c6e251b2df13e320779bfea4872e22d71b6b11c18ede80c3f78a97f9e73884b9d27faa21f020e42e1e049cb8da5621cce79db6d9281e9651e7ae93bd5495f6a61beb1d90e2386306876d6cdbf5670d39e226460ee8553ea9affced257ed4328a5b797d150def2805cafe633652011b902037bc4c775192adf2cbd94183337c81f5c87d972dfca6d07c4675598f6aeb1396ec92ab516772c35203edb5c5b39e326b8af0cd78abbe8dfbd7e2876a457cd0978b017febfa3fe96f3f22a0f11e9a300d4d12260a7080d9a325c70a3439240072c08dfcd63e4b6c8ef730a1a3b7bba9b8e87818b04d4b079faee1f7fdc8db075e29b39c4e498d0f640952a617bc328030905bbbd6db5f722335eea0d65d3adaf0ac576b33fc99b80be328d082c35fea1eab72a62aa4009bd2e4ed58ffa0435ce2aa21c4baaed2a359fd645b6e247073dfe8919e1b9db4f8255afb7270a1117c48ba55e2e9338ddd4a07e1083982e49c4dd0b62879c99b047c03157ea5b8060597736635d4ead00c7c4b594c8f091187879d7858bf369549d2f18e9fe88ed6a2642bec54f7bada529fb931c4190ca0563a376326cfc9b32376106a24ccee160ec6aeab7e33757ed603534455b71f89e7414d4525c6c0cc4b3a1a185b5304bd35314487a138a7a45eed8f9378864b6598d8d88bcde6fc90f27937cbd6156672e344855994b1a1809139f81e3b7d825851b67ee39722104d6d84416c1f3b07eb55c82e349ceecbeab0fb67c136ffb9fad75ea29fbdc0eeb70e37e308192e6c235d0e38304ca4b8718680f581ab18d5eb462ecb98b564d0baa3f9681aa7c243e7fd0d53d6ffea20715afddc77c60ee5145d948a6ed7c0838133074f4f9a54b04bb6d9ae55b8cb2f35df593592cb371efa7fa348384bcfa5b3d9add16b9142cc45a4434d84190fccc3fc84c5c2fc8dfa8a9d06de3a5c782fb5c50eeb37ed1fb4e606ad0bdb72ca7b5641d530237053ec7dd1a1f000c32611ee731aef627d7c8cb37726aa8cbbc8f4b5ae425df91a4b29e8302daae93d2062a54715301ce90612cc9dd05eb7057fa216ec957940b26e6548b6710abf49eb958d14573449bafb5c8288d3b76a3423f3bf459eacb3e9e8e3f0c5549651abbdcc8aebdc8122a0422991b30f4bb4a6f11dc372eff0c80ac3e93954a1a90303799fb9498dbb06ceacb113e153c2afefa2655c1145b6a58827d47faee5f020b00ed354f0c08ec24924df65af0376a7eeda358546c185c8335771c7b66540da4b78041a02a323dd6a3298249f11a484c9c78dc1eb55621a2a7c7487e3dfb535145fbcfa69f6eba02ec50ed1ce2347dbfb9538e66ee3d9b2e3530d5777a3c8b84e74a6913a479cf48cbed8d4c573f1e48eb919d63b7fc0e52e1d0e0dec4d22e7264a0abc4e164de88b436d4acd354cc6f0212b09ddecc3a8df61a81b3ffeb4753417954020c3e986f3d7490ef7b27ecba81e0045ab999ce11de93a3bcb1e8e41562c4bdca69a7d9575caca8f58570e51498277b3d1b0703f77811f133b0aa70e28181bf32f277764a471263470c723aa9c079804aff448fd1005e7f210ffd84a1602f63371550c6a2b298fd1b17ffe23887a02ac29107780e23a2bee13dbfb4f52e9635356c61daf11d94c3fa9c308bfb72096524259dbfe62d35b45513ec0861b500efa05b04e6095a434e28818dd73a7db3ea6101d8b877a4c0a8279170dbfcf860b6662f2ddb93006ad591e408d1bffa88f34e2c2b24a83bc69f19fb5c0161d62192db2839316068e682e6254d4a5006703dcd9589c4b42538685e033c17c8d48765d452ad03cc0b7b816cbabd725f7faa7d753fefbd3347c045984580c09a09de681e6b0bba83ef9c59a0563a91535d73aa2c6665f6d597706abc1e9bcaadaa8d8385a9da93c2cc3e0bf51b471d92733d829cf59ce88657b8b824ec6ed063858bcf2b11eb7e0673c42622090fbc71487f482b9bb3850a3f14f8da49a7bfa515325ce93d0bcfddbfcce66eb24afe539ea2f4e954050d3081467e30f1357252292eeaaf39682b62b2af33ce46f107795bbc3f2bee4594ce3b6b9eeccb544ab524c55184e0d7ebb604966e5260e6efb1a4a6a6f5b08a47b883deb2073137e1652bffdc63dd501a8d6eb467dba61902643982663634ddf166aa74ae0f4f33c87db4bb519f2fc5a089bf67cc9c12d1ecac036056d6790bbb196fc5c0ea74e7258c7f0931f4c132da6dae9f4475cbded3a1d384ff43d4656a434b0c39c48cb4c7b554de5ae879883b9225525a40aeb6eec57df05276cb18469e7fb9ad678788c11b3b6a6c55f30e2005b5386bbb188fe4283c3486c23c3554c64b7fdbe49c55b737c638bac4cb027d325415ea08888c46cf22560e0f162e96320cc9df14b25df1334b1ec887e6e39b313671bf2a101b978db07e2d5b932213835f946cb5acf26545915ffa811917fc31e1414f3491e7529443f6703287ab3317bfd1f1857e4e80339aca955eb7b52c5e0aff7b84c2d9790bac99caa0c730dc554251ec91d7ef238f2e82c138c4662cd52331f8b5ed1f79f540917db1997505b54854f55705414529e6c0f748b0d8904e5ab412aa90cb2ade9b33c65d9ca0a2346a3f757202b858d85dcc81e48649d445ec111e20db5a4a273d2f4d707427334764720f2abfeebecb84ed15c9ebcb0e6b6ac77e375355e7af24460d493c8bb0c307df8f05f7c3dd6ff69a845b8296596d4d9f31936006905ac9ebbaaae7fa250e212e245f2b7980c2e9431afe5eaa98d720391b535821976b926badb58074dad60718edcff3e7ea0c3399735765f9ef011a4434b20e69b4391b3197ff7365f27dbaf4b87e19d480757e8a396068bf19656b4cc611ba4ee1cf8eb483fef7697c1bceb0ea87ce03c4227a392843d9c69c2dada310a55238ca78a9baa7376e213e17a254e1ccb00957bc4605e0321a1f1b181be8532fe0244e9e90989ff1cfed12b33c3b3ee964e9479c67229b5b229a3adcc17633137cf27021ffbb993aa6b82e4f12e30abc44f48093c85dabe524dd3804cf8316b90b5f236bb5699b7bc9b17583ca79229d071decf7ea938b92172124ec76633eed4be1c57868cb540112ea14ef0370ec693980eb3f91849593e272cf6c367a1868dc1edc3b6371e191bc343b3b7edd4a30743a94a012c0bcdd18b30aabd9eca3311886b500f413c5dff33518e86537ff46672cd8a861252a55af3a21c584eeef2103f337469c3a4ae8294585846b66c0b2197c91165d1e902c8eba8cc5c57d88a6fa053ff64b92328a8d235aefaf25c38591de67fbce81e56b3a813b164da5908906c448aa47b9946fa7de2e820eb0071c15fd8ad12b2385ae1f472eb880b0ec4be60d6ad34572a25bd4e877fe42ed4010d4aa456ccdfc9cc2834123f255d049cf0a47bd0e0cc5be60a02dc38ac2f9f11d69947adeefa22b7c419db634d8b7655c2581f2a6f697d1680cd9900f6c939ff11669b5f174bcfb50e1e1c6c57e6085ce8a1ce441a2d65ff61c9e85b245181f9bd412185835c17be50c3cb37f27d8ce330f5feeb4f785cef464a26c862848bf074e8e15b1f0b5c3778a4531353affb6e617caadbecd97c4d880297dd27ab0a3e2cc51761f853dc0fe1ae2798c60ac71b4bae067fc445736fd9da189df5f8a86c56d0a349d3c12deeb3d334c58a5654ef79520c518cfd0cc5466b02592beb37c5b484a4dcf385f62891ab0872334327f914c31fe66b50e451b39080da1f12fd92b66f4cc0f621995f11f204a200f3a9308b9d7eba509890483b3d22e6d847e6027769dc4e2b46df028c4e4b66ca7ca65c4fc73b9875a6f0f360cf6581aa872b3367f7d29039166a6728f7d6b17e5d31467c1f30e523cf6d7e7e92465a5b8bc8ecdd855ab8b34613f1b31ab3092949338a16f9fce985044530ea522944999117c58d97b4a25bbd7239941cb392dcaddc6d86da632c34e6c1e26a370412c1a07b6267b02f132a91bc8ae93648be0c9b952f4db8265b8c7a7d12d4d5a0a71b5681414b7edd308bf2130ef33fc59444254ebc1123d2dd5b29cfd85ad76221d7bfc7a887f81a53d7abade79b654aa55cea1df415edef932b4910502e47fde5af90530b0e2de0f708b8f13b97fa65f9d1443fa820a5dfded0f90d899cac4712ef0b03c3ab23e312478433e8888e87bc344d21bfcf058d2ebfc6174164da39904df2ea4714c3b14e5c8fa17d4bf094415bce5c206512e740b4e13d49a42a0a77af90cf456c0b49ed4cfd4adccba8dcabdec2235a6f036ee3f843f4d21ef229a28e89ee47b340910629bfa5af576a708f51c9c1dfef64300bdf3d4b078ebb3db06c5cc1904186215fc6273637642b5257e92cc35ec29074135780bd292954700507e70a3ebc055d890481d20d9f8743caa4a1306411f8ac87b2e48314ff1fe7f4d80586a1e95c8152bc24a41c89635535e1426187bc8c0e35258f4f119b88872e7302bf39e174acd5bf74a4c30b04f85bae6f511ab18a0a730abbcb08b41f2797f2b94eca86107858ae91d3075ac01e53f7ebeb6dea51e9f397fc3536aceec5b7eb884f5ae229fcaac8b69042963e0760449ecff2ff2c4d8108295ab9ea9a0b90aaebbc464d28dc87077bf18d144b6cd680d5c9d7f017a27e76e883c06a2cdbe17c849bbe4eecc65c78a7453e56eae50b2ada2d40c8087dba16c5b6a7ab4882c26a81e41931037d49dcb6f841c50636597d23c498bd04220901641fcd1bdc9a456be2b53c2b6593c0ef61c83c8fd5c18e22acb338c6633847592e31d9964884d2960ec5d1253736a6858132f9327e8b6a6ca65238a3b1cca33e7c4cf7aa3d87f8f5f4161e469aceb83cc6b3ee53fcad84cdc7913df84ce39a08591217a5c0d6fa0091eec1e18c56ffeb2ef08d4e1694bbd9c25375895214930f2198252372a5598b051223c1dd6dc288c60d0389b7c9a331b01ac354c1210ca1081645aeabe6d50bef7e3a5de781b4915f145ea7ceb37584a3299d99ab2f6e44173936bd15b304c75a38cf62e1ad52e0afa26c1c406da985171e00233702be1dfbc57ec1183244fab920cc60f66e4789554790f866542425e76970942d5e9d9c29c3fe1f1087a9cc9db87b2e361da67b7c929caeaaac793ff3f7bb603d95f8b08efd8630d7885620926b11a30dae336938efd277f41af5884fb480ae81bb036bfa9d182fab2489a108e058ff6a10ef78ed688694579065e6c245fa02c632f936507a8ae303d1c4ae6bcfe62d4c9aeb55d9ca97efa847d6177885773a430ec5884c5ee68781d9b7ab7f1de50f6ff340894e4fa401a63c5cffecd1d43a71022432efeb3ad4874a86fd5ac5312c783e07e67842dc9423290f93f02e4c5f3cc6ffd51467907cefce47617b7cc3790ba2b34044e1aac47cc6d7a03a968dc65cc4c17c9c77b5dc86d1fae4eebd61c6601c59bab6f00005f0b2b6564459c878a3d1d7399da3db4b2ee2936efd50d9c6399db8199306ce86ac8ba36296ac540d4241b9eba823923aa2e08733f6b80de8cf11fd7c6e8c1a20d9ba1180873cba2fb9d500b857c81a09758cb87da629cafffb14a2a0c53e54e91e6fa8f573cee33a1abd1dfadbf6626050b2ffdacc9c91ec88157159dbb7ab3dbf13103b7f3f7f680b2e8212d956984fc31e3a47c239905775166b9a79892452f5b2bea8c6f31b6cce0a211b25b5378565652146e39eeebe4c95366dd25c9c4566b36f9ee5a27c50067f038d7ff4ee0d161d962c3bcb6b3c5f670882009873110861a28e6fea0b64939b66bed18cb23fb2f2cd62ff4d94f08cda920eaf1ad3e44ba2d41d4c27b79a3bbf5ced128e6622b965bd62de8fc235643aaec35b72921bf26928501588b80067f4cc943b07fcd353c0c63fb2cb1589fefb7008c8402dadbd47937406260d56ff31ec5631993be87ccd14ba76f085b1186184ce733b1a4fb14ccdfdf929bc9341a669a927757e73fcdfa02651691f75eb5325973f65026025913550a74aaef9035f1a47803d1eb629464f224097d73c1785584acc6da2e2e9f703c8db5c64142e95acc2e0c719410f7d1e9a12d566762908aefee4237640589252583b422126ec076680b73a25d68a7d28d9a3ac303b0ec1d1167d3cd15cc12c0c93e52e843d8360044be3b20a9a552662cc2af18d03a5b2ccd6d417804172cd1b5fc5ed13e9815703a0f09541fe468cb2cf247e265d05c465a3cb4bec1797b45784a323f146e9f3f6892790c1b753592a5e6ed06f1184081bd377d59e4d48f7f299a09e9e58f10e856cd36b0869184531153626c235acc555963a47c37f9df9bd9d91c3ad3218f65d43a490507c318caf09490033797c6a0c3bdf5552571d0340771cda6b061d5620f457e81908e075ad703578d69d4450fe6a23f122e8b383c3d2b436b80592d3cd9a0766ccbd28862f941143578c54551a928b438f22f80e1b4b020b20b8f5fb925319273e335c76d69dd272761e6cfdd3a3b29f863dd0c7ee84fff5741539e235ac19105cbd529b283cfd669b6d8eec34fb4b24c4bc7ffc2eb6d9e5b4e5f2f8c91605671700bb8e3d01e6fbd14cd56febfcfcdccf52aaed4ca09884bda1247bd024af5eef167bc4a63dbc9f678973e6e41e313c5845355b3a8a1c42713c8b8f1bf81bff91297e5cd7171d3fc45f4f2a1b15309ea21c917066073240608c8e63506d6e9f1933a74753c9db4080a30d73929a745c7d7d751b9f63b28eb4ba3c3e4ff2ace0c5ed00fdfbffbf1a3d88e7e60414245a730ab2b9e3385c9d62f55bb38e26a72190e481704b353b61b761164f63b3d7502c93b9f1a53e2fcbb739892f5ed26f0fef09ea0d3d5484952175c4b415140cbf05ecce74031892d6b9f4346401aaf5b4bb01c12fea7624d82689ffbd92563e685bdb16a35729cb81038d03087f73d2e7fedfe8573298bfdacdf9c244094aeecf20e554c44600c3796cc46f535c9f7c69d8dc340c8883163d95b2e2ca29ff417b4fcc9bf762e0dea789d9f507da53c47cc6d8daa047fb2662930cf19e3299b99c666ed959b29d258d334635aefb4b20076b59194d2138d0e207a6a0f4d05ba310ba824fa19efd5799d6bf7d0da13fdf34a5ad1cd96b86ecf767d30828997faea2300d9a501170866a12489c257f6646ac3dd7ec8c895258c7af0f50a97d18cc18e266f3b7a4f4c66e672d821f04717f403bf846bd925fb4caa126099ef9da7e31489fd5dbf53f71da96967aa4a93572abd2c0c9f55b94a1a8e718a3c78633d733ab0183d2b1c5b2126403c19373333aa2b8d6c0e3da670120e22f649dfcf0d12f09cbf434f2dd6b8fc3c7d60eea97056b94c4718f9abac7ef8ec8facb1ab5de1e7e67838cf676fd49801cd91e46e7be84912cc2345b634a0f9a97ea13ed97ec8776ebe350f55525dadf113561af27d6c71b2de2dd2e2bc97349c7d532d705dd5039b03ab6c17d6fb29b0e7e88e08208bf9f949f672595ca386b5b322c1295fa33f7193b6f4755bffa19e8aeef99c17b04b9edb4a99e01f6ae3072de94eb9b87f950446b215b12ff02260293ea0df3c37f10fccd84762153857a2911f9e41fdc3349857d669d4d4ea02132e04f7b7dab16fb2ee4b8a75b2ae969b21238a785181b291692e83166cb755add1ebb297d07203f90a5d2403f87e5a34b2157e3cad5ae6c995091921cb781f662796a5cdfba0faf3f3ca77b69f4c7f94ab8928368f82a6d5371872136752974d4989412719c5949653a3b22ce805481906346e84030cb782f990fb54f75a5deefa3adf48ba06009a1b769e201717b941ad7670c2f08e87cfec43e3f5d28866aafad630b11a9bf72218552afb0f12a0604e72952e36de13d41da4747847debfaf614c6cd65903ba60b120777fb6fbf34d340109afae24641a8e305d4f768acff5259d6db8c755678a7b5d96e11a5b75919ceb21016bb4131eceb96948600b73ec545858ae64f77eabd0182711fa7a2681c5b8a017c2964993d0f0151830c305701469fb3896a3bbc4447d4670963a1b16ee27cd0905f5b37a8abd18059035ffd1fc19cbcc12a500bd692a990ee03438caac38def6ab3f0485cfab09f496876bc3c806ead45545dc9f54dc1fc3d0313cea78a96c1c1c9393e8739eb052a92be420896bc8b62c283f09de4dcb168a49f55da77bb568daa2d6b27188185ff03de40aba611d5d8d63e7bc62cfb543ea46031f0d7b6c82f0f5826b66ef4f7b8933c5e5c1625297f2c625338e2485a0655d9a258495e90a8f0fbdcc4c67f5252ab57efa68d416c99b3ed9118ffc73a416359b46b5414123175287d0c7cfe056a0e2ae45b5c127c13cfc0a3b8099e24ab98ae8991decfec6ed5644be4842f439c848db46d3c4521ca2e444e835945f4ef5063d9a3e17faafbb76f7011794191457faf3b7e680ab8856f7adadaa4fed99d84f9e7fe6d0eb2bffc13918bb565be637f7a4d5b62d6d0aa1ae6e13ff2be54c41e714c3e36e14cad1be28c837c5be95fe470b50e3b4ccff06727455b7c46d7fb9c4fdcbf77a4500a9405c1cbeb4a69a8389f0975388929b6a388583ff2c197fc2bff86ac6abe3886981fe3a816b7fbf047cb5aeb8277fc77e7ed3e7959299d011a0effe4738df8d800c4453a66f832b13a83c0ad3c7fd0ca4400ff77fc795de4efee79db7a40cf838442ae36a2d63b5dec686844dd36d93e36dad938ab97c063ea83df47280ad9780f35d847987f25b9baf8588ac694649a6963ceba73082a1c370d51a68b64a83d1c6ac9d3aec62b995cdca58fdf03f520b0e3bb4ec05fb96beafba3fa59e3210fe6c94c92840eec43a30b006cab525751122d8444ff493e0e0670aeb7cd3d64ccf7a21b87c1ed79a07af7f7316403ffe154baf2e521bc5cff7c031820078bc8e99b1529a47af4f71f2d6def9fc175f21032a8f8c482a1751bff25c5f70e657bffb8f3598a2562ea26c0cad09d8e388ae8fd7157d23787f7f324f4851d6963862219fe802b4bea172a1f0959c990cd8fa34bc1830bc90fe519f4510be1544bb23dea7a0879097e3b7a97d8c0905536929045d3edcfefccef967a0a348bb6403232ed8b25a8adfa478c5f2bcb963ed572e685abfaff829e7ce3df0d6ac3cc1c2712371d7434fc6e2055ae1060d1abb3e68c2c402138f9303d72d8ec67b3898472483f7751486bb38283f20fd322020a2e2e6605f290fd3490bec743d4ed8d2a55b2ace57d7c47b5d88f71c641df21f04db0e6b62d86ea5e7349a0f4a893c19c5e8e9a6c302121ce40cd44042d35f419df23bf9743f5bcadf0dc7afed6b750aa567b9a9f79c4b58408332f5365f99b33467533da102267b5fe1c8d69f7873a9436fd7279ba9e8936c4fe1cafd6a7c1518121c16d100ab78da9e910be02c98d4ebcc51f3161fdfe7988d37469f180d5824408b2f6c4232ebaad8a762b148f6929f3f647d1b6ea5d26279687b802f3976173eefed0220dc6f47dde6035924cea3a65762f8c6d1dc6882fd4fd7aa387146de3b45561f26b69137a5b2a60ac4a30af3cbf518804f56f4ae21632dbf08c8bcb66b85678f7f32b407fe8d7afd447fdf76b2ef80cd43dcb573dc76d47226b183a9c8824c47f3a33395fb535f4f343af3c4bf913ee347c586284e0039e4fc3deda566100ec99accd76336dd104689b96b789b4d3865ddae1be85f23685b84a156750616c48add191627b18682ba21b0acc5da308979e1c011196258e204cb4cf38a5b8f3ef91f6c1f88e4e937633ad533f2d99c5802ba7b18805b7ad564da27d5094ff2b84ba371213a267210e59646e3794d7a16c13f54054abb641f633edec4288643af9f9ef2d2e2cc118b19d8a8cbb002f424a42e04362dec9ea8f5988fd7fbdda785d7e3da1a000eb1fb947280537be8a72b305f10cbe3e245c4a0af9647a6736ed292647002cbb42ad59b72528e7764af2403a773bb76eebcc5b92cc1b6647bc90b3b59594ee74d73356d93832f33ceb4e6bbbd4bc21272bfb0413aaa1b6ce5f56cf7a3bf9d311a810564378eac67a9eecff0f3e633283cf6cdf87de769f0ae4511a36593e3997f203bc433533ee5b659a1b092cc4a72420a80579a8120cda9254a58e975734bef67e4d9735ffb8ebc2b7654d4bdd1fbabbf50d107398f61108d5f50d7fd37f73240733b1b8afcc8c767954ededed8c6789ca62b2d8534ff8c5dec91dfee14d08ecbee658a24cdd034a02b5de08ed719ac5bdd9c88436e4f013c027de99e0a4ec2d5d63c1b23622db8383e2917235d9355cefa96e788b592cec3a685a32cb8e1bad1c91455d3e14a1eb7f961aeb040752dc822a7043e9522c5757c26a873bcc435e20ef9a8d591f598771dc760834768734c35599df616e01a30a869dfa295c1bda39a32485f84e5de994c51489ab92624c28d18d6d3c4f71060e6ca6be064a11c2c3b47c43b03f68416adf319006c037e9e13eddda2ec1a2adffe102272f10ba69707703cf66b0a1eac23ad4f6aa18eb4ade5ba46c77f4f87e2b57d28a77446756bdde57bfe99fc22ce0a232bd43586e9d6bbbd8808eee607beba4777112962b902a6213df7088d4ffd3df470a6a12d4b988cf7d3206e90e5b7293790e54c7cb08c8b770fd7e2fa316bbf93ec8908057d92f2298dd39b76f3c2b91993a9c30c72313d1e35dd3bb448a8b743fef0aef84ecd397915990c8e8ccfb879067d171470b3fb7558f3a5da549148a3ed948666b200c0a9548dde42cf051698987175630294b1f1439f24d4001dcb8c9d2dba3350f104d664339aeea0777d973a822f976f82fe59b5ea8a833fd908c0aeeeca55338a3e4958dde8d18cb01d5575b313db1dcc9e5356dda810d51b53dbf1c63ff49781b6b67911fa4fd6fd7815cb633e94cb9f3f2a063a6492288a8db20ed5c7bd5af43283aa3e1b0787c076a62bd47d390aedc9e64ec713879f3e3a8d91e64c03037dafa6f3570a635e6033d85855c3f36e77d56570b66e6142360b0864e45a0deacbc904fd93ae4750459945e30bfa9baecf5dde7e5ff3f72608f1ddc8a50201b465a8403ea4b3231bea7b2e8d987cbb5fb1d4ccabf7e1f2709c3cd00d309b892e21d0a2413d1068b212868e68d76aa0292422cbcec08b23814ff50ac5d144a13314eaab63907f0884521fbf72721ee654b96f0b26be737c67e865444199044367d2d8e30fb43096756627ef11c6984a2eef982273f7d3873bb73f5349d814e530f864cce73a7ce56f54711e987f23e032d1b411beb7254375210fd5c02a11bb273db8760d595869424bbe0b2214099810b7a787546a433457061733358814a2bda8072dbf8ceb17bb515e23c6613f3b19687060b7349ee848aeca95934d416fbb5a61b33dfae84410801b8145c8bf5cff51aacd282a5f7e02ed88370bdb68e7cb5aa7c735875d0b46b03d9a60b6287ea506f0c5c2e3ba0cd882b3fbcbb68a2f6a9621f3385b0591fd89b1fee6b07c3c481e78ba229b15a90bb6098c108ace7a73529eb357234ce7be5d35a9ce30d2605e05a662c0f9cf7a37a647e6363b20279c981588f4c83289763130c1cbbbdb3b641a68f78d86a754b2ee9ddad8397671698f974634a2cbee1ac32a0a7f40e57b91372f2e219f6a353a9efba0a6f590e426885774ee7e97bf60e54372506e688b7fe221c43ea5fff97181b3a57abd2aa5b3a1ebee81b94d7a6fabbc7dfb107d0c0b321182b2708de89aa6ead05cfbb6b382c6f4f56160b877aad1496135ec38c8793461afa7bce8a9bda0cf75adde3bef226338476732da27c5dfb045470b3edc66a6a996b3913c2080a8f28bb757ed636e1abcc75cfea2b5c54e3b401509e9740398f2486b51e209d34ce7a4dbcc9554fe66acca30069949d394c7d4fa8a6336604bb62e5351ef1c53afec8bc8b804f11d71a69e8f727390f3a6a22be300b1b49846847a03a75047b0fa3f26ea191802b290fa11830901132e0cf1a6d46bec0d400dc31ad62374ac9786b21de3a639590a6446b1de2bfc7fb33ceeffa39924b2e7f7c4f7b111c5b3439dfa0ef8f5eabb1cebf62947b530ade05580ba60bbd53df928925917419e45df94b79ff57f61e57435a54b226057a83f20d09b987cfb7c09eaeeab7c4e53587dd60ad3e7a0d41145a3494da0af8658ade532ccea562483cb02a7594f1fe16b29e493c18f55724241446ee2168c4940ec9e542970ba62bab0bd77cdf36fecfd666514f325b1409b1548ade31a5c8239c6830616c58c3cb046b2afb594466835d816bd7220c354d07431c1730bd800e0cd5e62105232cc55d33d9c63be604cbafdfcd890468484af3ad8f32906ec6bc99e098f986aed3ad38ba1b61ad9f0ffcdddd645105a5629d31f85d69dde5286104b64b39cd3c7f62be5431368cc89b679c2431812b8bcdb2bda7867b3ddd10ac63901e3888ab661811effc68f48bcc6869a490b5868a638b21f82cdcb3b9f7bb0cc895582f550681bec7181ce2f3aad231e255b729fc704ae688401b2e2e79f6cb3c221669b0d3e1113263c347abd0b2bf3672e9d35986dce6e87d3722bf4ed0c9d163604bd1d27fdda3a852ec608312c84a49c90ed6c2b6dec3338b3b6f703711fa6af5e1734cceea3413ee25c8fd98a3c7150710f22756495f250bdacdc3d5084f56e20b4bbc9cd9ae974f71a1f198c1f3727b9114dc182887c5ff42bcabfaa33c22865f22240f1c9a474f95de1f881d24b1d5156fbf58376d4fe666dac9325b7439cbd76e808420f46fa884e016dcc0fa43dbc1ee4a9af136e7594928f930d81d5d6128506a656fea51c3551e4a6d89c55b71abbe1f3ebabc27d38893bc6fe7d599bc0226fbd28d1dfb5c48e03351e9abb848e3a03db39ba811dbb87f3e2d26e1afe6879e88ae1de26d42fd6d482f6f825449950191fd8bf5073390d1cfc4278406fb1940d8be45cec5f24be26844b3241ffda1b08a610280b675729fa16b0a85d964fc201ee38c340704f0998c95cb4e7282b9a4b0408806837205256a33eb7a1651553eab107d3c1e05cbb5d3c89e89f4bca11e5d2b2e40a41a0b854e9eb7a1c4b89c1bc14dd63a7a1d8c3a92ce2416d8dee19b4a10a9c05db475ffbbbcf9cae880c7f3e4c8bbe3ee0a3e6984d04dc596afd676de974b1cab360682e86dc8fa1add4b200c9edbd427e3ce651da090c4c03c5196245788bb66a73384a43518caed8d172100559b782a11b7282d0ff225861c02b86e92844dad4268c17b6edf2a6e25162aadbf8bf9905573e4ebe92c75753324f8ad524e799517c67c05c17976f6e677fda54c5c8bcf25b1abb627f9ced81ec2130a579ccea595921540d061c71a76d8a8d079a1f69ba3b55bb66efdc2299ef17fc02015bf55336f8b4983f11a0e20da29ce7fcb53aef9dae9aaef4ed25b54c4a54ec3fb2a561d70a659dbdd8319d39bc5cc79faeef9ed5e7a48ebae26d5af45dc202c8c35ca680e957935f7147525437569376accfe67306086cc394efb86c8d1d6134f7a9f296b197155ada24c856e909f78db30315d807e3cbfbb113011d40b81190f1efd7f7c3ae43ea24e807bfae1885c0eaa883ce6398c27a2869f2cac6fa182d7076eb5fc56240a38c903eb6904b62b7337575e284cafc4cf012e4e56004de260e438bf8c6bf58e23ab20a36cca00349b6b9a59ac3cefd7e91f511d52f1224046f4b03b550e346e28c66f17a9fe83f16e18f85f3c8651e3146a1c81f7a74a75de7d65dc11ec35dcaae1443d0ab2a674eda063d473cda485745ddd9fdd9b4c830fe65580f825cb9967c29eccf513d84ec02a91ba5b2203083e3c3f08f5adc685cb276efbe50b2e834a8604e556752ecbccc493d719aa79baa57bfcf629a23c1496a0a4b76d521c0da459701336bada6e805a495955cb1a5c6f22d697bcf32ed463f83faaac66ec30a569d10466036a026cd568785b1bd314dd5d34a9b5b92ea613482c3aa13dde8a4b5c27047ee35aad63b2657f8f950f9e2bc2a2e0c29754f2d9681e31d94d1fa1127c64a9c15e8ad4fe1a6e9b8e71aefb781adfa4fd4fcd0f5ef647e22af2de94fbd76cdba01b9106d29980b556fdfc352527b0479ddaa9655006b6e4d24b0dcb3dca4023681e8eac86a455f969efa2b1451360ed0afdcaee19963685eb3e8c0afbe9e836df3d20df696c96d231d687a2ae195fda8f1966526fac76e121ff7e74a14586168396323345c0213b3183b65206eb444b6b23dcadad5c30d20d0c16d29dfb6365b21e995c02d84b25393fbed2978d5c802fa441b327b75ce4168346470798ea787cb0b4270667f596b480751658cbb25a732e403c8be5a3dfe466b379d7c5ac533840fbe3eef2618ee1b92ba76ab381220f1557479c69257e3b7bb3183ad280e841c2c553adb9200003f7acabdfaf6537ecb6565972c684b351f9d27c88fb3d63272be1bfa76097972945342ddfcf9414f05d3fa5bafaaa637f0b31b1f35c5400d879291aa6955d177bc9599e7a2bbc8149d35850ba16df6f1e58d78f4a0a19f8091f6324067d4af7274a46b301941d776d984362752b56993ef598403b96f473a164505f0a3192a32e868a7bdbb5669ef54970917e096ed1ccde2e3d9889a003e1ac45ac396fc029a56023d2895df8ccbddde0b42a3a64cbe61c99d48057eac2acde575ce564cca5eedf7bccf6c884c3097c60adab3fa26e58db83cd00c82615cd53af36ee61e2efe168543314d3edc222171950a01ce07a11bb33f6fa4c0d60952c51efd036e51acdf7b3502dcb427475d356b9dba7960094d446335e6cefa8166c884ef7e4a30ded188bb0c065153659fd34e6b5c09b0fe936f970625ec8232e0d47f9c578155ca6893573a3d8b6d50fdca4077ee5167b058e04c3e679542d663c0be972e827a9c59686ca0fac2b416176f8cade6f799d8b964e86082e5752d174a008e5778ca2eb59178e02ab2e91e42eaa074019e69f502f409b979426d6e7ffc5ef179b6f473757198b7c72892b15ea9775909427e9c2dec8b98dc31e7d99bde49cd8303d93907edab737c4daa2039e8b041a72721934a68a0dedd6f49573a5120c719469dfdbc022d5931d24b2b7fd2854668705bcbed6fb8621748171834f7f9e031be0e718c196da1499ccd995b3a9ce585dead22b8367f5981b0810a806f9d6d89c6c802fd0ab8dd6f9e13a179f6c7f673ddce6db2a0833f5a55b3fa9f59085a0280478d78559d6787c80fb371365bef41db64bdd4d767dd4eab494c6b3727c1811801aed0a07241e1a2b68083ab467ef23c634f762057c1fda39d2276ab4a7a1c73dbac3ce570a4afd7b7673ffe4241c085febd9ad3989086bd19e58a67b5026830e50731b41dd03247952e3a27d5f0b40edafb005c4969dee6e71971049b9a693c8f867c82f04d8ffdb9bc44e2f8bd137053d7680194d31fa54aab5853efe659f62b2c888c39ef3d9dc0bba10bfed7f870d5fb5bbba5b4f69c079c3228d7fbea1ac0ce78ce911dbeea0c927a5d65cbd5e90e544a0627df15f8180950f615817bf97f13fa002dc011b2dd7613eb5c6652a0ccd9693614f0661a6b1308abfe2d2701e80d866a30d5ba577a4c2119853e47bb9506bb8dba0894bdb16bd62a812b55a57c8855f0c983484cdab74c22294a5cdb329326c21a350146b14a88ea4acad7816204e23764daaeaea464cdf8464e8e2b699ed872684fec82a76d2322d4bf4597776eb5608f1418cdc6eefdc03bbb901786d5ed9a414d9c9a27919acf56ab2b7c0839ac402745bc781ce7888badd2edaeaf81594626152343621bd2cb3ec1d2133789d622437975a32ede78bc9bd6192dd886c2372dc17ff757ce2e0455ac80fb5ca15d137b036cca799f96b609d43b6a5147ae406ed0dc4c3d1727a59db488e9c08e837b148441e3e77c7205673f4fc5d7c5376c91e6e0cf7b0ef6f5db8ac5e58b36a927d42468ac0bcd421aede1cfb1be16b95a8b79ee1c8d82104646dfb8f62f469f463a251f5df528412c1505e67c305f1fd701b4b76d15376ec5650efe1790dd97d08fb46d1215dc096727dba40e9c06a59ad8d80f6699a9a1e0b7fffbd7ecc74a88482f200d280f2f5be6013dccdcf31a15bd665538e850fec417a3a8e087732551f889df0923158e5ead30182b83bc090f84d7f2428587d7409cc99a3e0a4711074bdceac14293ee0eac5094ee3331745cb4d4456979354816bb753c1be0e978361648350f35a9d0473e90dc7d198682c89fbadc58b048d8bd043e6d0ee4ef31f8253d75f93e59e23d2bcbbb44c68234822388275aef028df5dbbc4b6eb794eb157231c27714d2a68b4cfd79451b1426ac3af6950b7ed4f4c3de418470798f9f61f60e99273430c0bf39e73f8fb4a8d2e427bd47f3fb1ef9526875785ee8bfa246b23a883b8659388dd8bcff60c0a5f05c2713247c0693d0a8bf42c60d2d52ef50c7dbaaae5398df523fe948bbb3da6bd6587e58c837fa6dcd1f7985e15c105dde2d5fbd8cec3c4540412bc3f54704ec4e4d92b323baf9ab747cfd156b80130a109d18a8e8cc7964846086a433852242cec4573778ed5b850bacb7fd45d19fa81524e0d37732934bab7aa01a883865de97becb5d459fe82db0c4dd35d3344ec1cfcca790265dd720b926e4b50abb7da0de65c6c83516aaf01823ce9d12eea4ab5956270d4b1ada2dcfaf779a28df657e00261c8a2a0802facd3b71ca808f822cf64b2e395dc7a36eb25c88cc6bb914664a9779e34547158272921c2aaadea2ea67b5c003b60b188127ceee102fee74d213f40c7c612eec6500800fba0c9cd3921c9346aca3540df2b62b194e70ca4e14ede00e430295eb86a44f46390ad94330ac1b3a040b73b1c6caae51ee111fdc554e4a8867906ff62cc02b6a5d8b5da22d9da12ed80326ab34b9ce9d42ac89a5330b4d78b50e07f2cebe5974ddff38fcc6c7b733a29fd1a38ea1a90226301e9ff3e37b57e27759970200e251255f89175059f4e9f207d373d8171ac0d9e967ef9c431612918409650cede8ca88a9772240f7dfc244cc9c0b8a70a197f2d200d6c87288ab71a682663dcd680ce9387347dd4bdb971d06782ac5d38f0b3560365a39844e4f3824c8312be84a067be1731339108115b35eef615c8d3d283fa522f9b6bfa0f159d31877af0e67406676aa8feaabf28381c49562d14bd53761a1875fbec9f6d7f19d6e5f75bec8b00f5e17552fcac0486b15eec68ee832890def4ed7929946a39d098e0e1ae1bfb007d3bc895ea72ad4321004efb5f0d27552a96b9fee2ea6ae168270c93be26d68c9e1ae457a22f864e468f51d03877854122378b78d4fec4eecefb996b8b8c58aa852f4e175bc5ffddb92aaff2c9659ede37a35e65db4e96575dffa0653506e5b719f13144c36e10a430be32f49cb30427b1a02a2bede714a95c2435e257a5101801b830a09a0e77afac67aebb740fabbafae812da900713927770e81026de15c6bf38815360a82aa9c30830988500fe0953a87b79e595ef04db8ab6c2de3cb2687f3e098ff05d216aee7014ba07ac5fd3d7a06a3a659c2d19ee7e2b1072ffcc4c092c05f395b77d1c6278424a29c56d98f6046cf870e6d1ca62d6edc7e50c16d3af576afa66219eca73fd0fb974be42cde17352e2481b47cc8f2b8a0552564e3cd27ddc8f9111868d9957fa1ddce68cab38e7a50b4d013f1f32fdca7a67e42b7c044ee439292325724ae1a944ef5f53e484d6055719d97f29af67262ffd4068fc93f433ce4fa36fa6b26958f20b2e255fadcdef38d6d9de0acab058a50097f836d3862ee4945a788c610d579eb7631aa3e39d77fe6f021c1c4047297150777619f52c1d605b30fbed1d0f64d81986c1564236d5529536da08cfd82c8f4762209f8f2cd44497ffb4a020ec8078d3a026cb22e6877f3e5810ae1c9d28074dfffbfd93701a640ae9d387a98e82c7c8a1401cd01a891d7fa50bc48eda6f6b2f638fa2a9ab81972b64e1496737d9c85f80cac24946d99ff76116a33825f4224d3e085c061defc520bdcc61693d8fbca6bf0d9417a01707614dd4b4e13f9eb40fd70ecfcd6f88bdcaf4b43bfcd1f3b8a182a1063c2ca3c0cd04ef80ad69bd4ae07f2fba59283c95a5f6f57d2ee8a0b41b5aeabd0ec80b5e22b0a87d039fe0c0a8698818555e90e0679d0c6b69fbe1d1537286379b4df4ae7b780054839dabfbf3afc6dc00af7faf9bb8a67e84e137d04c96883b6805321b82410f6f36419d3b3fc24fb0e2bd78c8eae12b1d02cdff9576d4f31f143599917caf355e7efb5aa512b776dfb8a04464ed6b89dee2062371404deb080ba039b677b144c7406a2a25afd34a42c3e802d7df3036dbe6982e8da9462a8ec3311f382b29e3934d044d8df87f0dc069f4804cbdda1f1ef8e86c6410875872083c5657417e4bed53f47b9f6f042e4c633fe311e1a9e9796b86b772bb7ee12a03a67d6c4dd538eead8d04ddf1c8f371f852cd6608ab4ba6dd88a75998a44c7cee3e4ba2c21a0c0c0301f1ed09854729b4878c7cf8ef197168e75b877a3353f95a6f841f14b6c3c25f406c84419bd9aa7c826202a64fa06f34ee3e94086543820d8729fbd759d6f46be6c7ac49e43623eadb9b04521f42f126f5d3e6d9f3bad32f5973552365a470d308ff438db32e988da35416590ab95e5126ffca73867a8b8bca1de5a5ef6ccfc68771bdbdd9c42d03a43cce983766651d3975cc4abe6e5e49d5775e9582be05db66cfaf38525fc704a45fdf850fe82b508fca465eeb31b7262201e6ff650662188354ea1a23c0d7bead470c603934d971557a05e9c0a8b0eebd1379497a844c3ef3ad0d1100eb536c7a7b79276b5e4e7cd56ab4f1edbb071a26ce0a4883c6332c425f1c1d9333b5cbcc7870ee71edf7b67def224ced69b655aba2a881045769021471cd312a7dac1f1202a7ecba8d6b77d90c5abeb08b254b559967487c22aadc401aa8f25e33820b1955111a044e4d653914a1e415f2ae7ba1c1208fc74921be0986619e34e3da9de9b9c7a6e5bb9b1691eb9eb21b816a49fc02dcf4d67fb0cfdfa9177ae09b4418d36d5b430aaa1d42e65bee162face4706715c30b22678671071e5d97c9f68c6cd46e8a44db8dac40c5ae48e2ab0cb5e5238c757d04e2231241dbef4fa27c441dfbd51aa819dd3784f1e823340bf21728f527d2d20b48dfc6e0bc4caebee659e07e2f16251aa59c8f78334dbcf34f54fa0b78f53deb1237556a4ad1af0545d043524314dda11368a4424cec508e436fcee07348c09b49f4b41e27655eb4687513ee085be1de902c4100a26207c6de29f52744e02be8f6c27b95caad4688b161035b4154686cc61115699bfc6abd3fcde291cd97e56ac467fb01568f5a95d12592a77f94b42205c004d0e733506e2707941711c4f7df12bf4e5436f8344082f364f8ed0840e5a82534aa70dd661cec046cb0b44ee47c00b822cf5ccd400442534f03c129ec1747d73fa350fd6c60bd4b391c67fb796505bd4fa62d7fc5a6eb1d61dd4470cfe2b74e8478093afa02a138cd6ba6a6ba160e1437ed0b693b7f21ecb1efb529d8dda43b014d3601e4f27f3233d9fb9098f235c951c6dd915721533071296f312d7feeab51ed1734a8761164c0ccbef1f614564a1bd366d7b34ee027bf838255da92df5c3f811449805435e81243a1c333382056d719c41771832102242904de32577ec0416e142f234ce646550597dc79f7203503f12759152e976b4976ad251b0c36d445995046b2eaf9b2e7fcdb11a096ca5d355c2364b9a6979533fa168d7f4c2a45e6f26677b7199bc828f91e4d9f51de30c76e4d3b73668b169e6db9f7963c1b5ac17ea1b074624eb77663f184f608f5ade723151f5a0aa9653e9dad3f4d2b9b74d321db22244442ff5e8b02fd0f18b292994580ffb0da17883abbd024b9e3e3b5d540d98fef3dc3c2ddcf5757d44d48049d0a8840b7c2c7d2d9dbf214acade8c64750efd064269cb467b3b28fa96be78da2eba907515530bf0cbc62cf512b03c004f46664692a2d5bc05772e959c85c99c11664a4a1dd75d636f2f50ec6f102d9a8af073e061b7d872de033ea2f1334da4bdcbad03835a08ef081f1f7bb67ccbfa729f726971e9d1a9cf457d0025ea95ef8d016d8cf08687d664ad0f73dfeee389aedcf397df12851c7c749baedda05d0589a6632d8f8dceb968099d4b237e6da9cfbc6c750359018667a0e27e28c0731935cee0e39f9594b9134bd11c4d5787b86ae92fea30f951bd6c688ec189c01ab94bb9aaadb5b7591e1689339908c91073801e88f17cb24e0bab43270781247cf7da3c55f7ef37aa76b1568239ca788478617481f455782cdab52ab29176ac5d4e83265cc6dc524ac3e83cc706120da516b07b3ca2e629080a074b9739e6fdde96e5e8286febc51a2beede4a69020a6808f1c7e1403c4f002d95476723ae8b4a5ef9a170223d1b45e3bf20adf761bcbf975e65e9809b25f67784aec30aa5699563c9612ab8b00eaf2eecdfb1056ca15e05eb896131bc90fae981a310669d12dceb7ba3f934bf3469a8083c956fdc924beabf9efe2fe7c333bb360cd42ea8a1ed2338b2e1153b352b637be7793deb20bc5bceb6cec77633875c4f479d83ae6482798a8f29a247af82f07dc4106877622bd1cf53b8d03828d1e8a74e0f5548ed4aa7a24f163730d308260906c47b253b2b78844b62d48dfb7a01d2e2b7cdc2e7e1acf7d37e396a165e6809775fe30bc0ff52e41a891af117590a2b2fe4685298e3a72d25f0b1fd5511375bff09b98dcfe18568746e73375e56123bdcc67935aff71b0e6fb933aaa72cbe1638e8d4fa21fa9357c62e08c6750eea6f32e8caab526933b5795c5db61c69ffdc24ee822098030a65026d3502b0165cc6d215312850c8cea3e5f3c5e809dfa7f08372fc85eb061320370bd52e3954218d269cfc97e79d1b7c2c443e4983c0f3ae38fa282a2de1de1909f589ea1c71f4b27f884acae47a6d74e321b001e1f3b873a43cfe0869d1dea5dbfe18777ba7671ad0130d1ff011a0f712c8b3b621d6561eb5f0f110ed260412a231dc918f08b75cc23bcfb55dbef3e46e1609b58bec78152d854a70a328c6f893f1eacb097504b7d134716a94a88d0f6cb1ced938b37e35eaa7e46c5df6903cd55047f9bb026f3c83b9e49e59cb9de3ae36f44a9f6447eb8e9956ac07b6156877ab92b291bb0590f4c8db9a343b405c6b46b9104f1b363ab76998f3f5cfdc9284d741acb31ddb76e9e35b1c73ebadf26f13454fd5feaf929158b9f7817108475f84a82c04221cf8f37ee337eb0b7ce009fba19365791bb363d55aced2920b2eed54e4f21fd31223110f9a348cde1672f4686b5543ffd1132cbf8022e9919ba00b1456944e3e09ae7f97d6de1608b7ac8b5ea2a1cd2f83ab50d8faf456f2b80047c3d538183efef751721e84a99d784ba9e1b01f142b0a9a8b4f6269b5ca9e2cfada31b85d3b4e546cbdcddb50b8f21d4fec2db890bc3dd094ea3b0db3b011e011fd65c0ea7ccad727360f674b5e1d18c23323a272152cc6d60196385cf7952e84b18088a3432ad5d46b5f4370c240c9e23f1bccbf9a3630acbd6d887a6b9d9c2cf35253ed1c757930ccdf67066734d24b15de75064aa7788fcea2bc2d3d3e467a5e2b567dd81f8b6122bbadf6271c52bd7e4e02682875af5cf90199643abea2718367359b252107372238f60dac7a8826c5272dec1662fade5a3238876255c82cf3a793eb7a9445878943981e58a5ee9f5554c4108a3c679764435ea057b9e1c814d63b8ac0afaaa37e63cdd9e7c10bec4b86fe4c29d48c8554680daa3fbf3a67262bf293df08ebd1fbca8c0ca976802d2a690f3fc4ffca3c652e59390a6484fcfbe8ea9757555bdca711506d125bd553880325e86c87e46892be1846752354df43d04787b26b5eae35c3cc54c3cb34425dde071c2699bc2ba3f79a7d42ba3e9f0441bbae50423a55d1eabb53825b3377e1f315532332404bf2484121b63743d6fc5754c73537aa87c73e677141e8fc6fd537ce3de7bafe96e28c9cdad09f1d5e5e3e4b70a22bbbb9cef8fde6b74e39611b22ca23d0d22af012c38c867441f6018f5c11b4a0acb9f9eb2b53caac863a0c1fc279f4753027077665fe56b0a555e37053bc543a0849c739a7e5e0dc6601dc5c41f7fd55f2d4b50a21ef177f10139ab230170120301564608c6efb0c5a99697b3ba9c810dc3c56edc99b9ac59d61db9de635d9d124c1bd434f54a0cfdbc8fcb542d84a19da06bbcbbf4e5df03f2472db937d8a8288de04da12255e05a2e0d0ea7a356612dbb12464f5b991cf1eed53e32b930cca065818c66166c4a6a9f19dd2d0902ebce72199654f3640dfc592a34bf1078a4b1bab7ce2b771c305b6409190a6023381c75de2554aebf4b93f9b4592d7bc3ed7950f6ac081e57457306ba43abaf3a5b6d3ac550f8e9c16a7b13e9f93c4d93698e2d405a8e2ed0e3f9f957c8c6b82818f4e552a6c8b17ba08b96e1ce0d45c7464e981ee1e9a271b6eadd68a9808a6ad6f48620a9f520da009bb0169ecbb1c01dd6be6ca99710dd358d2f12e756f99f36c71293b36c317617262407e8b1e95533c46f780a9476612b1d021b15dd38be21b08e9c47e5b7f6c01034ca97caf81eb263ec48b66da5507bfb5e9f1547a160d45dbc72b1812bde186135eedf94540c05c8ead7b2f05c300efb8f62887fa4533de39f9fbff60d5842663990ad10ec23cc6c3d91376ff0b864c0c2053f8e47b7a6c955412cc002a3f2124d99865fa2b1e6d143e8037d9e0c8761faaa7a227baf09c38bcc09847cddaba3fb6a9aa3a735b89b99ac6c73d4dc4fb9f330d5573d40da42095570dd3b16871294041a53021a859030821f33134530fc038b7ea042789e90907991035b9659e0cf7e731cd7ec80be36db39c007f44f3d5cc8d5ea096262d633320c3163ac9eeab6aa5e642fef11a666e6676fef2ff9db1a51561117e584231e9eb2826ec8dbb1383876f74a1103a589ebd16212e9471d020afd428f7eb0400fec8249ae4169358bbf1ac8edde9c47fa1d71cc6e94d0825a54d923ea0e51e704c2e619b6747091b73fa7c9d3e520cc9676feaaaf4146f603d4b25f77ad26fccd19f779c38b3b2564ae10af81a67519309a95d5bfd295ce5f54cf2152e1f71cf469b2edebff3af44402bd91316b56f496adef08ba7aa6b86241058dd35729cdfe0f641d21c5e972a729b367db1cc5e7519f4a236fcde4f2c19434411147402adbf63db2aed53847c59c45ed1692567e28293d1e98c62ba0493fca90d396467941a743e99f84094f7aaad5d4ca36c3fe51fb10ae28f0afe172413948e1b34145364a68bf397809ce1e4201486af7c65c64a0d18e751faba66cec7a7ca8eb8cd065725c3aa427017714c01dccecbbe6e633f248b273b62435866866c37d2899ffc14857c3b14650e79e8a68d8bfe8e68eb1a5435011c33c5fcab9ad8da0c74d9234e463ab7abdd4818838b9d71c62489170258b9d71c6416aecb8fa0d367dd9cc4a50a83b0a5f085a1d66f7b326cf76a7d2e0c50ac1c0d00d2e58068edd297002af41d770fafc764caa8b52b23341cc3370db5df711916e0b90af12de0e4f9d15ad4077053b718bf2e9d6e112d1c05536e86f46ea00cb783a48d9f007734f176c17844415c7b7cdc094f7731c1a0ab132b0996d5a49bfc6033aafe3a0d68487a9cf357ca80b6a71569a1ab781c788f6c8c31832e559fad609dc699dc011517eb0a8b8fae1b06abe3698c59afb847d675af5644659fb104f1c997e795b9ae0d305a13524ffb6a59e924b93d8adc8ca2f778f1ddcf0c8c1254590b675680a5697b01d196c51a3121a92185d8beea8f41e0e69e7461eb422cf3aa0860515850204f9203b5d0d3bc60366ec6131a7deff77ed1934c5ca853fd496987aa2662aa3414824a16c702dcda0461124a0f9bad25cb431f9ca32ec5053980818e5d90cc00fe6f83a4279f7eb4454de4d305c7582e81293652acda6b4188b4770ef916c06e1820120c6e526b8ffd3507850a35129ff92be26fc46e66a3874111efb7b7c60c71d3b8409667d48c1e625c8e40bc42ff2fd01bd94797f62429c8bf66ea2affb8996fe122823bd25d39f1b1d6a3732217e1779af0edaabc9658359611739b198115608c7368a2bd24bba0e67f7ad2cb82dd86ceb370c6b40b81754d9da88a75f614ed7ec36e38b5660ee75c13c8127f4da0696e9ec6ffdab5ebde57c24cfc720617adaed7c15fc3c9bd683b62bedfe628c0e9e642b4c89070c697e114ba4bff4b3c5bed733db53882dc98fb42303c12f2404c213f0471e32953272cf2acb0811acc1e9c8d81b0f0884b7651f9056959b673fe95c540236ff6b9eb246ed93445f27c5ce7373dfb16b6bc0317034274fbd018079746de4bf34b2ed65d81d8c8d5d52136f1301f76f5fa68f790fdd05c712eaafc74142285ad356b41348ea4b04a6d453b15c96dd48d28e3f01406ff245ece539a9b4ca1eaf59e3bd6c391218819431f3771775da3dea16aaaf18aa54bc1aa8bd5efb3677a9ce145ec3e973345f132a019fea7884f2fdadff929d20d78575b9bf8996fa7289c3b26292fd0b8cf0b65b06d9b76d71babd981fcce1d4d61e0af34f47288133bf039bf49c3b395a84d1515edd233d896d3671a09ba875bddf331f27e499705c89891ca957e6664afc87d5e21075d9f84c01901c5bdbe00675628b49fb034026cde180849865820ba58e1d660f3ec914b447b4f9135e203d88a892afb66711f1725f6635446eb2bf8dd6ac8ac1fca0770f125f241a431b56b775452d376b2b72e8c96b24dc3db450ee15e5f023fba93080fea069c0a7ebf2352bf9f3cd70cce63258e7025dda4014c31b1141dd9bd18fef0f76b5825b1856fa3f582915ba041899dd3be1248c555297817976afd7a725d70b7bc5e01b570265c82402a0355b1ddaf2c0adcbcb829774960009c2a0588a5d1b157c0df70de929b66e677531fe9ce68ef0ddf90ba3bf9ba6404a045c74aa3204af1c53ce5d7acd45f4404ed5ad1bda4f2d20dbda0195ba9b5ad48ec5267aa2b1dfdb27e6c5fe3c998563d5495e0612a2f185463d5c51c01baa18d4f50a79ffb4634efc4f93d39a6fbfc4c533a58e48b0a1cb5a3b64dd8da959130ccc5fbe9f7e83e434d4e781eaea4a9bda14691ea5da3e35ef9190d990c8154738f50082859f0ef8d4c61edb8e9f7df4eec8c2152bf84faf07a9b58df4a54a5c3e496d7344306fbbb4bc10876161b9031d128f91bbc3977af854ebcafb6028a13a224f8353d21cf6859c16c60bd30da4545150cea71c5dd4f8178fefd38bbba6b1e03aa11e18119f747fd2e4e70cb24f45dc472b58ae0e13db21af2f3188ba635c24c57485ed744e7352511fc5c12fe239c1fcc59a241f9d46c5bbc77a8beabb0f200d6e7cc992a82335c5e75f7df45630a28f44f7dc32a29d76184ef24e35dcd380f4fc46dea39dd58d48e11614fcf902a3f3e3bcbea86c7bdcfc7f936bdeb28dc1d7c6e381deb9057e61c02b372693a0dd9715e7fb7ea62fa7e0603d3e8674f876d024be34fa4b298ea1486b5b91a0e8a794660ccf19d4d4f4a6bd4e698f2350c7249fd3f1ee2600479a390f444e7d0896facd4cf4a971e58e1de66faa52df7675e37fd5b5bb0f8c0441b9b7479045c50fc37e07aa9418589235c3fe0075caf08cc24d7f6a40eb577bc99340f174b184913236f646e646abb58b0fbd201de875260c70c9d5037b0023a714b7b989db67f1c1e218cf0b5b4e1e690b7bb7286c8bf1b30ea5aa3dca74d1930560d2c8cc450d9db921398ebf9ceccb5120aa0e41dbd29b5630f93bf5f6fbdc69ef5eaf687ac33c17f0b7e1ce0f4b0a8e1d21bfe34639e7eb8778d7de86483b0212048fa626887d1106f258826bd77e4540853a268e9cfea6604fabd356731d7970aa750e093b2e021c8cafaaa21eb87f275403547faa6a07f4d906d88f40f0c68115fbfdcac566ac9443e50403bb2076cfe1554fdc364b1e301963bfb4f1336f265e600086f7b9e9bff59ca297b43732d7fcba4bad18d6cc5469134e75df3c1427288d5258795111b9899e42e403d1ce36a797afaab609a003509be1362fba4ef4eb8f5c79828991b64e2f7bc72376557ad185a75575d7205e1a9d42783381c876ed2575fffb4fc4564d20aba4b6790d528991c391e65e0cbeb6dafe64bdbf457ed6c2bce7bfb96f1121cdcb2777ea224f25184f4ae3964896eea51c4b0e98b586f3663b0f30ab61585d27b2d9053127766f572f5c6e872862eaff785487170908bbb5c7028745b2194619f85f3c34ded3395c5ac33f9030eefdad112de31748416d2dad9f843035af441294f41a287bdd34789b3bf96618cd1ca1b5182cbf7ee0e1a532056b20c148137f9d4fd33d92e7ae828e5e8abeb5d42b87dfb3ac855d8526947fd3093783b61caefa9eadb9b114137bf63c0647ed5d61224d75ddb85011ce935bf2be48baa65ec6e7742dae81d0491804a56268690a7bf8d9480cd48df6d999a7b2a730cce5aa5beed78d23ad7232325f38480b8ca5d676aff0fb1d87c017903a9a66d9b0a14f4dddaa00a647a756e727921245ffbf04396ac01545177d8ecea1baae2f49dac945582720b636421cb388a5b2c9fe6ec08c1afdcde9aa4a179a7991719824bedfc9085ad3e8648d1d3368bfda72c1a01497ef2c661983c7dcbb3a9a91761c9694ab6b5f20410029c277d2957215b203044dbd8c06b4cba8e3f6713ed72358e288c2758048f2086563b3bfe8392d7a67f51334de8c6dcca9681485745e3b817d88bfd88260573d63d04f8360ff5ade869fc306ec02f6629ef942231002d06dcd281cd9963bd11edcf8a43fc99b47a9b68a49221d0b3f09f5ebc4e97a422d49f5e46e81796be08bb390a1c62f69df6d60291d72eedd0b8257ae76c2dedc904479b003f0414f6f52829d51b6c3ce8ff2170a124bb2ca9a78607ae259ba46636f025136e7be3cee8795f2e80a366866bc0d0ae3053c39033c98a8072ec4d37e446bb48881933d6b2062c72f6644be97d3c81e3558215edb19fcf564bf512224ad0d4188c2d29c5e70419f08b5021bf15ba55f9d68b37eaaa3c5b891d16a6ebedb9eb47de24a157b5286c146562205623428e58b7144ed9337ef79845d29470a7396d5b3dd224512b08458ea1fd90f56ce09de70c75f5d005206061e388a6daf46e8b2c3fbf7aeff39ec47327fc89e3941fc50066e0b6a0ce6bebe13d9aeb29803264dbfc3df6e561fe4d196bd6778b7a6a45e7a31e61c3a0d58cba98e659d5605bb0175bf195c0eab96bd522e516673a43619e6b30eba495c50099ff52f274d51d3df951cffdb58a4696cad01c808d4994fe2b9731e0b779cc13e26354331dcc90041f773bc6b996dfe15dd6c1d02b4598928394c9761bf60897a586c5103ad942b2eaaa24613f7a21528415ccebfcccb6e15f4680fd935e82ba602b660e91759096bb52a7994ec7911bb89b8af23effeb6261e05cba199ee960ff788bee168906bc493a0429165445f375c4b455ec1b5599b21cd2b0fdaa2bd2b2b83d4544428dd7aa2d62557d432aba8d655ec1d4ea21224ef151cc9dddc034c03d1f1e1b32e7ccfc7962a8e81d58d822941a29955c0f5c5d1164016bba0ee833fa6355db279c5df7227be312195586887d407c915f11fe2d5fc768dd899b600e8ce7422e83d50aa5cdf4ec55d830411b2b8b488133afcb24d5ed93b079175f3e58c2da433c9f4a0c032397bb7e36b32387c969fed3514e4a991c5fee1c0bc0bbb3653b07004332ef02668bede5d24331792ec85488b761249cd533d77cab266e2e8528a5ae11330692bb2ccd6bc09ab831cf556ef1807dd7e3835d87c3b893eb39c56eb14a75ec2ff877cbf419a5c783b61e412bffe6e170298adfba6f68cabc13f6cd401d5dc6873e01122bdfd3617983bb5af7d8214b31b995ac77d139840f1e7737cd05bac43e0bfefed963f52c027d77e0035d53c8595d8ae0f7b615cf14fc469d49c2f94b4ae9e4427a501580caf79f89272cb70f91d7e64e40a2ee1014fe8fbaa434e16f8ab425560a9baea4777d4279b963b9d98d6652bc5645f577e5d9e977ab5449e5e8ecb1abe0f3625a583ecd7040da4779012c827ae4ef9e2364aa50aaed888b22492d49d5dbb654081e9d3e2c8529968841d7279d2899a71b89a68a0471f73d3cb958bb4e25256bf37e55e80e2bc80075ed83663fb3765d028d79bfcbcf17ad1c4909be97520c8897ee724c90a1bb608d5de507bc97d108ba1f62eee418605859e6e2f02bfc7247e5006a8af6614f8d953b5b52b53d6373188624b27d869a2db62cdc534e6a9700f209123279febe188151dd1bbe4b826e7cb7bace40cbd52ba3b5c3fba01b2d37c0f0fb41d3f3cda1112a2babfa8e225aa52d4fea643fff2451008dac140462c2710690ea5849a02572e5125d5bc6d392ba79b0ebb94e2ba4ed5d84218621ef541e6116b7eb7f53265e429eb44dec42ef859fbc391380b26b67dbdca81f1d78eb878efdcc40abf84e31f397f31ab46e4a78c1c168b4471bd1051396ba94c620ccf3a03bf44bcf9e0137e2ae4d39419bb57c59de067dded6fd59f1fdb3711ac7c025519a65fd2e8f60d4ec6fd30a5bbe3bceedfd3b4c3ea4b9202c5233bd0bf533d0d62efed2dbdc05a6e0c53c77f26ad74406bab680729ff2e9b2c4b67c94f574d4a136a409370f4446810b729344bb673f5f15a6d156fb638c7faa6bca9f9b533c6f0625fb84c5352fb5ac491272c9c7f7fc40d9a9af4384e546d63f9e54b3cc4e40f6e2e743781c59244b6850f6c83219bf1f48977af3f04c3433d399b0c4d73f5656a514c5896e35e8b3749cf9fe816249913187fe7458f215ff196e346b22b9b88878f3afab36d09fa31887970ca4df4525d60ed09d6719aeb8fa4717de556cd99fc15bd4c7b1a1165b65922cc1d922df9733478a9261cf0f264253d28834c8f1e9efbfe1a5ec488431e939e2e3edfcab559745794367e60e2cdc07067920a296429f329a1fd8ce9b8d9eaee0435ad67f80f330a965870ea4d5f8a8a7a3135ca822833f9e7033a73051f024d50e71b54e0f82e3dce4bb07a575d50d01121e79bf9730c8d93a7c6da9dfbd97e83a53fc699aa53d33d2a611964f0facf46e64e508c411047e26530953910f6d23d0c61e717af154c9740bc52b38c065cc95e4a950dfe1fb402a48cfe018a81cd893f0550a826ed7f22ff93b2f732136c08271480b699f71a4c1f67278c1fbac387fbbd186cebff77a1f70f5a907ce1498bcb3f95a56fdd48f12eb406469e2dd08f0bd4bfc3de56fb4d53b42ea2416aa50df55739d272a24ed2890c1a5bdc831cf86b147dcd47318adeb7b671d01f3ae183894b63444a220ccd9cfdbf5f24dfe3a842470eac438aebc4e802ca68d98170940f72c534be40935921fd5caee248abccfbd18b6212ce17590a6411bdcc139d41508e18a60e6c414d88feee95432fd2bf3fbac0e66506dda144b3efa18bdf8cd1251ee391d948121678358aa0247a8511e32ce31204a2cc7c65e8ebf166771ade047c55486edcaf390a5e7eb117fb97a94fea52193a3c8504266dea5151125b242daa0cc6dfe9662fc624227836fa6fa21cae502d7048cdffd252611a8449005298e5ed2985f329c42c9e68ff561007880bef7152a9c42554bf8995210b0f4f95e3f4f2952ce00512986bb222d3ac906360a6afa2aa494c219f227237db9e185af0ed4d6e2d378ef7c041bef1df0755daa347c30061e571ecc3ab5a3397d0f13dadc287d4ddd07841e3035a10570894c86266e1aa47622e97bf1efd2a2e2284932af97ebda9082eff02280632c543a8ad70770af983a8859c04a959877dace846f8a075ea7d26b1939640992613a024bdc4d4d69f27ff38e3214efcff363605748a40657a8142d9d50df6a109770fd797cb79490f6274672595098e1431eb7c04a8427a545b2c87e58b0d7dea325c85a40ade27d0158853ad38b3502daf3cc380770afdb10d25b622ac21a09b0333a801ebc4ec9749f2faa8e5e309f977d2ee87ae1fcc7dd7b823cceff7bc1daba6fe824d2a23c289095ea400e98bb2b7a9006eb83d93f164c92445e83376a4e6d0d54990cad15a0545411744ddb4c0887ca72e4ec1bd5fcac2c27ca301bb898a206485c2f7f4953b2ff60483025eb165bc0b7e3e76cf036d49558972062af0466ebbe8f996639097f40aba3fa521344ac176d85836aee7140db0b73f22a12e3b1d961b3815a16e88f3cd3508327273ae039c135b60895f3975a2a923b8914f4aff2b2cff98e1ef27887c1569b78dd5ee4260e1248656e794ba04099a60c84ee66b39ab6ddc1ccf48279da001acc9644cadc79bf1dab46402a1ebf5b5815161b20efbae29dfc5079d3b5b146ee473e08ba555d813d3c1cd244909db2e67da7026ea84641d619449f42740e4ce87233d043bbff1bf9db7fb53c170a2305cc1d69b322cf97a2dab273ba38adc93fb6da48e5ff3a894e829e2cd832267643e36c9a7ff3eaf74d9a2dcbc26fbac841e484dac739b5f1f5f0feb7b811207e922e1b944104ad1303e0d77ddbe136c1ac725a95ae9b3f4333eb7a0491efe24f962c08ba87a6fc96b82c65859684bef76d4b289a2017849c8bc05459de91d995d9d82c1caa36fe74cbb54ee1cac5462a3f503d6e5c90ee716a6b5c8170632a6f5aefeb1b7a3f68c1726dae1fc3e51103d4d90449f4df89519cd47c1dd15373298d4686bc5d6dd49561b9903e8d131be3d663e96bc78296463e7783a59c2c42a5c468e330bcb863cbe0d58ab2d1d84ead224ea30e45aec4b528a75b9eaa5381aa86891ae346c5fc3dc9371150a2ae71a9f11fcbfd181a654a6d0be220dc9e943e4f4a310ea1e465da44c45873053c6a02fdf988c7a9ddb6c789bf0ba9895b92520d8b31d16597d7bcac6f5621f6e9628d202cef27bb8e8f2c40d84a87d48042999c69313c1a51668be2198455fd800277114e0bf92ef741f0868b72371c2c77a164e87777ef3cd1139ef8ddc6806adc45ed4b031a17d1f723789e9431877ce07f5a105f629ef561672e13ee931bd39a95fa0385bc7cfe5e6ef5140f4f0b785176e79a0213fb3ab12a0eb2b1debae7a8b0fe3d945b6a77dc2d0d7361352bc366ec509f36926a03efad2056838b114c8b1bb6db587251a99b2fe5a698d6c99babb21576e1ac564fe14370a4829f91dc3c751a710218128048a097d6183360c5e0cad11cd6b49625b34beeb0fbc6e3db1a1c831d711c6ccf700624fdc0172f3b30ee508f47371cfd55713fbe9f5bc2bd0603a811192e465725a4d9bc8338d450eb16451b46d9acf09cd752bdc2299cd9a34d001162c5386f3b5c9402caeca6d9deba53523cf0b595716a2b1d761fac293b8c54d254aa691d390e7926e73d14c63481ea79b1dba59cd097ca664b7008fbf4e90a4002f23a28a828db25ae0495da90fac26f9049d567f91028d9dc4f9be05439b533733a50072106b1c66fd6482fb4597d9177c32b49cae8f4d1822a099b26c95dd79cc17d5c4e55600202e9f8e37ee47ee8667629f8cfa125fde64e8dd97f3200c23cf98755fc5e6898d03ac57ce52b7f07e7d21a240b99c9da4748584701a0c5b8a55b700ec4c1fb4838708b9f653b4f74126a7799f4d12afe0cc732fcfb4f7dc7209bfbf5da3dce2353074945a3b78cf269f4bdd5a70969317d7e42938fb9d3333d95d44bf950e5930dabfeff7a20c4923b6a28e4569022e8ab750faf8a8d6f3aa6c57e3af54e7d4bfb1fe36b4738d1320eb48cab7a48a53c626cf1abe9f378d5921e29fc97de964a431f64a9c14380627a85f75daf8d7c6dc9d101fca679717d8b23d7e8410f52e27d6e1c6ea1a6d268523099045e37428f959db0d11a1c3faec6a3f2dee363b8ae0bc7d9cc08ea53e4e439c546afb9544890ae6bedc96f0b837049fbccb35454e8ab65615cb078ee77dfe1dcb4aa1a1e121529931c7705ed50d4c7a763b6d4c2c3c099437115a16cacf0a61213789561eee231f6a456fb0ea485e21809abf4bb8bb5612d1a8e682955a9ed9dd4a2c75a45d79c03d67c50e1b94368761aa754ea057059b6dfe5cef7f0201b4738a1a794ee4605a2caa3779843c116f8a41ee0f5c9c5b0d5f742b43672e439c13e07843737cdfd28eab0545ecac71dc01d1597cc059f6857f609cd32191895ced35fcf9684f44e7091f0b7e83b63e4b6428842610afc4639e8f2b79de506cdb96da107843f50afa4b45d5d5e3ca3dba6247a0f9f24f7ad27bab9c547946f431452cc89387dcfa1c72f9a625863502f9a9f588f182e2672e2dcb20ebd1695b5c734e758c62d9ea4679ddce1111bf0909e7bc3b3ca95d4ae554edc8bb868844a1b038b2469bc3694f921eff916c80bd070d98f60f5d76645f1807e230f407b8d98e175ee25970627cffd47472fcc2e1cb764a5e3a330af860bb1f58d20332d43680a5c3fafed00e53da82e00c0d8ab5b6423c6be6f6b8b3288c3ecf18fc597c48f5ed18a2061b69c560b59689b2261405c1d486c6cdc94494d4a8ea23bbbdb533c45246f9f79a3fbb17c41a3c3a6bbcffaed6cef8642eec292b2bd9931127c8cb9f197b590f54a0837f2515793d4bdec214c2dac07952b3556af282973409447605647ff6148917e928aebd5fb9eb5a0c0ce7f5beeb08ccc1b23feed4f10b7081744759038c8c431525c6a3d2d5cc33ad217701317e863eb56d3b07ba2cdccb985035e67bde6aec75bf3bedd312deab79df9e576edca571306dfc294d78c5d050f61d00dd4c0f47c1a2fef7f7440919daa8d56c84f68b7400964f309b78959e03844973b15fc3245ad12e7e72b02cddd73cb43a71f7c523a6e43d4a07e99dcf8f4f8ef9b4bf242dae3d23b48bbbfe534075795b6d4222901008236d4ba539792e446727a3d99ffbd049199fd5ad51dadcf76a4e261a330f2814c76101e3e2606722f564bbfade0d032eb826117699ef27046cc7ef4129c7137b04a9ae7e51d2bd4c03292d166cd99b5791dcfccf778c5e96b88c15f7083d1a66542784fe414ffb8e6527ebd828179e6cd37ad541f1ca55e80135a4891ec185a5ab83543b08618be87e2eb5b793fd6c507decfb651c0b26a46c67298fb3e6343e9d56827d8720baba2c259564c3c124fe07b216c35b442e1646828aa9eee8b39a6ffb46e37f86c07712ae07419d3601b633b2a5800347647462983411a044cc20f8827208c3ecbd11d5e4bd15fd626260ee77113cb9b24e6ef2a2158f9da4e274be6b361ad18df5d4512d80273cfadd8a1a97d8b93aa69680b82b3301bc61d3c9d353d1385fb99ca313130578995d533a8e62bc1c3e6f94f8119a4023b56e1cda60d97be0404cef2865007348b0d619199520669a692d8738c207de7451ec504d8e0a38b1cafa27cf78cb4db3a0e7d8217adafba92bd1e691962c92db25050485c48e56e7b7d3c8c0260fac59e89e9fd265d4354d59b1ca750a27e8fe75d69a972f1352bf9e8f214156e9b957341a4d858001dc4a613bcc4d97d4a7471e19c2ed90adbc259d48f899021b7043427422bda432310290b88a9831b073c3441b26cdf563b16ce2bd203319bcf40ba9dff80efcd01db2010b0df123d328c52950474d6036736454686d22c08b405c526648be438a44f881e2985a7272540b03b4e469085383cf1e18742d4b6d9440dbf7506d9d35b1fde987c3047022e25f342f4c1f2e6cd2e8587d0d181c1442f5f345228623c00bae23ab5661900efb09d349b90c75c35b41c979ffc80f4ef92948b1a2880a444142b1b08183bfc4af0b606b1b2b14c1156e904f993857a254193f6b945b2ec7b1b84fa1d666a5686cabf2d97f7d4aaac5a2f793af96c3597ebac8f1254f1877aa5c86b4924caba791b13960d406aa37aaeaec27b99c7155c49b75c01fab6a42ab7c805295a602fecc15b2257fcd0d9e41abaec3e478351d62ba20094b5649ad66e10e1b18efe19325d1bd986e2be9ba6bb924248f1959638aa3c99684128612fd6362462fc265a225ab4582e08b716951aed4ae2007deba0c47272548c4ee2c184baf0e80271fbec09a4258d1ce0af6e0cc5ea40a8a964d19c6e3bc68320fd9894f3623198f62c12dda9042ed38248eeeacd9fe6da6901e665aaf7cafc60510fe76acb43caa38963cd1fc365ef4921a86f2e6796394946191d45d8f4d0c6458afff13820f752add7c5bb072104c68a22e84964a7b93cb35e190667ca753c0ceb0ec67d2591b3b09bc0b39ae4c7c6a869bc9ca095c73af2a05d395105a044d7837d1687f8519ced256d45a883a0b71cc5d433fb6eccd6684f3867911bcf0c370d140b8d553182da16a67f4c620d8093c51a609fefd5fa095fa0cd654abcb10a3fa6dfe20c33178b6f645a2e71461fe2a65000cda78a386513de8211a4be99d8d9e9dcd68c6610258aba0baf5d401db0a47ed508fe3e8f682724c566a8373fe11571ef2684387fe9b326dc5fda0d236429a1f0705ed0535886ad38ff261b1d820d2bd0c66bfe58e6d7d049cdcaa45ebc3032935f7ab2be8328dc6a74dace44175f32730e7c58e97d570e2d0bd026b0de31207a8e771fcabe79b133d719b183778a0af60189bca51e10ab23897d6cc20a6f5611c9cd4d84062fee8e719375c175b62c5e7e14c2982d142d7fcea7121f8eb91effdddaea60487d119e50eebe35e44b197b04fa8e5f4190671d11b00b927fb859604be824c6cd5649efed767ceeb7068d7c636a5d9d753c45a221f720e2f44d9572370c62f77cb46629bfe3e95cefed7fef8480bdc3eb8de2680ec9fc51936fc3b033e47954e646ff476c81988b35590d1ab15f3db95489fe846158942dda755df84b1761a01befe200e93713b760d118e368210779ca730dba61394f541d360eb1157386716eea4c2410b2034ddc8bfabac6fcd82bbbcb02ba7940ac4b250111440c5d8a081f6052ccae7b10196ae116ef8b6baa4085211a14c1546d3a09d2caf0d1ed33771719cb0ec50614aa2f08278d5da35ce6e0aeebd0c832513624ad574f5ec9be153d92db9234d5366a1e641e6934112ad4556be8f209a1b7cc9422f39bd80ebff8178cc35fb77b535cc3c99f4a10d3674d7806456e4d5d1a5749cc599861ff84b22cf61096bea3f60924355b39ab258eb61ec85130f37eadc4d2c5ce1adec4b3af249b51dd6acee1d58a5e0c87262bd474457cce0d2c898a653f6c941f9a0bb12553d52b4ae33cceabb3bae676bd84104d8369b62b3522bfa973ebc1c9a1e12101c13aea907510073066e064f6c17094e3a554ee70538e2fad7aa1704a0a2a0bea99c7d765769a8adadcf9112fc99b22edda0eb5f99fb848723d42afa0cc1a2661d16ad68061440fc2f83820a9b539f910884af175896692d65d87f41d6ef9060107145bfa918ce81038a2b9fe0e232628641b9b91aa774c541a9d32a124374024b4724d12affe640b462d17ad8c03fd55a24a108193075c51e3bef4b0b41b8c388b3aa658af7d5c69e99bad4909cde601ef20d943930133952b3474a6dbe77ff346de7db01f4f804eeefc40a7b65007a509181819338a5517bdbdba246f1d5be79b6ed89ed08d6838fa32eec70b5489c94fb8aae30d7b0b7c7c873f0297ea18b03a995dc1a8e37528b89504b59ebe54813cb2e4fd4bec3da6419cfa1d1ff6155cd543d3ba9c1e05c9333155706fe46addc3282fe903670a97577c67cac119898c74771d7a8ce764857bc9d3694c2a288ce0a94639676a4f43a984356dbc2e4f5cb9ab0dd2e35c68e2688178aa78dc0f3d306530fc90566cc614db6330fc6d358558123921fef4c4778b8b6140843f63a5568ec0c031b017a2206436a8a746cb6bfd1448cc892e1554fdb9b20a6a0d7febcf24ee6bbe19f11e0eac1036e38c7c9f73942c18ab6accc02f6de2848699b7dd3279a9e5ecad3aa78855479d7df61f310b94f8a3eedf98f4e44b49c9d63e166fa947ea8d4324831489b28d47a2005f0cff497a4048ebb0254b063b2c520bf4906d1270e763d1262e89c223cf79e1890f8df0b8b482875b141eaf5ba23cc7e7a4d5a8cec02cb3777a77a7961abff7b27a57c739ce851d3ec2b0a75721c61c8da6e3fb43f7220136a770b24e5c9c8776a812760af61d310f3c16fa35e3afa61f54cf2da9284e97bc87995dd452db40fb8e4cead0cf6f24299e5fa7966242b7ff8fd2d45bceae102d984f672e6f7041c79dbdaa2f0765b5fa9913ad0e7ab1d66cf850f4cb280254c36807c16cd3607d789513c2f3b2e49a9b5cad1507cf41bc010a45b7393db282f8005cc86ebef7c8bd8fd9d0e5bd7cd49c9e8453f3fca6e0f0ed9d6cfb2f975798de868d25e9c8a50a903d28bdb468ffb9a40bddcbafc96c28197e41a8dca8ad3af36db6d8817409e35a981774970c6acfc71b4096ce23b1cdeadc44bd047285d867c471c871f9ed04c55bb1a7c922d5fd80e6f2dd48cbf660f7a6411cddc1468cb947c5205be7261e9862977ee786a4c933af2c7d772db30774b50da34a1cd1f51db2f587817241fe578214e4b3fd37da6020ae0f40f93e8e4646dedd72d045191d00a999a194e511d0aa4b31d13a9592f68a26b14b018a47a64692f63c3dfbe2775e294e411873e3d54f94636c22f06e16c861a023dd04ed618b5ce3dd6e1d8323a2b4a8fdba2d18dac7ed35d815335f5368d04f6322c0ca2c92bf5fe302baa2c04019991042adc06ef4bc074c91a708e3d50f062cc9834a2084e7d5a03452026038239a90d72dfb4e816802b1579d7a02a519c99451cba631a9d526907d360b410d6d4f1dc3f370f550926d0a1deb4daced6fef119df2bd47c096749a2009992aac15045e8562be48f5e27de31b63a58f91721766400324026aa410dbd30635fcf321ef2b4b1c3f30f8a2e1c10d3e6a5b1473d928299135e53213a65008337fe325b6f48166ee059510dea197ecdad7cc4d1173c94621a40b698775c40fd838522def73825b0a7aa3e92a21c9cf114ddbec6b983b368442cad48631bdcb475bcbbef902fdffcdc14a9c86fd0966967542ecf7ff102037ac71a5110c7ebbfec3c0b2edc22647bef826fa3f80ace99aa79b95bbba35b900e0e956e12a072edd67de50e7d09d77b049fb4b0e9ee0052f2c5df2880a7224c3dd3bbe233da12d25c2a246b18985d369c21a00a4548c674c26dc74e2f2484c21399dbed021850f2957309ef752195e48adf2812715cfcb638811b28f4318417c49fbc5e4a863a06bbf94f03a5cc2bd998fbde2f6e9caa30c4427f3eb59957c074393960e400c30c19dd2a9bedd94a485923ffc43f5ec6177b57b9d387618b899cf72e61291bc521e065cc39a2f7e6bc7d333b2c147216e771ba32ae99094e36ec798cce35e762d7a67c8b271607780c506f89340c795b17b4a58be63b92b870888a859c30defc7d3fae701ceb52d784c8c28672abdbb7b5fd6fa49674c508a5eacb9438f96f01bea0d8d4cfb3be21cceb96f7f5f7abad66f1ca1cf391880301f89968c72e6d033ca275d213bae6dfbab185fd59f7bbd5a3f9e3d1ef7e7896e3096ed961714bf2aabbe7932b76464484d38efd408c62c8b9a57aa933444983d550bfd709388ff2ecfd9df675a8746a82c9b2ae9c2a3607cf58e26f330fb08a7dfeccee5e54905840c9f8e987ffe0551e414f6c4f5e2cd8b24b06f53061307d18d18b260bf5cb7e09de5a7cb7795825c21a62b471011657ab692ce3449dccb0778c48851b52b5a27bcd11b758a885385a29b3a8cbf6e9b659824633a8099f3d3f28331966c255d8b1935b5cf28531da220630571a129a7517f854987ad901e9d5c30941beba9f72df20345d08b66cb807a0e824b583e87ec4c382bd347726b35f22b409dfd05b5b88a8614e3df64061ce9ce61aeb210f0334fddd75dfaef82aaae80aa915b55ddc9e20c54760f64868e0005cb4d98842b37a9a8082ed6d248b4270b28ba8a8887b01a025fc9360556a024d33e0b3280cd64424d2c9429da8f72fb275fd1e612fe3e843b6403dee83af74bd4b4cbff53992e12c9cfbda559ebf4a87a07897b716e673c53d129994bb6bf7124368326fa969a4a04b6f12539569ed102086ad54e52a547b70b946235703809039dc1c9de333df25cd5a5ab9232ca4d857db026fa480c47eb7b888a80f69bb911c2f35fa08ccca6a4cb14ee899be15e00b6965c190a9f3af9787c586be2f7ca14e21feedc1641e2ea32289afd0fc9064619d96c257f396db3cbec5aebb0515cd078e94c267a9f19fffd021c9e94d7bf8a07dba6c96019075b787ea5beb86619115e82de9bfbd3d436ac834bdc9e826f0b7abd04dec13aa914640eda93167f5be87602a152aa6062c4c3ec282318ebfbdbc93c631e5e8a07488c1f905cf258c66dfd29029124cf3ea5d2843ffe4621c22c945512c7792c49d125be5f79d0022c97f94d68e2765101ea8e3ee44ca672b21454a19193d1b08d94d38c80518cf4a96eb3a176ccf2c1a59089bb7aaabeade6e0642bcad0fee1d7f8f3b461d07836138e170e39b038b57ef0165b0ea3717b01507f7d45b1264976ddc2d4f90bd264a35ccc6eaa591f962a4b14d113a5148381e925e4ee023acd11c8fe91e73c486f86ef326dba4afb6e4d1c2d07e62882a96b63f25dd23431808210e6eba5d19962638b3d5c0c52486fb13cf4c2f59bd0389cce7763fab5af2f7523a9ea81d052e1bf69f50d9b374aaa0c48c07fabc752e704f467b32667eb05bf86d46d06f5002056655febef512aa4b7040c837a9578e67c8320f62cd753a4aecc88a951211901578d16fef2994517366484858736bc89a170e78497329fba82636486d9414e9f6a6ef2c3b7f15e778cf66c677ae53d1ecdd4dfcd23adbf1d168718ec9bb4e8c37c73bcc5b61a2e8cde6df29fb4f2a92cdcbbf8b76aefc21f88085ca53ff4104e187ed598b0cbbe6d90adeaf8c32bd58ab396a908a70a13bb4357bcb9c4d647eed9b6c350d173bcdbd28b45dd2e822af9c2d8d44c06bf671e88d9eb435d2dd8db8065abd3fe1759874eacf30e3bd602678231c0d0cd0026f0bd0880a6cfcf420895886ba38873a87c2d017c8e6eb77eb27778acf16985b8f424122d6eb68c06a764b407b5bf88362f2a77f82b331f6b0380c31b37de75daf056473c34a90d9b4909d24a3859f479163f9d47bf184ed18601a354249eeb715bf2db969fa2eab607478766abee90b5375d550b8b8c00a5c07ac3b522fe6b845f986063d64674c0221fc8c607f707799d25064183be3872888638f45138403830c127a6743b830ace5a217b1931c5526d720b0bebcc4b13c8b71f1833be85bc6432ebcb8e9b3915a025832c2a558fa25a519b0048b716b453df72f6175e24cdeb6c9e9accbcb45bcd424bd7962a91cbdd9ace122a94a1fc18008324b4a790478d8a8911363a78a8617f35e1726e4a28d98a16fa20cde7e94b670d42cae1033e257e9c62c79311e4811bd0d54a0b35cfcf51cd26777c03dfc4c9a06cfe32705fee105cae01405eef89c392132b090e2f9e6fdd1f314c41029830489614f00518a8bcb40ba74ef8924a9cbf16a753bc0680a0c298cedcaf9c2b15f74245f708ff62c22ac27f8be91422c460458c781fec4f5b8695174ae97b454f204001afdd6c84ab95cac294ad945e7e6247a8f695157296469458be0d7a3661880c26bc79781916cb4514fb5305234aaf69fe3aa89bebf8fc88c2587c446d01149e2f8bc6746a11e1eed9264f48f90266e6f5b267474e39ac090385d2e023f70e6659c51c1ce2665e2af6acd9927b96f30642fa8938d96aa42c2722cb782cd1d086a4c7ff2c7ff160dbe93541b52d1b4bc9674d6406b691ed95ad65b60b4fbc31015035e389dc1b783c7193202735c81f604e20bd581764b30b84efc9fdefd0e36e656c34f789d6bc5863a03dcec91c8ebbfc837b677948ab32c6d32e2584fc2becdd494addb278282e21038fc84fc653ec4cc576136119621d6326318e36ad302ae098feaf686add07361d1b2f3e37f3461d4393fc493465f786dd616b97e073b9d5d3e4ee6a4fb821776e3a7aa6f19765cb71e8860033be2badd2bdb14bbcba2052e4239f6b8ffe90c0202d9bf38ee68afed51e140783a1018167474dffa52ac64e4a3538fe498763be0d39b4a5026234742c502df5ac5bc29571989d7d4313a71d7c2e57a8934439a50c5da05fbd81a3f8d5c33c7c80d2e405a586b891d7dcbbd3f07e8d7423c474b6a2b1d7e3ad4cfe8c3bb83a863b6543e9da3131ebc765b95f9085611b24975bf82b4e97f2786ca3c56cd38a0cba382e04a0ca5ae68352a39c21bc63d151a3b31d727b7c313bb50e6b603a693239980039dfff023fbb348d8dc057c95c27674bb17a976608860c864c01fcc0f2e339eb1f10bdd1fb4edcb1d095fe62ae64b226aee980a9b4d01a6f6df4d04e9fe81d6584ed7c97c186891a202b6d979aa0df2e9145001258c6923ab297f7b08f483288ee149948a2bfb36f29cb084875ca4c79a155eb673eb3ba2bb653c86c907cb109408ba74b879e8f797eaa9ea587fe45af45217fb7c649201b4013186a19be626c34c735369abc6f3331c3e8ccae5c768f6f589c07d02934c54fa8dabce197249071ca9960fd2c0106f3f79220141b9450a52acfd47f04b42004edb218f23f1193df95d82031a8663c7d6282b4fca88226038b2e3d07b800f6ae66d5910b47c345de117f4c8173e0399f78ad78d3e33e9efaa165d40a3868f1e7c56a261ce942932479581095cbf8f94a387b9aabb31fbaa731182655811a07f9d92a9d892f7e13b842795119d67590c661edbf7a1400a3c3f81c589e47ef75b8fe17a288ad15961aca2e4b25f75d8057d2f0d80c6dd54ec4e6e8ca5d95e02c40701d780671486dacbe16e5fcea280a23aa3706cf19cf7b7843d138db44471ba6a0d3b16fe33bc0964b909392d25ac878886bdb4e2310042214cfa770669db6aeb6a9a620d789a25d6deb96f924154e74a8cddeb072ae939fcd79e674cb4583262bfbe13cdb89fadaf27739873e6c6402064f58261187bbf3551b43e4b3f935b5621423e4df79abd5a5d60eb393ec34ae684f015588fe1c50345e8ec49073978449b00a198e658c602186b388e1bc108cd022574f4220ec3f6aa15f2191c55288b2e63e64d104b62461ff3b084426dc7648e2e847251dca4ea073bc808c160eb3a156c9b6c1a091d6144bc71616840b0260638ffc024d3545c280ea7169fac15e380005ca6fa17fac91ea8edf2649aed5c285efcf5f26a92867ec77b157d60ed4e699368d18105fb7bb5dec1c13477ef7ebb8246b6bb28e7a4bf1e1ef4260bdb36c7c95e408409f71b39624126cdc72c0cdf4eef04ee7a5dc98530d4e5227cc37bd971d6f09a66e61e01f49dde84f41b6dc22bb4dbc0360f6310345eb5f2b3b53411688f9f77f5b577ce529677094566db85e2523c3b46be63355ab34ea384fd0ed2ff147342f73dd4ee185ce67fa3682415a68eee6344f34d275744f0d33c200a33fd612d990a73db93d14f3ff77a40a57175b8ee0f6265b3efc621af73ac3fddd63e88a29957c54663325dc2d4bdd4bd8dae17e118dc1b9ed7450a96a62ba968aefc1c8d6af51b425e1435689c7ecc2fba5e7303a569dd60af1cd7980e4035453f8cab703b205a7ba0b032ed2738645a308b7aaeaf6161f2ca06907de720b68d6875b846bc58f0db4aecc30bfbfb49380226f5d842d23fb16b49368aaab82f79c183bf303f7be08f26124247f4170e655bd2205219dded958369fdefb231ccd9eddef21ea93bacaf3a6056c0b305d0bc0ec0bebcfee0ec8bb39fcb501e0175d970ecdce3d588a768257e73b9b30c346ad95db4f1d59c65bfe78ff266772229aa30279af8e3ba47dfe34f68f3a0c67d2e41d73bfb1c7178cf576741655f2ac31bd649979c38a46bbf5ad4321b8af71c774c27129c36ca4ee3fb068548c0f69f20e2a8fc450ef471ac708c7e20dd98a822744fbb48a40f994296a3594f6c094648aebe2ebb2720eb11435eacef61fbad8de57a722376aa3f6f1250a7aa7d31ac1223e419bef51925bdc3e169f12af5589c47e4c6347a46d3727b46bd3d4fef2e3eb47069009285d26b00d28a2d7f7443852f91779a55404c68f0372343740b040074d2e1858e2b3e88e9b538aa28d2e91e20c0d2e0aaefbe354e573664efe425f2c86335f548cfb0e68596032eedab8a39124c0c0976253ad29ccc62ceb08c77ee1211a35d4c1a8025617c78daed5fd55b84638b7a8955e9c9871e96189fcf7e9a5b63aaf8f88e9f0ada2afa5b66594508e3da2e85dffc269a4af7b05a97e7a93becc51e82ec8771f2ea9ce5e88d96e409ca09cea2053b443a6925d732e0e77ea9b7d481b0ef0f2b83b5112fb1221c4d7a09f14fb0524edd5ba7e600bfd05124ddb82b413deec56cc4ff2e277f49f7d698efd673dfc706df23057594919c99716155089004cd728e6aab43d0d9d55fb8ab9185d4b7dc6c317bdd6246d0579d2f128bf7defab3fcd81a9fbeb34b742b3ee0aafae72efad8e8f1636e9e8202b0f73f57fa67cea68879b896d1f94b0a660e69c55055889700d5fce79de21d03b591ebbf93c9991bf6fbd8aa9473b4c9c892034df72f301a499860aab96f29abcb691e33ae727e36cce19ede0bebb120df95350d059e93ba069fb15d11172054fb554ad1cc9f448b4da592b57b9d556d7a4eafe3692ea5fd6fc00381cf7b81557d16c68a3d550beaa630db57617e2427d747dc39c6f70ccb11a2c0afe5a4d99471c08ecc937151db8478a1d14a02872ad2a81e1eee9f733a196291b9bff282f1f83c9c098c77a3b14d47baab770c127aaa612c7d2d273bb433ea20de2c64b57d53439878099f3a7c1bd178d5d1b5531ca3f2995fe9f87f2f2a6ec387354aea64e81ea5f503ab4c98754e0822fdcbb636a2f4480d9f7d05bc5aa0a3f5ac202dc8455c46ef30d6e3a7175bef0071f608a23a15580c8ea2cdec57ba86e2cad5f960ccb630846c39760284b0686432ff3573163071200f99f9ea9fa8e119b67313aa42971e7ba05024c6a99e3bc3f98f46ce62b07d1a1412defe4ae539c2b2c80eded70390c07ebac7836bb11493b407ebb4cf0300f392bc19d480bd5d368b6b2e53cb2b3499cbe8aa5490ac5d586cbcb993e162152bd26137d82d5e0a219459c848a1687c97409724999797057987976ca797d131b105d073a53a180aecc59596d4c1ed0810c891809bccfc9175d42ddf5992bf3aa3c3d5bb996a39b16f264aa521093a34a6bf0e7e92d73c415362e5797273852f58e51e89b5ca65fe0ca532df64c6e2ac0da5fb86a45ee37b24ff73d1cb48c2a56d0d2a7b9b16483fc8f134fb533341d4c30ac134ea59ab4c432d803a4c8e3145ef03c374fb07b44edbe5043d18e74b84ac0e38069daec37bb59fb06b38155d05dfe9df66e1eaa0d5d5e8ebd61cb51f3d8e21d4e3660814b0529675f27acf0f105777a91746cf129aa858d48a6b79007f04192924039e0de120f2da19eb57bbb9bf54b25284f290fa85b30c34555d766bd0825b7327bc15fafb957d0638bab68bdeb36743f79e240cb6cdca82c4c4e710cdba09ac9ee9b55cee363fded32a3ce1b8918ded1ac29c8ce89252209a303855319bcbb27e3accf86b0427e8306cdf48808f8bac7f4a3b2ae835b6b1516b68ce43533333619dedf17ac43fbe23448e1a56714150f8d1166eeb5ff63c7d6df45341dc4fef14449b1a12ef5070df23bd3bddcb2e7961efccd4b76f8a7d5b6dc004751b0f07b45ff4e47a8d24838931fd26c9773ce69d25159f6a645bf0b30fce75e92ee8b59b58e286d128992723c908c9dbbb45c5eb8985ce9302e59d1f324da7d4075c7f3afa92f3728098082ce1e68956af2fbe764804a1d6463e79389d2f42935579f60fbb72e71dd7d51e216adfbb610f590cdfba5c364a6ca6a52809581b6584a7190679622e6cc03e67e6b7c33a20a97c3045645bc3e4f241c8fb6a17c11d0ac10e2ab6a678bb224ac769a4bd82c434a7d7ff7bd92441508900ba99689e6f383c7e79d490d3f857820c02f0bbe640122488a37469febb5aeee4d921a69ca3af449894d5762f8e585d49cc9c2974bc5075bfaeb9874170df35030f7d9c1811ea23c422fa078918ebfd06b7fda636c0e0cd9ccff8ca6e0e941a024504bdb71b7619119b75fb1e43be963dcbf777bc432b8fa96df40d7d2ba6185e0d740e06a9e295aebc6457aeb784b0d9bbf99e03bcc84aba5d4e84a5aed0920053f5d19028c7edca194076242bf09d3eb6f91dc310810e5d99d10e0c79c4f2dcec3bffdc43c9d9b4753cfcd016aa18a6c18ce2f87cf91923aca3753e147e2745dd64a14137ec118701818a9a2d7791b3556a97876eb4d745ee48dc3b17555efc0ab591cd655340dc5a0c86cd31915f60e1fb532d436bbac3119cef3dac4092d453e748c66a2ac51ae695d46312d7fd5f8ccc18a2fc5236efdfe9149c1e9ea56e3b4b280d0cd1d66eba34b203cb6701bfe1f43e9954796bf181e5377df7255af0357398b825b3ddd1371d88c38a8f9009ae10c9833726930ed2e6a763cb3b8f4b6d81427b56091a4abd7295fb84bf563a043959955ac3b77ae0ac0a3148d4975b719561c4e8b771e1844a68e2ae3321e8951b21de7103825116cdc2e195738f4aa360e0c8a2a9f3aa185b530171867c7e17d234155b1fda40c99af479ac722b4709c41919f4c0e39d6b68ea8be988890f70e07c21404de0185fbf1308c91536e22fcadeb062017d777196c45494236f1d2d2db86230998ab0a25f26864604114d4635a1299d711e09b309783504a6b6e2ba3a75651f04e529e72a6ce20af009f661a020c0ef305d80ac0af90dc4e4413113af5de69893ab773d29b2784bfc9174fee41f2a6e32bc7eb5412158281b630b9520afc3aa8e624589e17770a6fd82c4e869316492511e1eb47f74d180aa9363e00092da49c93f7c47afff09b36c30c23342db816723e8d33170d8063a82a91d353a97d6bd3492c2106e9eb0b1da23275a1bb7c50f60b95aa45a7a55e5368fc24bd69f93a1c8209b5c29a66249e17ff177dc78a8866619686090c72d9e3230736f2147d2607adc263f69a1e6e1a08e979f4633d434a3c2a0d14b5076d9d36ba48684e7cc622e09910eca1a54fe9cefd51f0b396e85cf67d7db9115606750332c8ba3a7173ce38ee02ea1bdca78cae52729041cfd3e68c56d5aecca57f0303c3235aa92eb7d662da4962d9052b0f40c7c7f89d7f84a2dc36bf5d0e475a920edf5a9ec30dd9b4312ea26b8f691b7684422079faa9ef3e0b9f3eb212c9d7485f53a114b6047cfdc9172c53e35b8a43ebdf8f36f02080794ad36c8d9cb251bf8c3e292d6648f767ab44aca127a1595086bf54ff1258970683d2a73148965007d421ec346746ae9d6d8e95a6d51ed144876be3e61fdd9a1119e8d15a11db906e8732e8a0a86d515b00e14c6bca04264211b6a5f218bd288399bc95b123f580964fccf694306329fa82e66e8a03a2d6ba0156e3a9194c590aaba9f5adf3ea558e2c970b94d702bd3ad698a3273421f95bbc25c6dbf7d4626d442932b79b0ec8b089355eb82d92e0d3a40a85b07ec05a045819291ac0d5eff61dc95ee6a98e1d57f29df637027621b8edf2208fc3f5505bbb4e29403abfae65712bec4fb3affd56c19f1cbee44c0adf4b64d2bedb09ae9b39daac3c614aff7e0a2c89ba3e22e7864b2caf3a3f1fca59dec7c20a6456efb32d0e40574a70387207a1e0fd05dfeee31312f907a4ecf72a8a2b99bc0ceb8690d7cd5d5a299e2f12a93529081714faf10a99a26eb499c606ea5a7e85c7c32a7056f8a954e286190bec00bc6e134e6400b0db3bb102af7b58fa81351d1f61887051611be64d7cd48d26a7a4300410ef7f75cbc05af6a1376bdfa2a531472741ed2fb220d7a502f73b2d28719fe5184c2afb411f30b2d4ba90df9f443e1e0113560be2b5cee9db1e6b670e03b03964ce3345394e9ac760d06a51a9b2a7a7353d4805c3184adcb9671a9aeb114963672791f2162148c3c2696fd413dd4392afa28ac0c6fbf99311eb6fb92ffa94d3a557b753af33b02d17ec16e5ab91e5f12150124bbe5b6de9f749850fd93500305d4bff4c62b1a0e8e0edc281ed26a4a36aab0ba224b07028833e70820f65f2a25a34beb7d1ae663e3936bb58d0135be36ce8aa46b9310645763f6322336358794e9d57e7436db96ed1b5653b5511fc07a183f74f2e585a8398edab978a923eb56ec4b94ea557573033d2da993c9bd631c7c404650ed15f11ee8ddcfae264936464738d9cb18e6c1c111a15b15ec8deefcb6878bb90bd94b031f06cc595f5436e3fa9d677348ea94edcab9b5580ae19d954070a576fabd37161082d7ebbd83b157c4465fe31ea89e4931b2bf96766ee7d4a4a81b837aea6cce8b141c716ae870b5de3e7bd23df7997134ac2291ecedbd9e9881ee567620475b0b8249fb11ec4b4ab563782a39ee86ba46e12018cae1cffc4ea4af34ed05383740d92785d10a1b77582b6212e4f7765b9b78d17d2644327cdead887720f5659a8489ccc9ecd64a32c4e56674245f58632d949b7630928ca4f81b9043794a86e4d85e75b7741d816ddbdc1ea305f552ac0a61ea62992b73954c799d701c1747ecaa04cffab896f72f05945ea18091f309b113f4280dcf6e58df68e950b1b2be00c8a5116e66e317d344fa5d44c71e98242fb53020a0a3d102bb582afbe14a0c6efded44d5e660c494a970e3579f14da69f7ae492c7c4be592142de65578cf5ed62a651b92c46177e8a8486213749c1c29dcffe5cd12cabfce1f8808bd839c244e5f12ae7186c44b3641f65204c2ba9f544e13403c53d34465d0547eca28110905b8bec6db79495a5722f6b1c4a7a4b09235938d45b8089fad85524892f992b8afab85e148c16573864e6bbc8589a3f059c15d5cde3b17ecc42696e6593d020e101b1bfdc0d35aeb579105eb4b48b8b95564ffb5478268e8c7717852c4e2e0d5ef53d2a725fcdb011892a8113a1dc71b94c79bf87deebaa7648b58996d4c84a44119936c62910a3fee9e1eb5c536aff0dc64ed395c99b46155de20fbf2d6ae1e959c4322479d98fad23bf98bf4ef303b831d8153132523529a085f72eb6fc3c6995d05e7284b7ee66b7ab520eb123a68a7d3b8a005eb242e6b1599f5149f086cbfa16affcc0ce7416edbfd9ab4fa112443d7cb082d43dc3b927df09301901d122a3dc1b5e2e02d6d1d5ae5ccd8530019fe3ae7b86713cebc5be09769aa04ce538d794e2e29a75e2482d0aa51b223c71073fa1f37f2585391599c8ff040fc4d7dc1985d986993a1a82c5124276115be7359ec812a72256291d2d643531bb02fd51148fe5aab4c38d395b79a375e78552cb57e7efda05a29a866c901166866746fd94f445e2a70cb01484d7b972794f7852c012dae7ab7e9b93c63e1f9cee8698ca29e3c1eb91389ea52b388fc3603d9a5b48b595aa29db6ecbf65c585d72745d1982b95d930020656f2431017e2e1dfe726ba2bd92c13bcb75267ba30c7d7c4ff863be995cf725f4c57abffeebb63252d8d686d2a195b4f355686e7e965b98f3e0b6f81a9db9f26196e3a20a70d0e5d45565cef23d6bc281a5340d808714d5aaafce3a284057a1c264bd92a75037c36348698d24e06ced415ebc77cf54a0a3dfb785732ca4cd6ea1a66212b33359d31b68f40451beef54115b8f17470a5e47bd7549ac71945e59d785bbb967051e3713c9fd1e6df6e2044f1ac631f67f7400dae1bb37d4e8dfdf297e8256c6267ffa7bcbedf37667d774cb0e7b2d206f0d93f8a822491bdd3385ffd3bbe59462bab0c8b5f394c88a77fba7b2b32b643042e3b0b8688470a674cc9102805c4ad7c8c7b2a2b84fca17432af891fe18bb48020124627d175f0dd88dc456a895bf5c64daaed1310fd592ea4e86ab1b3d0c31739669a735deee4d09d8cc86b1aeb6ffdc611024f44d5799c70664562b3106c2a8a2a98d90de4bccea52b4c381d67318538456a8dc0a962b652a7ee16804219ebd5185a36850074e26407ce0784d1d98d425398f911e68567e848fc32cc3f161ebfdcf0c9f10afac77f19550c33b24f1504206769173144deab6ea84feb32345f9ce3dc1346a9211423b0e43c47cf29958bab49a5bde775c9930b568b2278a2e7cf56ad73bdff9120a3e178387fd08cad926a8ff06241085a3c170e2cbe8d2e03d00504fe39d6a79670b5e0fd1c31e29bc91b11459857049f57227a77a924314cf60b9736e7a007335a18e10e277713f7e81719fa7e5e3a6a4bef4e1f15ceaf732b61c8ed5b380ff0b20a3e46f9db0fc62602c379714565c442486da7dc4b67ae5659f127c9184c1b1dcdf5c9557e158d7cde4fc0003b89b717ac918563b04918085060da71f12346c4b6d3218af6a9db8ffda7d0a2eed2c409fb98ea91c3c379aa17d65bc5dd1a8682f0a371edc9790fe2c4ae78a902c62ecdbaffa1d57f0f0df68a6f69a9057d72d19b496dbac495955e5bd98adb28557ae250f0a6bea1669f244f44c9e816c7fc4b60fa04d16408a88db0b7c25991deb9077eb6d1aaa3383c4876ff2705e8085457c5f6cd4519fae8fc32c04783028bad11ac9c95cc9ce9917ed72a0f1ec21b5340d8e7668b441c80de97b8ec51f4b7ef53c90ff13567041917670ec7571061a3d4e287bf09b68d89a2357244cd16a94e620ddd308be852a6da3cc1138273d3c4b2fe66b4be8391b4c27a6de890940b2ee029d4044ad018eb201e5cc68b311d9912d3699f8a6a480b7b110777611357283ac357189428505877e532acf143c81223022afdbb648ef00ae808d38e162f0e32b3624d5da9f5bd9ee0a7a3610ff11023f566d484215ac8349608d3cfe2f504131a1cc173b99af32997471b079ba4899ff9331c0c2f55cdb7a3c8b847998e107c1a13477aa1b02a0c9b85a0ab71c9fc6b9fb6babb80aee2f6f680432fe817a2585a9f102cf6bdc1e96533851251528c232b0366ed01d5eafa4d95362ef963b6416ddecb50d1e1d90d0e8970deb4c1a46787eb406c0641fce887314ec1d4e815fc89a15cbfaf2f37888d6d033e8b2348a6fdb371e9b8f4961fa7f0b697ffdeed3eff5b74e460c2660a939d6a212244a9da6b345dee78411b2c2b5c751f43160d9d245f9045ba56423014302e4e93944b4f116cfe997ac1399f07e807c59abf4225562989e92d76c9fc046a73062f16dc7e65082f44d71536fb3fa8edbeee22aa78a8efa0cf0e25d85876ee434b915f74a27881d51bbb317046fb55a6ce0139d3e79bdb87d0323d366a5d4ee529e5bdf155c1ac0922fb71a61426612acb83ee78ca8dc9367966240e097f50dfd8a26fac37c9dc80239a4201c361379a671fa96fca1d662818ce427c3f0521be49d82ca09bc668de4762559dacff75d3f3501b79f9bd8b9aaf8a90e3ca0b59c695d2c7879c404733847e5fac9f85b87592d09141bc8829fb21b1c5d58f038d81a3e24d6e00ab869d50d8e0ac6e8b355af8413a8c31425bfb29ad6835bde87ea149f6127a0ab0c4787e7e16ab4c610afa175b59ab71395cdfb1b1878902f18596cdc500b8b726ddb1bb1ced9d45ec3fe151a3c03195db734a8518c77e2262c7a5a5b57810d022e3b298dc4a760c173b3a251f2dbf759091e7796a059bb4ba8d7b32be7ab56ee026942f6ce3ba962815993754e75f2b7c6dfaf5b8d3cac5421c2b7c2b0ba2dc4efab9f63739ab74b8f3b1a42236765e1a73ee818b434ff19aa6e59df1456c31b009d10718ea43e1c92efe62c12ef761414328ec5728ff31a6aacfe0adbaa0a20a0d3a54accaf2d0206d81d3c3414d2abc03040293be7553cec73fdb3a37da4c229e8697028d857e261184fb3fb3c4d52bad2b08c560bf2b1a6ba0ebc8fc5c907784ea5f46b46bc6e476b3b3f51407936c50fb5125eb79a0a6505ac9d65f8bd34e2a14f772c828068c9861126b089e9eb2c866584a1cdff474dd819a5f4e619f5788430ee56e0276959020ea80439dcc39ca296064bc2a4149e3cd3672c97a091d413bfbb834d59a827edfc8088233a962046a337a4f2f34f43adb2132a433f11e71ddd4912ad2a09fc1fd47153031af5c13c99f3c2278a85abd6a8b0a9096e846392aa170e44ab691567e746239e344f789a452c6f67b9aecdf651661158796e30e1736f5767e9c26b0c95aa9fc987b4c7ec43ba8dbfd49b3e9766fe1b2967acc3899b89fc1e792268fe9044fa63ef6542c4e9e65720ba327de740b2e108664ff6882407a17080eac0e5ea93578537dc07a8171aa81983b722c57f5438832ca34317577a74ceb96920bc35903e17722739db2ab4f3cd60dfb9a9be8fd689d84a02bd5d1dd0c574770a78348aa7411c68cb3ed78fe3f4ad27bffa10f955e38d1d328f8726a9cbd4e68ca6af5dc6e6fdd6c19e3c000b50818fe94c99ea099c1c7b7ab6cdb21adce15e4d6c0992c22850d1dbbe9eaba068f3e3b819b384d1efd097235b1d030b6b0a2fb4dddcf92f349fce337eec4521d9d63dbabaff5309dc27353fbe472de43c5cf018113eb283a96b0a3d631171ef6e792880e2cd98c35e2573e87061ce63fe6a34d8f950d9c7bfc3473d501d0903d3eede2906c92479992f087be02f2b21cdba0e815eface19b7c92abcd64872bb895187886ebf33561d724530c4954a9f4dfc0473dd84cd17c5eb2602869e25eed5bd14f89033cb11a7ebeae526a47a6356c87ddec4db3446cae3a0f0baea0259df09827f79b6265ba3beb732812a254affc1786f3e4b0b80807f33960b83c98d159f7583f55e8892b66e44716a002cf3e1514d09f42ffa18187f0c647c67aba10ecfa558bd3bf0e050dda2f89a5009822869ad0674dd84417c0913387075b6ec384b29f1ab87f64c5de0a23942bc17d1fa9a98110ce29a027d708c256fc39d0efdb896e5004306d4d4ea0de7e5f4f6cdc0c4396fedf988f5f95fe1825da972ad3a78d785055d5afe4b238da257003006772cdaa1d86c1c39da63c93205187bb8756c4c4a02e3cf681e2c2e5041aecd66e01365397c0a0d2b4b3273e3295eb3102566d1ebb57d30770c649d43f53bab57537104cedaefeecba0edacd8a75c30f1bc806cd920b48fcf623444ce77988e591e06d24e4df1e9790ce2b64139170a845991c1889a4d1f6da1740895f83bc182ade909f85643a131f096d9d88eeb4fa85debf156115abc88cc8108806ecf5538acc858e4228320b9362aa7d7b0168c133fe9a01d190eefc8611b4af2f3fe6ee0c078a23cd5ad32be6e9cbe65fc340eb0afca8b44a91b788759212baca68912b04a12565842bcaa3cbe846e7b890d515a5780fdb682df2e068bb07dcfe7fe3106a25b4173600d51b5686a13ef0662c1bc5307c5cec8306a7a2fc0bc013540b4fafc5716ae4042f937033fa5c2edbf634e68a0422f7b0f570c2af446bbf208e23c95ce1dd91b20de89aa3746b84b0c2a089cde45092224a92d144a36876a7fdcdcaec018f99d61662f7cbf87b6c3b9a0a4b2049dc561d78b966a88f674c5e473918f46c6428572496d108bdae982ebad10a7bb9ed57209737c089f1d4f2ccf6532e97e290cf4d57a67aa3cd4a071bea20298be24c847f56a22c040ccd8a2ce63c740c60b25eb219703ea9a0b597edf0192cf2602414be3b7c486a473f95508ef6db1b2efe0eedd8a42e8a8187deae0205d13edb86109a14a8eea5d8b665c4728b811f2b33c38b599180968b6d126ae6c5a99b3c23912d5acf382358ce9f43c9972010773c0cd11e9fbace987f828ac848877cd72f021a8f78507ae6fa5d579c5e5e8253c225ae26d5d62ad04274e71cd9dfb099d2a74429298e4ef1d3b5157864225995cf9c759e1aa5c1d58f428b6a210fde5cf6f9552e778fbd98168f3e567930b0833e1ca9e613ef6b3ad39aec8364b96d39c4cd2968fccbbb36af92254258a8235407f94cb2c6aac9aad979510299e1b2f8cbb3a54cff0755285b94334e173a42810ff0c0f089942cba448a00dc4a9f468857042e3286f05741510064666ff21e5aca57e6f4a8ea2533e61416a76bd9fd12abfa68848faf84194fbdf94a10d9e279572bd16e4bf2a27076f60e48eed494fc6a39b41f099f6e69a60c65a9ff798132a65ebefeafe40ffa057a3239b01c85e58526d30d420ab3d40cf5f20c3eaa3a45ae8e94727675ffbcbb01abf88f63595d591cfc57e3e93e290b4496b68c9b09875246186c768d59c122608a9ab5a8b1f69a7fc7e744db4c54e93f75019bce8ec9f99d51fd34787f5fd25dcf678a2fe4da8fe3e3678053a4203f1be7d8754578544af06cc2c3ef471658a1b7d6450f5875ef3996da582f6ecaae7f004155f05139f5c735394af50948227e96be58a1c07ec3668daf5c2f8384ebbb9eb35d257126d1109a9cc265200589e0a5c734ed27cf386cf5d3ab8a229008ebdde8377e4d7e75c4b2f4b7abd45c2823192d5edd39103b9dc846246b1085be1a21f4edd755764043191262a1c2252ed3e9b06adf255cd0e948ed82edeaaaf6fb311f6fd7cf9ae22c283c3548ba9799da21b336644315959f3f97cddec9c6c20a643ae6710604b56088439a7794cc140ee917e2f4138b6ec7b741f45f72ea3db93439c8bbf7095c07eed2c7dd9f569c1f5a43255e1b4b46279a01120891eb0f67f718b6fbc1e17a5425d8bce6f3c36d2c39ddbf13e2cdcd4a6564f15716310feb0263404ef86c162d56b9ea91506f2b8bdc8d5439d7939a35717f94036f27532413222f4104cb9621a0db7f7d2a104d3b1131244b208a21be30f31bfcd6361e79cd94c699a465eae78c90140b26c03beec5ce85a50fcf4eff0e9965d75fb28da4b4c8fb46f991aa420fb4e9f80c7dbc05767647d0ca0db97ac422d904dbec60bafd9e92294c7d40a299dde3d5abebbefbff13d3247f599fb224ffdde4ec3d24f9b4c13b13062e83919589ec4f8745a3bb7027913beb04b47f13da3aeb3328e4fdf75546fb3daabf5dc11f921e89b1360c1b2bb8838ac34a002aa0155a03a4a1bbdd8b719af446e3109d01f610d919a9ebe873a9eb046e65bff8f7d18dfd14ade52003e34a69081e3cf04f85581ae2f95bed71ab8acd069ad382062945ab821b0bfa04c9e2598124e2cb5cc5f5a918b1a9c8a9fe40466f1adc5f3175222c20f3754be66eee98ddc5ff5482d2b5e775acc80f6682df10f262b68da81fbeb0fc30b2f067a596f8658a58abd5d9c6b522c6150ce87c962556daa5c931625d015b817b902d4295a19cc14c2c4eda69b26fbe040a11347bafd400e1bba608a0d084d1c6a89450cf89ce84a123790b48e1a3eb95102a04cfd2290379669e85137bc5e063d64550c807c3cdec127d4bc9908d5bc212671cab3fef8b1f2b9320af50a0ef20c9b5bb13a4020e979f4d249cc6d6b537e84311ccf5de096bd2bfccd2ed4d4e7aabf1879566d23473dcdc8cd3e3b6589a1cebd60dc63ff38813784521a74a2042ea7b30643d8af7c434dc9a2d9c5a72751ead62c04f1781ca5f1c8f342638b4ec9dca06b1dd485c4948413514ca3ce90f484f86accbce1262ba3a6b82d403905d3d5b98d403eda1d6363f66fd7e3de764e2b732e140d0b1be84fe20e9c11eec88b5cb71f209a187ac6d37ccb97448430b1c03a92a6c216f8fe05368b5e62bd69d1ddd54e3a5b79364cc28e499e5bcc68be5a7ac345f7e9af08d24f65289744dd7d530b32e7c2e79828677a618fffcc858a4664bc372e9d0f0ef424ff769fb8277a9d6d59016f1c3aaa9b9a2f8c551f47ba54cbc08604aba8b63be6164c82d8f17697b070325018f090863e6f6afe913f1e9547dbfc4d1065c9f8d32957573b194a13a0f89244dba9d8edf8601915b1503f9ed31c58555bf473e33a8b728bf48e259e0dc9d38b047266e60c46d31fb2d54d2bd6efe25dd7be2051a8d279758f15db6ec885876b18a529e685e51c60e1ccb697d7e43a191ad97afe79ae87e008de9a42b6afd44796c5966fd373b78996f2cc03b75e4ecb4eaa43a9ccdcea2534a9a842b1d6424b416b2a99cd975b66aab4b91a2c245dcb8264847576b6e00d96b78f73ece66e165dabc0cab76a66890ba48a21a984f8dc0c2744eadc2f1a5b40e4e594647205c11445425b4aed875543f1f696e0585251d04f6838f533c2c3e6205671618273b36dad4af054c07b76192bf131852132c92712baca383b8645abe557978e4804b56335a668b854d2ea8d2c150774f4f25f9c225169878831cbed551523675026a2538d0186d6ca1e3b84db6ccc88310da2f0788d79de024ed6c6c363bcc7b8bdf4b672727a48546d2db51e0907c5cd3f12c0d889252da3f09e8de972efa27529a1acd61301931ae64eab7faf7619feffba14db2f1a26671b5ff048c31d83857ca8b17ad3928a9f87762356c66e2e15768ab6343dbbf303e5657d0388b66ea2728ab68db9d2a650cfce5f18346b4b6c7e66a15b6ba32291954a8ff1f965da1c4940ff4f91903c9ff5f3055e0758153b83f3ff4b284f5c559d3ebebaf716bd15347353c74c355cbd499c6cc0919574eaa7a42a66ee270447d16d2db1f97b7310abb1e051961242a082ab3b88d5e168681864907411d32fb3a539a2cdcdfed3f81877b380637c768a1bffa39c08a3ef96cf3a2e69b9f3d582ec711bf84c7d55febe694adcfd6d7838e6653b7d53befe3392f402f533bb4b16fd6961fc680672b6e58353ad930e5d111215590089d8df2fb4acf687e14f1a16b769fd0ebadbe53bb0457ad06957819d34602c27a44a90315d89bb3f6e2c3a0fbb8ab5cfbfb5380f0e105ad06128af36f37b79455276c054a810be9b29111dd1c97973c589dedc3913251b361b0633e527bb313236347e5cbdf44ff812bdde3783d55d86079bad8f881e84403217daca8b87b4f5aa1807d3e9d0e991ed045b10726406123ec8bb40526fc1d792c7bf71e30b90fb083618b751249087ec9cccbc1e7c2918b9504d43e7363b8740ee33e93c022f530cbf969248d1b238a3301a957b27abb6cce28df77c8f186101d53e48e168a2ecc1e142878ff8c4ee7554e39ddf6ec1dcf717b2b80ff8488ff64878fcdefd648ed5cecb27e3884f8da5127076ef6fde72d3475d43de498b720352a4f3baf4e650888061878a2f4e81799205710d900df51e578e93c22646990a7b83f926f98d3998e8d36c64b910cbdfca78caab8db418dac5b320755171e5cca04a9b62d1ad84ccc167bf57075bc7548dcded47fe3e3f11cb1538b71e27a9c4869c275716ac82c1affd826acfe0f7c820142bbc6f04c28e55d48e57b77ace0eb7e60756ff588f5d11b3788ed096f569ac5b9f393de4e696ca5b6017e3f2ca635be7303a685e8be3ce2f9d4251f559f9034b83f4fd042df2d51b353a3c4c0d711caa8a7f7a780023b4080ce9cefbf970628feef77b85cf28d6c9a8323da50133851491991acd75752eb65e3db63a801368297558d30de890b91d03d6ce59568cddc3e3ec53b659a82bf66d3717a83fb9ac11ed497a7fc4489199f5578bbc55fa7e9e846f3d38d7276929b5cb418aeaa9bd10e921aae638578870e69f6c9448ff8d427a1f6fac25ba836cdadbf2c86fc264dd324e628bf1bf4455f4a2c586c6db9fb45160a45621b05a02c8890583db17cec2e88b6f95f3f07ff666ebfb0db3ff4606b60bede2fb33fdbd1b7d5d4f6f6472fddbe8db7f7c3be2731e9abceb3c20914573a6a7aa0db2feca67a889e1b4647dbc7835a736646e875d8151d2cf4a8a1cfbd27d38032a8681004be0d08a58378e1ec416edb1685d07a9914e91b9544a2b36ce66c4525fb725e41a74348c7b45e5b09f047676ccdb0259d1249a94bbb8748ff8151b6b10d0e2d8e6908c8ceb46031445c800a9d6c9feab14eb01e79b30b34902e3a301ea2ce0a69f81718f3b656c7fb16ac026df73f01a7f5d238f8bb4c13328fc2154e1af794747d6b839a06e159b339212a40acd6a63e0a178e3c67a86a19c84a013721f8c25c158b42f1315a5f99f869a603375d4888d1feac81a25e3fa314f495f5b19038adfe98290a71190990f624c5afa4a86dc92c2a94ab79d05484e9fb0b435de95adfea85253ad45828af3304857dd1d32edfcf3efd4b1328ec19bc7f7e3ecad099daa292c6195292e6c7703268e7e56d518c5a1890b95368ca33d7723ab7504fcdb4bff8da3bc845bd2887ff03bd76a831c7670cdac9a643449203d2c8a2142307a68fecadf192e85f7e87c987bf774183fea3e42f173fb43c27b02fb02f1e40327395277f4d28a25ff3a9c532c4cc89b3f8e56064a373f4603237a68f0c8349e628a2ef600a0d1d4af5b5e92c6e1e4b97dd5847c34f12ed2b4965feb6e3509fdc7663f1786ab6135e62e4ebf1bfcf10cfb33b8c3bcf5a8b830f0063a3be48481fa847121b0bc8dfc01adb219b52fde5b807c2664f369940faae6a4d78f262de0b6a12a115e9e9902128799130b40e957e704443481700abb24f1a20488874b937ab0c275ad745b399640b51721d345c4c6a58e502134c706d10641bfa693634a834beb94a60bc8f103d25410415e2b0cd0659159034869d2ce62d7fdc8db731c042003b149634272c87395d7672ce57c0a4c515bd9804c78ba204c7b625db2d0143213e39a7e1dffae4e8d77b63945969f726f0485519781d6f672be0552a873fd4b1d1ccc75b0eadbebfa11c7a58102a2bbdcdb1e63a5b0d1da99ad792289d50afdd663a93d937e6fcf4573720d412734094d155e8aa8f40cf0e823b78ad291e553d787e3afe81a54746679f3b519ca5d2a82316e21526bbc70da482b56a01dbe615db680e8df230a35c90573c5f45da7716cd5ee02022019d11f300e30d17dd7acc0e1c54d290756c5f9291e367e2277a478e041165890ecc9426c0c03791ae1339ad6251f7c92a2d17c9e297657815ef0e97e274821d2b50eaa5f9f2d0d67d450b2b9234ac96b270d9420348206252a789366e09ad34813a5c6045538629777b0be8d28c242605981b5fa1f190388e028cf6d9aea54c29ac06b8fdbb15606f50b6f271d9d76b5d3e1f6f50e0832e279e93a3a2bb42a95ca7c960dad5d4bfc28dd83b71d5ef73b61c2ab423997d3178ed6b12e19a73641a15517390e89b45a2746bdc1c0471af235a2297edcc570b93a034fd2853f6d9ff3c08a14940212b091a182ee25e01f03428d3f643d39afe36fac0e8a547de36ee6c6701e678f7be7a02765ac8453dd7a1721bff82c65632b8440c7e286c22ae7ba8f3801aa6fa8625eb4f4b88b8b1a5e74dbc7a452d124954a1becb042893aaad339b648d4097a87131ca65dc019d1a6c2f0a1cf855455f3c4a17d69d2c42e6db35b04586852af16ab60ccb55aa3fce6130ab016ffd07164e43195c2ea6f89e03c686a5234161dbb08fec9f9b4517a113490a070be7cccfc7b0637a57005e8344b3193210140f0fc80bc2ec41ef51b9363f1c5ea9df4d2d878415ee98948355813ad0f0991395637bc21eae84d142b51ce89776e319ebb5c05607d1cd17e10c61292c64b67f6cd63c6f9ef703bda17be03edd84b48479a30ac12f0e9008c09b82a50d7c3487120ad421634c27fe61e6aff6092e693890c4d3cbf987fc878b7385933ea78ebf85514ae500a8c88f116a0ed45d14eb9f1132e4f85bb1e350783b4ceab034f8b7373089bf4a71294c0df8c2c5332e0800fa930400af4e38e3973ae8e99b1eb7b6a1013418c8f34b4d7ce8d59c2a8736226bf343c8683502e1aa6f050dcc5a711b9fdbd7beed5871c84a8adb4d13e1b85992754a89b2cdb6f226c69595197447565b9c84641ad8f0403f3c8827e3534147d22681027c11219fe5aa87ea5d828ba1b1ca799a2a1455797db48daf3481f7aa23fc15c061fbe39748cb2063a668271864910b983a9febf74a3e55b86e0e4962366d96df5c28e5980aaefb3fcd772333e84f0674b5eec20f69e5ae830f7a4d59b843dd218c50ba0056117636ae84b545088bf709ce45ab1a82af2fe5c5cbe80495dce8213516d6c6e36fbdbc9ccfb70d0200e48d0d891d4ea96eaa1a1bb82319a051baf0aca50c5dad8a70901f504545c2f4e06aeca5fc6c5c7f227e02b840ee84c5999abb288676ff4ab80765ce0fd9d176e1cf186eceb5a6f0ce09d2fa2000603ce7d16ebf3971b7e13041aba3463e944af34efcb8b63a6982a0ae39ed58504c669c557776dcc35cbc7b7a31456eb50b13cc39424d8a4c0941f6ef5f0faa3e65293fff00261437c0e27319f2b124538c64f5e8f83e1fbba461351cb401c5e770a1c3bce1828f283d08541f452cc19711da17e9781bb273a06e9857458ed435f86e9c23d0c23e0e6a6bb3604a9f8e31570fb7e2fe23adb8e89a0fe1bdd67817f3c210c3bb976dde95cd8f1de8c15cbfe804484e4e671744336488b514c40756208764f9a186dcd721a16b13027b04a23e89af0c148505008a7a650feddbedd9913e1351471e4a553d7fb824c6cfecd05d390570816255275e247535cfef75bba2673ea0ab322425ebccd016295185af36e1642075f29ace00bc87bf408995665fe12cdb9327ca807083ea5675ad7751094b931e9aba279ae9ec774513846c5d116ec946335e07f4da97a4d9f5d44338f854eb6790c4c1f8a8459646053415219b823ab6a37a575aacb522de7109757d60a81491c9cc073665021ecff5c23daa2746cf935ac4598ee57a5324441d75d60b42b6fc39059148865efb2c8a76767798c33b42e92333b26f38eb3fe2ea26710cc0de21884e536832d5c5fdfd42bb8913af43b500fbeccf1c5efda9f3aa47400b9f1136bd533b5df7890944ae766743944a9772f72c74718f5f6e6627c62393cebbd3e9ea1a635062cd9409de8e81fb9f11b48e7edc6b93eb406406ecde2b04e56b11bf691fc026a9758ddf1f19fee0a5124efe6ab7e16f7c7c54446d06ebdcc7c9b33dc555003e77cf03feec11b70bdb56733aa6932b5abe7d3164c084fcd104eae9bd13d35e6b6a68340cc4359fba40c09a44fad1fdd28b93155fdfbfd01e33e2e00c00340cbc261880355ad59f1883020bc1ab17e304916d0217a9701675101b549f42f957e57f87cd52b42fd63cd1315cc5954368e2e481a64eb0b5ebb6adfe608a75108364a6dd0b3190ca2ea8395ed7f81797b7d6bbc26d99c6fa058e0d64c2fac476f576a6f7ef97fc7f9cfbdc54d8bc98e222f5a93b0d8cb9e4ccd31aae8339f6717a1deb8e1676798837a1bc4dd50e2076e90f5ba7374d5de4bfbad817ef886899c8ca34e42e23912a9dd671fc5239b606fe89e5d156d3706c5e8efb06c0074bb6c65f233f6cfc67b1ecd6cd00bb76c68881f63f1d53744bbf1d17ae09215e56264f04441636463f5e8beadde8fad2a8e15a2a71e0a7b1e321cbffdbfd6dbef376edc1a9ca3ddeff2429e891c2272eecf683ca8699b2c3495300ab5067feac3226bfb57bbf6d552a96a5bfc5dc001c566e39a1d528b7f67deb4f4eef8b7740124e8057d3ed9b026ad985408424bda56110787e058a1710125d002a01e1ad8815cc7985bde21ed75c9989e25d681fa44be5c0ba42496fa9bd4b18eea26c9c827723b1dc80e1b329689524fba1dec43ad1a53661b80e4f0b1280457820f4244a06efaae4a32bdf3f786c71b149e6148ab63de656abe3ca81adee13e13d890ed2a6d92b3d9d9fd2c68ddd5217afa6d99d884276547ae089f12ad048bbf7ea4d147e050c3ed7f5ae19db932592e0c9bd7f7228d8aaf3903cac1294c6b3783b82ea06327e4aeb1715a3c67bf502fc67619212bf4805364f1a8b7b3f0933f60fc4fd4ea6070b540549e199f7b03641dc07b6075f9344fb7a9b10de6c2d88d30609149bf147d6f670df119dcdb3284467c9cb4c317b882b6148c54f212d7c7e143b87323424ba156815665c243044dfb64c5c1c5ae0399f563602a700ba8f3c7a900275579cdbd16eca5ea2efecdf81504dce7f997daf8b598bb974d45ac88f364d46142b293671e293f7c5b3a95109ece5b6941bcea3d14f817b6ff9b8baa96813f50f0280a8a809f614f640614b1e71ae4d4953a23ab26558543862581c586b7d9e5ef408ef25273d29d58d3e472d5fb8475864069490c13c85512bc6ddb426c45052049a245df35c39b4be0b0e5ca326ef5238760e601f9ef17d40162e27b0897f1c48aae0c2536972271e29d01f93bfbeccdb558ec83a301f8e874ebc93f93190517fb3efa255baa337ffab17c4870c42bcfea06038156e41400a7063e568fddc127b9cc98c89698778304d89148910042385070e0a22fc297939892c1c8246a1fcd665fd033d3e8b5f485d60af800481b31d50a539a7bdc5afdbb299b40b72235dff6e50f7d594231b59d402235a60bcbe5d8d4fccfbf7be739eabeb513f5b6fc4d0fcffd2c4bb5a58e9ad076ef92c60f11bd04165c0e55aa6fd7dcdfede92155552ce14f3fc308eb6570cbbb899d53faf2f912a3d2cba6930c7e04405771426002c0d086185ea7b826977548d25dbfa672f170bcdec99f157aead013db33fcf90682222895d7425bcdc662a098c3906e2b40f4905d14853634d271ec2c18a9a0d423bcea6c4438740c2af21fe7f0564007929eeca64467c83e54a0dc6833a3d6e8f064d25aae7137b57b3de31958970e9e89477858abb9da90bc4402fe72187d9e9c9dbbeba05342862cae7060bd8889f96a8d70343cee55da958aaa8aa6ef5cd1a4c981df325fab655175a2fcd3c43bad3b4ee84f61fb1e324b9e55d8947ff69fa65c79eb3f654454921c6f48a78951641e6290e2e96d938ab592df2f11948fdb5a8b9031460fffe2b471113ebb286e39331b3e242361f239a14b651b50a924a75de399605313af2d4f6395b9b5eb677498c8c7668ce425e86895be3cfc196cdea43e5e2ae828f3e8ab7a135cebac08229dadaab6fe75173aaad327716c6862d2b2e9efda892aa798e4be7fdb1aab57f67ac2385f80b26e87ccd2de0d6c1272eebe08484968ce301f5d88f06559d4af6916a1fd5178f855a677133a2fb659304c04f923b71c7c6f52b749421562147222b7a85a0e8a3567f837f0dd8dd43bb1d891ee8729937cf2547f922850455313217dd2dc6b4b6ed43a96d6e65b05c6ee1e63c76f1e467093967a8cea9244a6759b37bcdd703e4f84993cd7ee3cf5e14eee385f6d48496d6e108d0725580aaf5846f5987cdb44ecee34ee32ad24cdc177ab3831fd00f07b9553d5a81dcb96faa38b04a6845e2560118cd789efeaf4c2e19b75c388c14a0d3df2d21c6815125923da1109c32561bfed9f33344c9a7c589109d4094d76faa67924646d2ee5aceaf833da886cccdd6d16dcb3234d63445964491602ae4b0a0eef718fa5bdd094154324b28f4a4d0b7bb43ecfaad2a9d9ab14d7f915454fd38d63e2dc79784f4e97e8aaf655d982049d7f9356cdf69f3dd1d0020368edac84c9c32c5fb9304d719f8252948f79d310d88ec6c861d7c7d9396678a3aca20431a39b2ef00e047fc45f6fd3fb5322baa1523be6a85e5c5ff45c78f5d115bcfa6cc6fa3e5f81c95ab07605b36b075e9136a4777709db00546f491edef5fb9ed14c71673efd68bad424c0c3c26db986b6d5742bb683a354b837dcf73bda8c27be43fc7bccf0600308e75dcd299399355ac7857f3614b9bcacb4cb4d181ebdc6af781fe73a73b76a663d9bed2702c5c6d65390d30c4f1a0e825677e12070896f3622ccedb7c5da6166bc22fff39d9c7091af783ee5e34cfe2f0cf8750c71a6d03375251b8e0af55d428517bc6274e3b6add0c05a900c125e15f869ff576cff3ac0a6e5a66a36574fe49d3ea58b9292e7424c203a29922fdc20a9eab46184d93bb8d012b43d9c27a1bdd30c93f21634b97b4071e302a9eb2b1e66633c5b2a549109060c462e63c9daab45c4d0dc3a219f068914adc2109fb320fe9d781f979bbd5a9f3c67dbebbd3f8a51a30eec0ccc62dab31214236f21624a25044823958195f0ce009906c34549eaf8825b3e0210c8411e63f20602ca5d55f21d1f4fe21e43975a7da4d4197f8e07e5a8332a61d1ecaa2a6dabd690d5dbb3c398581770c53f05e51d3d073a0d1e78e5ddbfe2d7df925eb95a5123f9afae6c22d8af72f9757af971f8e1998193a228ff3f1cd54b29acb76e1778ae6395325e765d64585e52c82dcd76ed01868ce778a694ee4d6e82b13ae07051a694f75f5b48542522400355bf7ff22a59dce2c92ee481b404f26449c14284c14bb606dfe3cae5dab01d3114a752a6d3702758b9f9b4141fccebce75e4e59c77cd355b63a0b768293461ffc564a296e5a9176df933cfb8de5cc4e092495e625423f51bf34b0ab17194baec38fac39c9d10d6269b7cd4798ae489fe64a9e1e4e7a83a49d3fad510eb5bb7398db678f1621c8f440d7eebe0ee8d030f7ae6b10d649f0d11ba20813270e425b3919ae574c8c72f8b50e7b19ce310570974e989875085663cfe28848b87cbf7f000a91fcabc8b25f772c56de087985fbabc3867cefbcc5822cbf0e1385105909a94dbf0b0dbbf4f677a54b4705774f37cb935587ab1999ccf844dc016e66add658a47f33c17e859eb9bd82ba883abf8fd2db2a0e6bfa5fa2043b1870d99fdb6c976cb5ffa3f084debecb77a7e324cc7d1b4c5381915b04b6543cc965b6dc4383a145220f4c8df6ad1207b44058fcffe0cfac01f90a661b3469e10e77ce7110922a9a304a96e0635297b4f792edef7e87bd6ef254d8b5cd926693a365d6bca7810eb4fde2a33d58fc5a7bc85c42445c3d06dab264f5764166914966769f4dffa5db4f3b3822f2159bc978b0a6bb2910f4dbb828e8208e3cb431acfc388f3c9e3c5d747a83c3e6ff8b04a13a472a3af81b0f523df4965f26740e79794a24e704442c844d151166c1a8d6a20749ed313ecae6234892c847479959261d4537e68c82afb0bb35f60470e143c8a29340ef4a3671fbf6fc1bf1081d1c765ca3d8e89120a20e5806f03976a5955cd6dc50d5438dfd2e48e3ca93b41da7b2cc577f99d23b578999fa20ad4e121ea6f837591922b612f147196ded91fb43fa6d14fa0f67eb82b76c6f40fae45c8558410e98b95470bc0e937a29ebec6f1d748ab4b451ba17c72f7e1797cd741cbde509ca36e9e50ac8806df36bf6b6d5b6ec9bef05f78bcdfc2c7c8b1db97cffb02d22fa9a3cc17252b704ac87f648831186f9042fa999ab9d0525630280f9d951d29e174e602bfb85c4bfacff48b16a49067092efd5e5a5b32da7d4d89ec60a50c5aad101c103d53030f95817daca22f04c893d25e3defceaf387b155f59ddd47199983e99347989866d1df3403edb88ec5b1bdf600dfcbc0f31d66b3067eb84832f41074061f286f4f21b80fab72fa5a9aaa0f128da545dfd669534eae76777cee0123027760e3a37e94d378fcb4bf398d13ef19b5fb5c5eaffbb6e6d52c4537253959367c3216da20e47c5ced7f5314e80fbc8d9b508be14ea681213b66211b978a7934148fcdbbf991c9b5d74a5d8eed13112878c5e08f0bcbe4866caff238e3a49c1f4f4f20d80e12f8386a2711bd07e261617235db5a850492d87b0e143611d4366e73cc63bd1ee08061e3f56af49f828048807732b3e2aa3569e9f73cc2ccbdd6bd94d7aa74a5bb757db3218ae33dab4f7a08696d699aaed7098b861cb3382f3c89d83c15867ce63568b1cbc4a9b943aa4a1003a0f7272e96864b03c207c70b3362f6600d2dbb01d3dda9442eca8a9cfcfd3a9128e9303fc24755a98f9ab83179d5c3c095d8301b4d5eabc4ee4732fe32d76152d5a153e8981a064eac9e7191f6248f0e7d6e5b8f008ab4bcc30fc8587c7352263d365ee423b1c9a2cc14b948641fef844b041f2a9efeede6563f53fead818baa4ef2acedeab714e46f60a6a76149aa2fa841db0f6756cf7f9e932cd517987b46e9bf528a443b9da31a12e71260f2a97af48a8ffd9f463f0bd5dba7b64c77d2643c7e5636f11e3d1d71ecd30e8e3071e02f2dc286e0969a105f354bb38a6f784265f610bdae1067285f0fd44b68b7e3859671b5f894b9d89eb44b2f2d26e9ca2926adf15d2a7fdcd078a78b451019390bc0761cd3b0167e00fbb2d2c9fb68327cd97dbf9ea15f7a16399b83f15ac3166b6f50b264250492335fc2a7277d6a009e89c7c9cf16e26332d81f0aa7bc7b4088ab638cece6686d8edb8914c1cb63409c284e764a2ed663ee128137e3ee59aea2dc45a9f76c7ab5742d6a99f54f4b005db011ed8f2f907ff40fafa536af92379f5c18b7678f240034327d7d5cac64a20db1a9878d34c9e969afd00eac09f59c09415a38ae427c5277f069dd08d3e9fec559489aaeffed2df93d6b35fa3f010ce8b575a0d9a58f794601f9386f88ecbf14567005f95ed1505295357b3f1e6e61e1ab678ed300daae81ad5366dde22914a051a88b241b9bc59fcf29474fd9f67b90a021eeead39451d69d68db81de05ff31b12c9c7af0713e799db9919df96adc5e8969dea6c1cbb92f1bbdbb9c817dd0ba614c5b89fdfc7929248e95a3a574dcf94a19935ba4e6b762e8ec2d737c1062c539705ce50f95d05a273fd0918e436d7321eb36ffd2473583c190ac042604814aea13853b11bf44028fab297ba111258ddc7c0742d0b1f403b7f85ab63cd5b0107d59e748e856bfd7bf7c810a9c59284c79677701e3db0ad5636546cf1d4d8bad7270f6d67da0f40fd6c101685d23af040d8a4e4a53a446ea4c13a977e32506426154c702e7369896e1baf8d92305ab07c22f35e37a4206c5ec23bb2a58dc7aee6b33126a2d98182dc7452760d3d0c3303330caabaa74aee735f4b63028495b80ecfbe4ab1d2af90cdba9b11cca35f3b8fc08b53c6b2ddc69a0e85f52484b430b66df5daccc39e5ef7f1a6571822b0fba714b77e5698f5373d9b85c66b8de2be5e918d5216fde11fcfe83726a72821d8df131570f490197b6830ca48ce5cb43db10c58092ba0e0401e0dfea2d604ba0d3c9a630d467660d760242da8d1157b38a440f1f6d35cb105103f42fe9638e6426aa1dd1f2e176a71c96f369f1dc46b60e443cb3187325484928148039ba6c17bfceb99a015b274524893c79241a878801b3e28267119ddb9cbd0b88988d0a1953f1f1ee3a77543e3ea65cebeb9ce2f355a5a37248f803d6ad5ec2ccdffde7b8592635ea7eccd2514d72c092ea3723bf3363e563c215c3b374e7300989245fe723f917b95093078fc8073c9597d50817f35247e25cd6cb2341655ba026ebc00737ff557472462acd171da3c79a41301810d4e8e5fa15f15ff074244dda18f0fdcfe7e8925fdb594c79a9a8ce55bd493a4e6daf1ee392623098953fa7be785bc27d7a8d33983a1d42daabc18a8962d01bd0f6e18c7f3f70ec21348e8ba2b8dc3238066d3927011a2f78bfebf6d3c73ef988f154515d199c29a2ebfc712fbaf007756c90c8bf44d8ec117c1d8f5bc1d8ab403d03ffe2cd499126aec18d6f7f80e2deaf33b905854afca1ead6f4ac6e83336b87f17a8d54fc93e19e32aeda448bcd7ca9b87406b553231bfef71896a7ecfb5a61dc5ffdb073a2c1b5f293ed7d525de349267f6175c0f288b611c986ddd6e5f0ee543763a912c5da9950008b396d3b26743f79129283409245040806daf973b50ad33cc949def7e52a8803a4481148ebe3538920f834c7f027eb8a58d770b2699176d1cb1d45b9900494e285e80d8ef24373b9a9bd5e1446eb54369156ae07023365d294c83ee80a88447690bd728e7e018e628268cc221024bab987f54534b3ee3b76e6b05c42c73307e67a452e9211bcf75befa3fe706fdc88ae399a08c82124beb412c5a5ae0636d71a4d3f7500bf18f8176aaa29f2c9373ec0b725c59341bcdf744a4df63ea67d48a73a15b0bf3c98438b73eacd92d2f69dd4ba0516ffb292f63ee46f9ed27d225fff44ca746251ab6cfd815d32c02452609b8ea30f6531bcf74c6667fa70dced82bad5eb44e1b7533bf7bc53fe160fa0ff9cefdd81f12420c3f757c2e32b5381aef6891bd3b584582fbdf32b0678b1aba5e8cfe84480835004f3a26faead97bd00794d9b949c6abde6b7b831507fc5389f0aca1077ddab06db1fd269a53e3561e6b9f736c2ca7871b75ce39ca36ff1ff96ff1917fe9ff80663ca90dad569164a1e5505cec1c31cdda6b2764bd2bfb24a1ac27355d97852bde52b54d4dc0b0cf6250c93f36049714b32d155b159ab4e407675bacc6aec6bc0058a74e42707c7f45efb50ccdea94de044eb493d5734d49530f2c1631c8c59677b2ff2b7f84d54cca16ff204517db6b1e5a310aff6bce4ca35f9c099322cb8e98b6427ced557283692d7af075ad2aabdbdae681a5ed47d0b615be7017f69b28e0141f256468620cd85ec724d5ccabb06a4ef638bb915b4520c3c0bd2b5edac1fbe1ad4a0a098a2471d6345ff7fa0ed2b78c5902eb25b0255fdd8745c5f1b5006d0a663fa0f202ea281dbbfceb1101a65eb5c02418de3d856b6eaec25a481a143a6ac7f0e2c62e758d2c376e0105f1f3e5b70da47eb11b18dedf23219827e39adf5dcefcea26d5ccd421127a9793443b62e1f9008d6ec26c52b4e3841919908193eecdaac2996132c87b4d51307437ee6a73df12d96727c091a80fa2fdadf4467a4d8cc0c69aad7ad2c629a9136c911ab166d1a9fe5b98586485f965a351dc10a462f2965753e4f67b044d0a8f52718c2f47d39c957730640c92656960e764ab4f19f92175324dd6701ccb564a33aeff6e6188a62576686466caf929ea66012447ae7dabcc2c688157e19153f08344d8906793e3d399d54ae18d87fd74bbc77f039f838351497a128c2bf1693097e6e59c469471aa96319a3a947e69275e71f6e78a77e50657dbfc1d3c90e48e48fbf228dbcb8df3cc2b95e5c3d52778041a88749e856c2a4056d9f80c6f036a0a64322fae35f4dc46a821008943df956b3da25cf38406e379c9254778a56163680a2d67ebd42e45b9e2328e52b2454d440d48738e68f90cedf8e7516ddd173b945126688ba5350af8d9b558f9fd0e3963798f870670b7179d40e4a440b7c4ddcd594d8379670308ba7a281428212f5317006c59dea0decdb55ec952654eaf2eace5ac198720b574c4a1f4425040d1ae0f935bc37f4076b0e35e062a6bbd634b064d47be91867018ebe853641a0128b9af05821e19704eb2025c90a7f91e3675e79c3c29be9e0924bfb1537de602bf34dcdb1cf83bbeaf46a8e2cc3f3da20397a8b5ef54ba49b410dee98b15b5d3713d3170abd1d477ee9f08749a16fa4ed6270b950f36ef4c0df19bab6ff47f01fa445ab55807740a0a701e4b2bc91a72cfce689164dfef087e0f41b66a0ef5ebea19ad8a0df74332475f95d510a9f67bc233ca27d34e0e9d5a624d7d612f86457d0d8560502ff9bba1647ab9832523d71eb4e334e97e7e0e2d49f3fd2a44fb102e2372df9a4cf5eae8f88d4bab7ef7f71298cc8dd801b66f484a701c9e8d1bd911a6569ea0930d66a1da79ea9efd864e9b32965f2cebc3550d977c09d22bab073a55851722c10a251801b409bb3a6d329d35e49924adf2d3543d24f70ad9265a41a8f4c635905b1bb6d101d026047aa77f2b9f52f8b586bedf982940e698083fcd0adb5e378106ca03588c3b8b83060167efbd9b8ab44e6caab695311b24973c7b296f653f3cae22d27990f1eaab37de51ec3b287c4682b603254319258debb873ce8d674f83815f57652960998a98035d37fb821a9030cc8c23f69c991b21b2df9d73bdb42f3006f8335db1916715441eae385915893fdfb427f15fe3d0be9970192302abb23e10e762211b59275f78d2eb0d70d086503cc3579fe835c4cb8095c854bc763b34bf40a46572b9f4a28be1b1b4e1112922bfc3805624f8cadc0ba3ef7fd010106f748fd5e4f1df4b5161441f0af588e28295b9652c235d41fd39a0ef878eb8bb2c591e258b80cf205874e8e625bc6092d91a3124732802d077e0bd638c9df7cfb93854008c6d295ddf1b27f6372196cab7f6e3484d2ba4f2b2732c80bfa21fc8a033be19099f25e5393d981275576211384e5acd33389c29de7b7920c2219f9c9e75b66895f3a16a28bb31d3bb4f613f4d26d4481af32389c9d68cfeee9561777d86d494ff924ff2acfc0858a8e273f679f60682a039c7ffb9da39cf8cf46196e2cc64cd8621511f7e5b29ec2a7fc05017112dc78ad106c4ab53a64905e18cd3a36e7757f673ee1e108d8e3ad2c24c43535d2303edeb9f9bdde2be3041002f1c73b982c56f3c9e6a1dd40248c10b76d57d05d6aaf06c2b3f8af94d5beac5349b83fe9e02f511c44efd50ab621aad0172853a1e1272ab58d41e3a3a367f810ad7c69ef656c917bb945056db9eea8a866e3850ea556c26b4f00d425117185b8f598f90986629f8a8c71345e945c7f9fd372462c1746a38866a33f7d43a5d6494a49ce97c56d9adadedc85817a2a87ad64fd9e8924e53d5776b9f6086c9397d547852ea6d9fba508ac2ac89281c0d9aa2a99948b62c0af6858109252c36b3e023df8b6b9f30b658c97668d8ba2923d2e9e18667679f76e9d92ff2eeb711e56e69c1d2702e421eae9f61d27e3fb70506edf221fb10bb05e1ef73f8432a8da0a52f7c953dd89c1c0587fc0a9e73aca65cbdf370e30ac85d86997531a31190506139fc7d8e1c61e343479d3e034487cb91be549098449a60c84f17e3258f742231b0f4d7cbd5b0704ec6c0d5682ab9b4bf8a10c83f1794c53da569fd2b2fee81bf00bfdc035a76f728f043941ad0c215b361a26c10e7134e61c71bc257b69817e07820ae69c5c2e0006c44a3d8b1a931e843d306a2998e38d55ff566f5ae4f499d78caf7519fee93b41c1065934838dc9be743313bad5e3e720f8895d7805747565c55d2a3b0e25c411d6f98b5d053e785000fb90e0bd10dd0ef6d27ef3fd600c2b83f8c91b6e79a9768517d9dcf2c2c896e81aefe29c3b97e4dbd21f1856d1047b41ad7258c92fb846aeb21177b8febffac04c94634212fe3c5793e185bac35f3d35e204a7e2160628082e241de337f2dd662836598eb734474d8a307e82efa42e797b082f419a2db1353d8e624f6d2f9f4b47a4a5cef8e44fcab9deea72be618482a20b2414d04b4afb658a2c0e508d782ff7d24bc74e9bdaab063cc3d41d6beb1f0ec1fd7c85e8d900e08ee9b1bb8c1291901184a89a81a5ca4edd68104a4141affd14b665625424f113a019e83870bafc231d4829c22b00e4b2d85379c1e7736a89432f508b4366be228ef21fbbb09b2e2b2aeac62f21eea0d040421c2760cdf58a5888bf9618129793890f285e9a616670667c8bfa19f25e90a4b33e84277275a964b713aec41cebfce629d93354bd088a280a44974a3f0910a092dc48cb58487224178658d2bd8aac8038bb69e5cd8b0982abe2ab263b5ddbb5db236049e83be109ecba62eb4e5c33f1b81e915d9f4abc0826ea2f404a2d2aa83b9fe081c04adbd5b892dcb1f1e673f311aca38184df8ff963c341ef971b003d346cf17656408cee5f687c2748d142d95e0f90394e853cf98d11148f0a9b1f48f94489ee82c68517a13189eb9e25deb47889fe04fc7a5ba0934601b33800d23aeafbef6e2ddbbefa119b52e656a5f40f308a492a5c1cae1fa2b1eff1f3840544e4f364c869306a7df31690af2d746104968b0f00357c9c4671aa811027096e73526265f785775988e8ef3f8a92eac470ec4f7ce539189422a1a453dce55082295446e04e1f05cc3c2b0f80e973513985cfb2eaa5fa1d945b8dcb886bcad9abe448ffbaa5014f852973f95c31c768d2689f3e74447cde1e9a2c71158f11788898544a8bb03d1084f9695a39598b26b0339e44fae746e07ccf4d1c80bdc5e556437e3af7c293b99eb9e5fb58c405d3bf8eb6b2bfdcd14e2cd3b3e1d9e6f8a089b1495d66d5e0b71300bfc97e6cd227e833165d42fad5f7fce9187fa3540dee57943ee29d6cdcddeb3526548d0378f9ec2a914a0ad766cf0ca914c9ff06a286ea1f67b2f16523457c3ceebf42247f1e9bf860ff814f5d0b9e73fc080854fce25532b2260bca0cab6ee524020c7b77b2ae72837edb0026db652bb01e9938c4d8bf2dd74217eee91add4d3f88dd7c853d77720e6ebdb5adc9bf67b704b3ba7592d120b67730ae8edbf138d5356329f84984518aa6f39ce01d965d355a22017449e0c62bb847743876695b130d656aca9e8905ec6c57ac8503025cc70835dace76df78fc70230b21382a5a14c7f8a9f2abb9c54e8f16e577cd0f9d0936790b2aecb526934d1cae4b277d9fa1c212f93ed0b5b03ab7edbca3af6f54855884457804c677fdd01ba5ba290f16777884acf94a3eb8fc5a8f5fb3c3a8a0825773460cad864e5957ec862056ab1bffb6a9f6f24adc2d63fae1ffbe358fcd3b75d42f5c47e7e1f7f066276a72dcf2edd25f972460045d3be6ba8e3464c1d8287ebeb3085cd0949a40a5a546651411f983fe518846ffb1f7c6b9afd1d3db2631f6e9cb90b744389e686416a4d31d00445876df6d3a7ab2817e911438eaa297eec0a66681cbbd6f6fb3e9705016dea90a0ff17712af695625f8de0b5771e5e3df44a3531d43857c9d11eab00b78a27382ea189cf69fda1d5f8c5426e2e209dcebee20ead49b987dcaa305f8e57946bb05cc4e14a7755daacbe900207a6579ea5bc31296a01f64bb2cd1763b4c1012104e253d0b0815b5887ac86d5b1ecead56f7f724a4acc2075413813d44e78f6aba49b539225a87e1501d78720dd7a88c77b9f1e45c5af3c8ef9401adba205a722f05eb97e6018611301e4163264c53985bdc323ffb52032b63d995ce473172118ab19edd237af5f66e1cf21c0f9d01c025a7ea6778998a8ab93429f609de5c72e72d60835630086e435dcab17ad3e59c9e0e5ae436ec5c0963fc0e2055f6b6567da7919a07a8e5b4145eb4a3033aa7b387e1b8bc49c3c3ec58590ccbdfc2c9a3a0334e19d1dc7a9750216c67014f001d9ca060db0eceaefb0a90917809da09d607a24b43f8df649968cb6b982edc1df5f31a6acad7b9ed5529af3501c874591cbcac82650519202fb58ec3f283f8086ab12fea74fd00d76bf593ade0f9290ccc6984f895e6fad9b2121c21403db903f03ef067fd281ca8920d09873969d0b0e38851771a22b579bc0ff31dd078601c5dad12f8d497555bb9e6541199aee4424e60d74706cb5261db9fb37939b3d207c0a702c323713951f89601d238065c822fcb0a171d87463ac34275722309e88f14c0e5d51e1e99653e4fd7567484c533e79911b81a8193a9f6ef4fcccddd6cae093c0ab625328bb0621083c5d6b802a92c56558cf160dcef8d88e45627e5b643cd8dae3461a164c44806a3d3c0d343c5aa463489ff6e81b6e5694d90bf3ec1a718b7cbeb3417fa1efa9f6dc93f04a9edc73ab1294e7a9343c97ff0a751fe962427eeb872718f50a4f26f2cd3638ae9af174c5adadb8e457e30c83412c66e34c181750f33e94f7797aba1acf02c5b038bccbdf73f922cd9cdbf0d6c9afd1a7136a1e549369cbe87a6fe2ff5e7ae8555b3c85d7707530a9114b4154f63421d9aac438fdf57d0328acfed823734124755a8808329820f18e7ada6cb568b8384631bc2e78270c8f324ee934da9e208b32db17cc839bafc83d83ebb78044b25e280b04bb994ac940a4dcdeafd5f9bb1f20d55531190c54020126e9adcecf0aa49cd5cf866a4795d167f2e9928d27f5e6d967a4528a377135e7cd948b08dfa91f510bf1be10d11bd4ab38c171c4dffc440a59b7d30538c9c8e6f105a708ed3615bd61c29e49d405b847ff62ae5bba7b3baab77f37b51ad64be760840311acb84ac1485a4ce7689cc05c48cca5023b5a3bd1bca6e53a89a47f5570a05edba29c0e4a0bdecca2f4510b1052c946a791120085be4887f5a7bc53792905c4157921d0f6fda734c1753c611272c98df4fea64f4543938dd29cf7d823d7e96bb6e96ad32e558898363d428f3e8c351798136d5408e8e8993b6f5fbdcb6a4deda9c00c8e31926e9d645004ce84f6f06c7d10d87ef424ebdaa4b8efd303cdd7b558e21a2746d0c850dabe33cd32530df8cea6d83594cb8f61e422a9fdbac733490ac5d926ba15afb919e3284c6a46bf99da5f6fbb9d893d94fdc060d8b0830bea5e4818f6c9d53ec1331cb1364b609bda6ff66577cbf907888779cb82cb35d99ff4aeb09203e0b4ecfc3ba4170d6c02bb05710931bd0f7a01b16c67e971f74f8cbca6b332c52f45d783103ede4f53ae3fde5a0c0d689f7079e355e056f6ffa105189a722535718d30255c8cc2c24cd780e1b912c9a6ec10859403b936a09ac7d0aca536bb7c27207d13d46adc3d80d4de2a0c0da149fedc9070295a8704c9eee69d044f12b0bfb62d9873be31ac2ba42a1cc213192882c0e4f52d19df280285198167a9cb04b58bb9f53b91aa4d5926a7c66daa496898fee55a2ef4082ce1a751a6e8819b9722984c6017acd7002d9a071aca371da50a355316c0ed85c0abe901ffc5cd87a42c738733c0d4da7086f44e985f836d69641f85b58459d9f97cdd18a860ed1042fd0799b628385a25519c0395ce4ee934b149854d4d1f7243da2f9c262e79c0561d8001199ad8b4821e3596b186de13b67c88afbd64d80d1f96e3bc685f97c022a1c9cff38b28d89ee285eb3c3b623f14cdb7dd7da055b4a54967d7f8d80f14069b431be365dde9dc9628478abcd9bca527ad5b3d560459f784c3a9b78a4198727d4f7db239089d9350c0c727f9276abebf893e67482ad07cee2c59c4db78401e2832be46bf4a1ea688c9812939ee2249c137746eff226fae4c021d96c245ab44f2a61f35e9b84a2c9254331260f77fc3ca8596dc8baaf0ccb0e9d720475f4314e624f4e36e2c51e2191c4632c2c360e28b1e8e89dd6e7244ab07586cfc5edbd8f00e8cd9fe20146227bcbb9ca6ee0a102d00581570ebeafbfd9e83ea0b6a8b69917457572703107a91853c7533d53de85a17edaf6d3124ae587aac8b493e6a637c955bc207bec13c04dbc336c998474cd8fa56c148051c5081ebba69f3b50073c7ea6021eb87e4c8aecef483d36c4d1bd6f4284049724dbdba970dd2f73d88cf2f0701f96d87d5274dec22a33f1f856ae124d54dd6b631a73d6de9fdd5b31c55a83ac4d722355d7a2166a86e9f6a832e8c8379ce2ce7e1e1552cce44a6f26a4342c87e106f34b46b997387b0d7e4450f45e1212c07bdfabe6cbd522973bb3b98440147a6d68db172f35e6baeb22d8cf88cc7b4cbcfcf3e36e2f67c8e9fd705d3267d2877b9a472db1634cdd355e7caf9665a35f84436c4d09f92608bcf0259c3a6737f3af8355528cbda8f7d5dda374931121bd43bb74bfee17fdb358282e5c57f36acb9c11e0243eecf0c64810fca2a9961bed735f16b76b2515102d92e17342c310e9b8945f56e43bbf89f62985cb6ea0bd16404f3a3d60f9f48b2a5259c909b45e571d5251bf5a43aedb506c3fe98b92104b080dd60e3d108d7ff867511c850516638d9374df79f5fa32e4e542edd86d5306d540cb7b00b80d688d631341218a34d25535ff1528b6628e145d941d18adaf0de4931490cc0f546e33e7171733907ea6e09400bea6b38807d782620ee0617219e51eaaea920ca48fd6c7278682ec13660d8bf24ee211db2ee16852954363670b05a9be3f535efac54cb7cae0c703f960cbe7634806d0341be05a07e0544fbdf2a8285292310dc81ae293c3b91b54a95ccbd6509ff02a91bc4574648f234b2698677cda49704a6b146966a6b9d3a88cf9401566d1fb97355e44085aaa006b9a000c1f8f1d309dd991ab791f7fef9d3f3dcfeb15facf75436520ed17f140fbfeac875c1588143b1b3b7ed14f40fa8630b839b8d6b8cb6a1ea42b5cfbb652ff0fdd6098f740d5010312860845c6439ffeb539f4002120b6d300e762c1da44de6d0c6452540bc441570c617ecd105b05b2f18c736655c6f98abd4380350af39d5804f35bc2d0603bbfe4513a987da6fb3e222bee1145e28e74142b963a0cf2db65adb55248ca3684414f8bb1392f3a6e8addc5091b8a86cbdb1527267e8877f8978053e13acfda851d1269983c83b984a14b45566f7c6da059a2a0878fc57da66b32c7698e3df13ce5f2fb415932388364d4f4b80318814b0509ef1222476e1517654c968457c684e45bb4a9e35e19673ae6552dfee48ccab166abcc8e86f64d0dad802c284a5a6290d30f7b072d13be5a535ed8a0a912e88af8903a6056423100346843a2434e451237128f00a1e218c622175fe50f7221167ccbc9f292895c22a5946697a172dfe3e44398e7eda748b1193510db167d9fb0b5ad5aac156987be7c3bcf0143c559b5695b9edca1c379c63e82590f25bd7a0e55261c595b3e9101d8caa009d04dc0cabf164f321c7aa3f8a949ca3c9ce43873830ec413f6ddf2c82fec4d18008162bff73f0895fe12659eff7d781b51fbf3767c3b62d360a7381f1f353112829be05102032f60108c0e2f424530d5d45c891c8f3a0d68108c9cd8b23e40a6b44737c421a3fa923eccaab46bf11a2d723d6b1e53324d0c1562ca53b9189315da9a062e93287cd23440002d79b57e89e90d15c9a8e712e4132621f05a2a75e4cec39bcdd810a56a92b1c129f4597052521b9ab58a3d85a2993e223be24b066067aef443509b4befa2803e4d60b57c60cf4f4b6ba03bea7650269fd53746170ae185abb46a08a08a9b1e08bd86c6d5f4f81f12d83fd9cca34e3732b8835a4b3a0bc2b8904dc66ab3e986e79071d36414b634e70f6f4da4270db3d2957dc69149d748a2d47767e79157f927a50b1192990267776162506f38dea991bb6eb59696470726686c203f10923a111b490ae63b7dcaa132d6fd744681445091649aee8796f47642c10b8756f7b2ff79aa7f7c8667345aa4309338c8fd9e8b446f49d99c6e65878fcefd891d3a2ce97d209735cea3d07d42ef5490f7e48e89b7db04f8d48421e0c95987766b3e66d51f1575b891e296549c2d462bdf8303c8e6de5237cd53680f34df0b8431a2feb1b7618a1694df16fabde2646352798c38c122bc0ec89164fe167777da091c2ef2cc12229523bab2c200a6229d4f2672683c60c9ae224b7750160c5d3ad96ff95b16db207b83504de53d68e941373fa95a968486ec5f7f3206588b8fb14fd1c9ef76e27e1b5bd3a9315ebeea7522af33e69d1d7bbcbaaeb70f367f71719acfb4ccf13592d932fc4cf063924930d19e452e3279f0b3b6dded0d3c083cfc61693498e66b49d3d3307498005e2814f0c15a722c532dd21cede9df563b32ce9118f367f4535137dfee923c0340b291c58cc3857f7abcfc82fe144b2f1011f0c4446cd3e7caf7fe0e4494cc7c2d6138fa8fcf1a9245d5d07b16977680c8092a898d3acaca436f36dfe2bc40c05a5112047138a5e35c65a757284bcc06ec6440540fdd88e5e3103d7b05b815b45977a6c5f638c953ec65528caad8055cdddfa9e2853ced4c22c714e7ef30cce84af3816e85ee599a5490db1eaed13dd9c67263cf3bb3b76d2bddb9ba6ddfa74ae4b570d2a73149a006627c8b29f9f8495489ebe8249c21f8342b6389b6eaa9972ccf957f00863ec16b9a702c316629477f2b79ec7e3adc1fdc38b50af2de255b0016f1cd8b730f04322ef0f0794f1e603c73bb34bc2fc7c2c541abf842f93945f9353ccbc293a15c05789073b3a485fe4ba822841b87a7f5b23692498844c7bbd506bb3d423083596b12d1ab791e0d85c143d58e790a2feca7b37fb0b8effc708123ee3430a4d86a28e964f978978bb20add7dd2698d89521c8fb6c53f0e677b169e157d42843205b78903466c4ae5210fff8f593923426396a6bca455078fd4f47117f3068c18ff213bb31545965d6787e2091429c40ffdee3506ee8ca45d2fbf0859bfe40efb48a1c53da73d45e7a992893bc50d92107154b3df1f00a5be778a37e30e6baf2b829e726130031f4b3c6bf2e1c8ddd8f1959b53a5124bebaa31b849c9d52f474df7090569b10ec9ccd0543a229c12e0c135c3f5296bfd2ec8f8e1c26d704918ec95a4ffa50229355f0374ae6b3b52d5d64905820d8332c5756b0c3b9a6275a431b773b39c5e08ce09073859b2bcc9b2e3ab7906958e81fe5693926d65c68a8f4e8704ad287e4fb560979eb3a9853354d77c2a6c9ee1907e8bbbaaae80a8b51384865dbea4d226a36e31f5a7d7ee7ede41c6d5259226d3ecaab28a6446e47995993298acf6dbe46f1f112d250e99729d7b3c1d871e450b2638970a2cc9fb22c3979182acc302a6048ad59ac82289c38a1a968d0c41dba81577ade9f9975e496b19b8e1a2fba53dd790c6e1657a805ef4843d141b7f5654d432263eac998e39ec11fcb0ad7c6c39ae44b83289bcfea088e244c743c19efd33d4d469ae4aa9d02c2601b3ac2592b7c9e5734b3f66c779277ca2cbf9663c1a51b05aac9fce363eb6b9813065d2f300d03ceea29dbab12c432e862ed1fbb7c2ce968f1894d9aa878c976fde7d8869f16963279ab80aa092275789d6f1d56f0e197b306615d997ea1c31481ca58bd66202c6c6e670d3f290d87e0e4dc72b95f05f2156ea5379cd1f0e2f737f4f285055e5796f270899e04943513482ef5dd86177292b970db57a3b203b3e840fbc3fe0d99e4a50f1076742ec2225260b9fe2528c7d9ab2936f78be1c0aaaf69a28c1e901835333db21c47bd7c66cc0a53e7e89bab3e7204c1190e1f1a570e9495024b710555ff1865928bcffc181fd53702e0f8570736eb0d24ef4fd49c7f60bc7bff9e98469d14057293e813313e643c5edcb79eaeaa577f468037a13aadfc09fb90038556b376cea7106e132b574fbae8f42446801c8408ca2c12dad3b735ac6662983104a058bcf6ecad3b98351551f35febb122d01f9211d94547b8385440a09b47f6c7a107c5cd917725fb6c74dbf9d0fceefcdd849e6124f7bff4688dcb9505507743aea2856756cef30ba06e215db65d97ff0079e23a7d6a5d9537f49114fac1a8c6b1f6b7058d3edf1869ba50baa051f554f3404aae8d858dc3ef889163263a2e1227f5857a84463466b71b9b57acfa4228b8be93e97af7835c2a6c250e7c473f19c127751545a13f719353867592f3cdc27906330daea80e15b55e46ae7c571726b5c40b7ec4fb81e5bee0739493bf96e15861bb4782834478710e25deffe47bd68ec67ecd5875ed341de147394635a64ea32bce8168685cbd2c2e5e34adcb26f2b034bacc988f23275e25cd87964e01f291a2de93a07cb76a93500c7f865f2c4d6e702b48a42b56427e98f2fe605e23e37f2916428ad338d2a609c29f2cd56d306c1bdebac9ac2ac3446758823291b825288e1f5048a901b3d02c11f5c47724d0e87e57f1358d13e5794499f438bbd5358946bc422807797894fd95db9ec24709f53bb8accc716376d8fd47c305abd9a48120f1aaa8b1cec7b1fefd77b3597823e8f31597b0bf79ed7b32523c1becc27bf48853f261f7bcf4052d3cd2396a845408bd4eff9c2e11cc01797a593778a14fcb889082bfdb1bb46400eda54ce1e63a82a050b82a5c78ea4eae421884eaf7e44d7232b881504e7e6bc33d9e2f2e5a6745223b57dd6663b7fda6f10395911c51882d78f107b081ec904c46a4c374b6f7c3ab976827c885be70f40727b2d51a7d4293ccbe371d144a23ed81a0b37aaab468f3980d665ef62d46bb33a0e178456ffb3652c3528b250bb0f3d31fa7f0246e62aa193fbe3cb445b9a89c8e3b57c3f6605139cad7c1b4efb890a123a1563dcd217b4ed0fd8d99a1197a3fb46ae40a95d39785755353a4c373d99cd8c8d7cfe5ebdb26182e2667d6326e146fe5ef8e4de680838482211d86cceeab6ee18d21e2222565b54580a75706b2a9e1ec7b4d6ff3b4561dbb7683265fe036b6a00d72c1f3e90cae763bfd48a78798c287a5a43deb08642ddeb34a52af31d67c9575561dc21c9e59f9391c061d7f1c167534c48fe1d386104c9430ac68fb1b2dd024e440001eb67ff2fa972620841668d9432650f9349a0be5307353192dac2c72aa3ddb959c2de05c6c9dc5233a3b33f72ab395bec071def13edfd6e13ad25c9dae43c6bc8805e8bcb15adbf66a9ac755cbdb9ccc219584ed347f5b18b7afaeca73c8becae44a5d8362406a2d57c3a86a3cf9479f685c67a900774b232d508c670dc8990e0d4722204bd2cf89a7ad40ad4242190059f6f8a0280b7e443764d3de289ac4c13a23b4420400e7961b92148c1ebf608c754bc4aa894350a1593b89cfbd61941a36bb62bd86b2d7f42edbc9040d049511e67ad4cac4e6bee44f0416d217fee11038e8ce74aa56b175023ac3154c5bb3e48b71860d801f1b0d616cec227086867a05271102c73a85d53087232ff6d3ca5d8149b4001fcfd2c5252db0f6a7ab4e551e6bbabbfaff45343459ffd748f07158455d4f786a9521a8921cef4ec17f74c0f90614e581aa53f321bd89ad5bc80299cb812168e701b5644bd908fefe552195d94f40d9b2fed593c1391b4b69297f90dfc098149419dee131ae5b4ab6bd6783301f2888147ff6cc72717170f152e361391c9a7a1d0b967462d02a12203ac94e24ae4fbe895dde2e66691a0bdf34d9822c2cc9f5b3b4c3d664d1e3480d3121119c1d6875edc87a97ac4f0de513f2f47e9efb0f920e7512f5bd1808845a26fc3951431ade7fc196c5747fe80d3895b615d3bd937674f4ed81c57443d9598bcdd90c5e115c28462f3474f367a9e9b08cb2a71cd610efb92ca3b09074fa1318228ef3450db1787cb767b4b85108a35f8496dd7e499d741d3bd14d1c3e81a69644db34ea0db6bca0a43de77f19e59925ed8785b718315bbb3b44ac8c15beb5f37627aa19c35561da737a3493849becfb014c9520163d7aab123ebd90c51fe677ce1bad0db24889a00c0a11de209da2618ac5f012b53740b415b2bcc285d1dbff82ee4c79e0042e23fc72ac55201472d2230215520909297f0a46608d0c1db35ff7b9d9cb3535b895688eaadf3bd095d677c832d1b001890514056e78a15f14b57496434d71c8d422360d0a58f8f635a51398f7cc21fa12f72c124d7dda0123893ba6b24cf8228ece31db59dcaa1a2f80603112fa87c4e88c0ebac7e3973d7525da352a0e638d240bea1d2b55d86da781ee191d504c5ba71f64daa2293d9dce01f31a2a549636f0e57a3832af942b9d32d365ce409b570d92f1a0d2bc64ad81f35b58ad3c46d89d4d5e3b6ec98552c9f884a2bdcdd11c45ef6e749f801606e9f4d0f907531c92b2a2792593f52504bd8f558ea9dc6cb69fcd81258822022a003f20c903b6994b73a5a912b34382564470d3fc0a52c39ff6dffc8f64e980cb95c0cd8d3601a53dae39c4d2f830e91870af12de1936ba08e0b044e0c541dbdb0de38e3d5eab3d4e4708f16e1615600f8ff2872b54bfd8254aa64cdc6b0cd81ffc7f672321a016f78b9d5352f1947b81ad0709e84de8e032fba9803fa5502ac7e3bd008aa049e4e8090da2e24fe614805b86ddfd247c605df62ee9bbd79c48759d0540be1a2397bf5ff3646cbc00afb4b7e4ecda5915ae238f0d2dbfa5469af7f937670b92f47aa283028c6d09bf0d922d931bb3064cff6f1a961e7316f4801a39e8c486effa47840c43298b28d8992de978de66a956de265f507ef9bd17148b7acd213dcaa507272c3221187afe5a13e1351084c6dff9bac1b1cda079a6b629016c88169028301fc25f6e63e4777eb6e9b7fe8cf4c952fdcde71d5a3f6eb99197435c89ff0c2ee7cdd0303e73ae170385af32168e01f1bf677ab19e452194a45e42c9f8de434b6e193197b77cada942e834d57e0aa107c8500ecbc1fb6315d62524f6d1f899bda4792626d6360944867dc7958f846a680534a62c8ddf4ee1ffabc2f293f79cdaf0285e30df0328ec305b4ce5f6e87e144f0119dae7dbb26bc34f2541ab705e1128af6e96e009c91779df64b9e582ce0fee4f0b59201dc78a57cb57960434be13d3aad98d3ba467b2e58f56423d4c9fc59099e712d466189aa722d9aacd24571447f5c0a2d8653446fbb2f6602b982f23024ffe30e0d6fda7fccad7e8d106971410222dc1c3382d77154c908b60cff38e0a29a4b6bdcf091987c0046e9a624eadeaca4679147cf62e32d0c8d69f14a15d535b7873a83cd50604d32677ff88b5e7f00db6b4b5a22a8e642be650ebaab9a34511c0ef0eee09705c8fb3b76fe690f7cbf5ad7906fbd3f83d673ca7e6977c5c668a357c57b7a6c11a44f66b75b8364adfa39b6ac14704057757c6a4d6d6b995a21750927570cc3660885f5393c59ed9f70b0a054b363c24c2102ec7a524efc76a7f5d8b20817a4602d26f02d441d4b9844e33ad7a2984b53182c5f4d6be0ea66934f3beae3439851a1ebba933c0f940b3e699a4eb6cf1982c43422fff08ec3a6ca8e2c22dcdd34445acd04bb37ee9990eff30c4ce81a6b7e6461c43fcf3b1ed0921a52b9a422cbdcdfc8ea10980e6bc5b8cba137e8c74b3f85b8306741c1fdf88f7d59d4f62f40248c27e344f0874f0225cccf24feafbfaf6c75e63a4f77488956968c9ff3232ffe279bcb7b1dd49baa604e8c4f55abb1a6b188d3b4a08d751258bb2fa939d597f4ca9b7c2f03da145b9e36e76e9cdcef4f909bdd6d2e77c0ac100365b9f98b813f6d6332eaa462763b20432f24e0aad86368c3ad9ec391a7034681ad5be06d60213dc258e5390502b3b9b8a8ac1ea5db80cbacd0dbc1ee1304bb12e441a34f71f215059a85b9b5b8182303bf3215e1cab34ec13a7644cc644f56a6f533d9db6f5d17c1a6cb5df558b814d5b549762d9fea3a092cb57fc5708fdd1d203191082a573db9063bf83be6d53eb1344315d0bad4fb720645aaf1158bd1ad84219ec335ab904f23a78010e31c9e75e6b50bb9d2222693acb0c6537a0411771ab23d62d0f9e42fcec097d78d7cb77e691c144763a609172f8612b90497b8c91bd8463b1f10b31a0d9d2f7eade08ccda2869e7f5ed170f469af9097fb6b7eff98da520b887da30ddb1b9ff6598235f2c33d1d8c44e376fd2e1e286ba9c019cdb5d2575140230365fc7a23657d9411a55d2841b59ae4dbdb822f563fbe497b7aa5aa168735e8d82cdeeef0dd526aec1e3a092e802c833d572f60369ef48350ef9bbb25d0cc6b306757910bde565b01a7ef7718cbc9e47ec47eaea0137be4f9d63295dbd680cce8691d4b617dbf4546d8dc5c85e6918a63a326f71e53a9ee26973795f10b7f9c40540c53cb23d515a23469d70ad73cc89eb3e28c5c206a0198cd53623ded3b8317a645a8afb8b83857f97a05a80084b70fde1dbec188e69cfb9c1a16cca8d757d530ac0765f151bde6736f4ecf5f80efd9d80c5e1bb961ce09439faef39ecb810cdd3bbcf9883e430088c9b265cdf08718926877f5f91e4e3f7419e145a7e55b0902490d4c6a35d8755689c9a6a4c07740379b049e4b3705a642ed2a7d5a7e06856c558e8e61f48f04a38c306b35962c4b026e6717e8fd0b9e4dc1601cb9cfd254e8b8b695fd55abe4290fadb208e122aa623d9bcd88b8928338493fc59419e16bd3713221ac843bdd06a4c174f55374dc4cc31cf4a15ab1abb73a93f02d89f9088d4177b7b47070b249464594053031620794f55ec33c74b41becb9679f840d8004e35d0a2e0fb31ba36cbc4c414c8ab30a3de79e06de3fcca9a63c03bb1b34b787e77f0bf090a09213c89d029e7953ae7cb6955d82f03890efa8041c02eeee27ea853c2f7c5d5a660082d0e176855d72b7a9de125143e284ec5807d001ab43fb932b8b98b58deca82609403b3d3f28e93f3d0674a5174b1a2547659d8a21436e515dbd49278c66a6d0f3dd5ad63bc674a6a23144e0b3c25a8a5263e2c2e031ef7bafd361fda4966260a8b598904d5b674b6cad8b5affbf4a9d276039184053873d3ddacbf0e71ad48106cdae6d012254ac4cb30c2c506e38afa2ae15a45da62717db29ae466653247aecd85bc9eebfc05ded8895285b7d9a8927516ad8757acfbfe6d3b01556c969bc0e06b1bf797455d3a988c487b702e072c82127af2cdbefb214ec6919e2379b813e679ee4d68c98983168c870193c31dc82a036f66d744cc8b0c11fdee3b666cde36340f3025a77e23ae831d83377a20a828d03fe3120d647b631a3b368ac5f8bad8749bd358adb41ac7ce0fa9e9ae2edf27e1744772e7a9766233ab38efe62194e05c51915a4d3f1df2c20b3d1811cea8d45059c0a6127c1d6910e9f64976926a1949e71022c32b53825ae0ada1713ee315e0e04b04956b2659e57bddc10720e6969beb2a4a6e83d954d3a00f651ac904c3b8b6daf712d4802559e3cf11d171a4a17bf192d687469396b85a94f06dc8057aac2b3fe1b19b9b974bac5bcd4581b420c70f3993efd22ffd9f626f23d1c49e96c326eb6f04ea8d81231eae0ceb92e4eb0dc64a31b9308673a19a3d74c1437972e31dc277e6766d02b333dd2c7b627deaf3234baf1cf40a21b2e072c6e8be85f7563ba206ea5588fb9bcac9f811f7a00e4c8b02ad431e849fca63082fccf4679a433f64c6cd461bb65b155c10ac481eee7e3151c7dbd79f933d19080b208b15039d130bf18705524847a8e509e8b2be3d3adae8e2b49af653881bf525dd5e5fa25947c1a08bf8edeaaa62915678f9db4606184bc37f30baa391fb909be1e3620328a7cb86279f45d09898b7246ca8fa63ef4ccfc251c13ff6eae6e7906573571e2de9773abf3e4b6734371742ad017ef19a3dc3626df106a8f452ff077930d7f779e7a7b76e68a88d7db5b7ed208a16c8c49f0a2efa6ae88ae6160207261a63b7e446d863f8df1ff8b72ae1433bae750d5c86b6e1808a42586fb8188e3f2af7a14bb45ac0db568516b553f9d441a179b132a8ecf49457eae009dd06169c8e698a595253d39a2df1d33e97817743b98d4662bd04d40e37e005c04bea8e14f757cbe56c17341fa2e2080f9acc6e9fe7e105678ee186c36cf06774e4308f5849917ba00397e6fb32c33f8568cda4ebe3960f0c6bf76e5f298859d0bd70f590dfe2fd46430c2bc691e28df192d41b59c5c8a2327a418738d62f3583409d860c30eebdb7771073ba2c452b580f67688ddccc396d423dba54b4aa60bbdac95490f14d2dd9fe0e89ccc94d0cecf186bc437f32ae8718378f62a22601c6d1dcd2282c843de86a029df2ddbf00a77190c39823318a4f34da30eaf5adfadd8989ab4b5e82569aeb274f95b74e41bd50b384e4671154c63a9c478f6e37f2b3c113fa02d789b46ea1d672f8295f0fc5edf3d381cf1833dea89dbeacac0261e1b5eac66c83b5d6a9c2cebed4e804d7712a40463dc62764abd42b30c59c3f37cf6e4170cf50b4ad34682ce82ffac367022d78a9c509334a374f8682315509faa196744e2dad7ca3ba6e7e4df89b8fbe2e656f5b351ffceb0fba1e0a8e7d0c898668156d770db318952d0e7ad7365bfa53cd843825d98ee76308d3133da2aea9f5aff3cb34db1ddc8becb9fd477b2b88d6ac2954fe15fe1b1d30562e2474b83b12dc244293574eccbfe8112daee9429d5d00dfdb781458e9abc5424aa9ae2ef04fbd9e8911f353eb4c584f63fcfd9d3be0e085eb2e572f1b7e97d8af422cb5c7ad76105c9f21e92ab52b4b2f4766185d92e4a4c15851aaf8cc4b60c48af9cd9b5106c966a729218bda5944a3c64094e09563a993dbd80857c1fa1260036194417fd2c8921b4e88d87dcbc2b0188fab9df040b5c9b5775586aa6e7f0168befd45f137c466c2ebcde3d8234746cd50d45d3fe98c8b027504afdc97a4f4ae70010d69768b5e4fae835c2a5343b1fa4d6ae036a08ea4a427a9b27ae7404f2c535cd634184337a600045a1b24f79b6950c3c4c5ea367763da1761b76887fbf134df072dd92d1130917b82f7e34ff35f0df4a512f1913ad99b22724986d835f708b10f63b24cbff10f57282f2a850891e802b7e9f99c2aaafc603828d21f1df89c3b2dafbc7ae6579963fd35b38e88350c970d61b8429192a033e00a96e70bf9be6b896c23fdfefbd0c0856ade13c83d7c906723b01bd4f75f7a3af12c09821dd2375b8ec89a5cc631227cf575e75415bc263d93209c275a2c51cecd3f86af2a30834008c129c78101ef7e5dcd0534e2b37feef15c6d73c58e13f42473c1d4c5267e0a0887e1e0cf9683538a3d191aa17d183d54043474dbbb266cf6b99f2dc020cb8cf9aa149d94b4296a8c2ea7dc75037ca7bf5d8423cb2b3d6b2bc3a73c07311a7dfdc2eedd660c396e18e473b8aa0ae30292fb072a1eca9e85c2b1efb9c9029ae690f0a3c4812f9843a8d634526ded21ed7ec16c64994e4ce41e1bc0af3e29b317db53ba8ae8f160cc1bc928ae2f9c8f78882b11d61dcd886da152ffed008e38e806a6cc91c08d40baa861265e8626fadf263ea6676d935995f39ebdc9e18e65bda8bbe92b37c09b7316b7db877f2afcda65003cc101494a83325d48eebf141eba8f90fd88691939ab6165790fe594039d1b5fc875431d522eebf0359f757d631cfdf27a28eb13c42eef02e5d05fd2f325575c002f1918a0d563301cbe7863da6f04387fab643b98b447b7defa14a4ab240d88809ca7ab264cfd29ecf80e2178f27925b90c74a41b9690ffa8364073df6dd96d15f9905fb57e32e73326f83cf67a095dfa512cdc1e44d2c8401354a933c07d86b7206f5c09a18714679992131e2b53605b9ee64ab5318fea46d2682dfd26d71009dc9524eacfa2eb1b51d338d3e4c3fcb62787ca9045069193004d1d800865d23e610609c555d6070063ef8e2ce071e32f2673bc27b870c18597f9033960f3211b6ac24e42b34d88a7d397457960cfa980f33c988b214110784b128461c3d5b80eb3cec5846df8af19a480311b1739bb23751377f84a39141a5a5793453c6f6cbe8d18b4ba36c90db0e58dc75eca78dfea8ce9e0a6de72da202108ab05a7c2a6a52c12239f765ff7f04c0d62e4072eee1cd92e2bdfad308ce6c17e8fd3c445fb412ddc8e36266b6e0892fddb1c7719bd638d8996819bc4cad3321406047436627b23acf729165992ef5801945e59e65580285ad77b207e4fa8e900109a7848fa412e26e8e8a8de0a94c313ab22f6bf74890d60a1d94586328d496b7d481c981b46f5fea30af501d2294540f4283cd15c185db6ac251035b296ae0d99ac6e01ef64e3f4700dde778860d360a4e6519f6073d4e5ab2c793d9871b8138024be739ad2f39ade4c770f93728db3086577b24c1f08876b1d20672e27076904f312757399add997d1960dea548eef375445f2a7ef42e83bd88230ca80d236afb13e59c3a5e5a9d11294e18ad2335deb09b6b5c96ed75af8077dc402c5c36333e4eb22072065eba676f8243fa13ac9fa252dfe1b0996f7086defa25d635209ad7c1f4674cd56a326592f2c523596930b8bc58d7068a348d0f15055b70a30d63041b79d1eca258c39f925678b833a9c223991f28d5ac33b7c3823ae7ba80cff6cf510726437d16915bb0ae264fccdc1f7031cf245abff3b9d8a588e35f36f323e99d280971bf00af7ceb88fdcfc38e0b2d38445d94cbdb4e10d651167d7c7efb46ca13eeedcba30a3690da9e3af3f3c7801f60c64d67d5fd147b32f153fb931eb618f4d0ca5d6f28e8a01b193b363255fea024768c76ffeeec3d2f0054505def649edf54d476b1a729619ed81f3d220030f62b23bb9bbd2f137163e8bcef59a5e5b34e9de39c598b5c6ebdc17fc162b85d466bad3619724e6cb897a1c064dfa47c06b69beb9214210577729a29bfdcf70eed6d866ce73d18259dc883ea80cc27deba8bb479d4a9475c1418bf8969da429781b2b9b0981fa1231fdf2975321709f96cb968c845ef595bfcbe8f94b32aafcdeee25c598eb225f28dd2a85204f820fd7295819139e92aba285893c8a6e5cb3e63a0a8649ce442cbde5cd381d0660d0af9e7438097f9b11fde67bab58d20ee2cb3b08700ad28b0f15f2d82c945a740ac917928d23dd9e5c59ec293c64a576604d1b5f25edb3eb2217c5053a92abb4935ea359c867d29b31a5aeb84bc4924308a1f150f374bbfe4406581c9f3ffcfa943dbed6959d29751489eb0db67376099718e66c05b2dbbd1261c96ae69974873baef6774274d1cff257137c51291deff58c0cfa466b96ffb5d49e5f0c24a7e3a6aaa95e07d448449ecdb901f2f55c202af4c6b8bb64657ff9b59f14a1621eeaded7cdc3feee6570f9148ffcbac0a223294e786e641387c36db096a80238f3c279c693cf8d36b8e222a8fc8a8a4a429afe58881f76eae7ed92d7128bc9fce66a07fafd7068f05766fb33cc663c0f58d87292544c69e7d2fca029e507288a5b7d3493a6b79f7af9c0a0ffae4762d62f95be0e0c0b21831472b9706ec556cc72a11882b6b30990099882ad65822749ff97d3b291bd477e4aef91ad1f546fd5bb83779bc7e6ecd8bbe7352c8ee0fe15ea73ba3e682ff26b758f0d3e89b12c5e0351ec152192273fb3a2d91ed153c1191e84562bd4130830bb9d3e31694690405fdf4e70d53b62a650b3155c217d9cf947726c84fda26e135e7e0c7883f92aaa5e687246d5a0fc8c578c8bc281e9d4c4577840fe149eaf8c7d94539065a27442efe216951ddd393cfd7fa45d2132d212eb8f92113e3e992c66e9e37e5ec52dc95d362f73343c8db74255e4942b1e56878592920f55709de0c026ce647e1ef5b094460877ae958d668b86afbea887cead3718b38ceb3cddf870cc393a4e6626a1494aed4cda3af20e5195d9d2c9886e0345c2b06ce8509ef3e5b33e727928afb3f06d932a023ca11fc6bc4a13116365894b528c2ac8b1f469df3e1c4a7a704a7990dfb572a8313de058dc70efe485bd66e18fcc4aeec3c4f9fbd7ee89bf678e1f20fce1d4c3518e24aa7399786bdab64edeb20ea11d36abb549c964d7ea694b59dba62421bd43e9fe0a336070585fa4bb54dd0bdee9585d767b07229a215e60aa4bd609b54301674c6aa4934a3da71cf8f07b02d421fbdd2cad512a5a79f8300214eb56c9da0bbd230b44dc2684aff00067a70158beada774d8bcae8241db6465a2be1828b95fe1e1c6f01018ad797f3616568b2dc9eb1bdf0320fc49b40be07b0f428736a9b7e18620343c3d12ee981364389e446c08c2478b0e3a47aa37e8172e45708bdcff835fb972433a1d03d42547494d8e2d5337146ceb556c84d8d2d5bbf31c8cbb2b0f28d7dd097bf823e354d211c86763acfc92130711cf14902e87cb45d4aedd799780be61cac656259f5611b4f3a2c51012abaac4f59e1fc123d4a886006bc9122a279c72371a236836c72ebe59843433a2568e5d361393d41aa3621ab01860f534603b3b41820099c08c8231dcc7968dc2a171d048877b81d388aafc3da0f657ead8ddddbea30def0cac3fae5bf55f4fdb6cfe4f729b0680c3662f62ee5b212c0324079837619294d3f31bc6092043ca4dbed8d8b4894b3c39679a85795426d4de0dd11899ea8935d149f0673a8b10148be8e53b28c34b00230ce86e8069dc4256d3fb93aea6801388de9cc1002da72c4f313769a07e662a04a25b98773bde2b68e49db02dd47346558e55494ace37391ab111a7c26f1fa9d1cf061e27ec0700b345197ddd48b59298f2d89ef970ac35d13799bba182d7b5a26c5a642e3500a23d6411047116f51c245127113251a6f848f9c8621af619d2a0cd59a1abe93c55208f903a5a447985e215faf046f68788ed3ce5baa0dba3e6eea7c6b8fc4fda594ff8762f33dfe419e9cb043a771ee3894cb014c445ee5a99899e985bf073ed646d774415498ae4cf4e38be5045c0e1d5bfc8cae1a90ade71a51fb80c4ba69dadb5697dec2a300b151e3942477267b1bd58075c67f83023b4c0ad64872d3ca6f1615a5468556970ecbe5ae9d7d028960c4f5366e1e267d9d6146a1f6d05b648ad8deb92dc72d240fae5f0d4a121ff4fd4b35211f688e29de62d47ce6be00d40d8fdb1dfc74e8f45683454341532fcf637444aaeddcab3d0fa1fb4d1ffc4649c26fd8af55de470c94d32f09283c30cc4e37261b7747863e810c416b072bd2996c1717e88709a6a83138a9c4db04c21c4dc17260808267b453650dc53cff5d1193838c9de9f6a51be02e91a6af63a241c29fa901f7510702ce2ff550d9b5af27bfef86bf50e6a67df3d962206180390aa8ee69c995954e68cac40d7c5942f9e133a2123b671abde297ac40a6df34c209620c9e5ff5c1889a51c96fdc6ebf126b08ca27173078991e8a232656186a99c9ddad4deb5e42887eae5760d7e7b233d24d1fdabd987e0d56b2c8ffd8e6735e0ae4ccc1f3bb34c58d9ecf40d0e90d9178b97da6b52e72d3f51a8897753a2a6e47f07f1bebafb876def79fb50cbfc8fa9bfb01b59daa2ebed137116bf2b642e2d9dbc63b11ed123595baf9b6970e94b720d041400f8259eb672b81946dde3243e60b1f44e06e89d956f3954105a87c421acd26388d6ff5f691d02ababe23e32c21f1dbb25c0804f6b9241b42b2207132a8f450731982302efb58bdd8a7274317a496613f19fd0f2f680ae3a8316f0fe711c758ca1a914e20ac15482e3d3727951ae84d54f74c14c4585afd673b08a8bcf2c8114265ac7bbe4ee645609f2b0b3cbd3a349f4e158456fb8833e84267e34f670cdb22d7e758cf93b4cd039bf2b8bd10dad5854a22a3efadec1561409547ed099303b9c747be3a0aec4437432a28508c03eea7e9d200622319ec9d23fcc896d50c761fc8a16d24b139b695c71381dc68fbcca5f293add963a0b2900bb02854790b0dce468a78219a9643ae3f5fd9adbb603b2d3c078aca2940e13706883da3a0143fd1ebc071a51745a43a4956c4d105d8291bdbd077fdbf18875527a33d491f30acf9688522725d2279e3ee73ca998f7486bdf12ec23782f0c20924f215c33bdaa4b4f6b5ffa8450b2bd5c83b137c29c4de03bee422554ff8c6502eb7721a9565d3b4b7ab7d6581e0423ea7682af4fb007180721eae56ba216cc0caaec3b7ad7701dcf94e261a7d9749dda5038426e94347e52771fe8b9a2d1be3f87d2cf4c357e9121ba6fd217f51e152ba044ff0364c3ea9bd0eed85aadf8c3b6df9221698df03ea30a9a4ca7f35e233ca3a183eeb44fb1d2a86bed69daf3f0c5c5b421c5dfef5e841825187039044b4ce6db7669d88c805a953514c5714b6d2908d2ff13669038e640294c4a2498e08b523e1a396d37a6391660286f3e274f4b8f8ac56f9d2fa31dbd5d6273801570b58684dbe6fbdeb5927eee7638c1bccab14e0a86ef7359f4101bff9714c420bc0ead339a84dd826aad4bc4a2d63947ac4ae0e28134bc7969d726b4f76fb9852fca46c86ce3e9e109abdcbf25bae6be0cd0d0daae970de7dbbfa3e8902a71886838f5d3fa33cced02d2d6d14d92f93b2d3a48ace575bd7bedc08637d04b631a632f5c42e8b628bbb18fa43b7b86f3f353d85f26d03b7fc5bb0ca284a5effb45758dc52380dc3ac1f367c4aa956b7d735feb2071a343b4e7558bcd8e96fca102c3c45808db92398d729e50d96560674acf839d464996bb274e9b2f7d5adc68df1872712914368915807debdde20357188e09fcb8223499460982e0038810974953c13102fa9967355bddd70cd07081e86a50d79cac8dafe7e9cb74c1f770db02530305cb920f2f26443c08af4bbd813ec8bdbf5b0c84b41b83ac7fdb3bff050bfe4b44deab9093d5a76bbf1aa425b1acc4cfd1fb31e9935899c5f71234c38418c7142225160a44d2010a631177d659640495150010c926f16dfb15c74dcd635a447c8d710719a03e07b1263e1c2e5fb323d7a159fea157d8decc4d9521f4a3e453a581321b98d6b765b44230e6937463c34c3942a4e87afa151c26da4206efa06360141e113c3f3eb202a840e91b51f2347a929c4f0158031a17c2aece43085a44f2398badc11b5da4d757268e79013fda596b61a75ff4d38fc960c45e4877871b6373b3b9942387ee37fef6b305d962033c21b4535cd86288d26ba31c758c430dbf12ab248b99d817d43b41410ca061913a16cb1650182c844fea07af1040f67985dca01b723525241b8770fc16949c04a4f3c18b63db26157bd7a658bfa29d4184fe1c4f594dd45904548dc86beb1e90c3a91c1ce07b92d0f7c1f3aa84750f8f64ef6848042eadea0932b099c6ff48aabedafaef129a8fcf7c867e5c5dbbbfb963671eb40b9521f8a8ff425c45c7d82452fabd0ee5459aabeafb9b812a6909207cc6fe42253cdf66c270a742dadc14d0e4b521f3905b266d2067c5971daa039f7f6ad5e9df43f9ef622469649e8c48e44ad5989f3718121164ad2563d53a399bac6a190bb01428d52222a9b02bfacda096e12400b0ab728dd4e859c61e717244c36f73797914581b657cf36c4ae46a5d717f0351de1deb9da63573c157219782340c8fb17db524af7e71504452d2d1b2e5fcb81f3b18cf70999e28b8fe7ecae5f1d6cec344aa7a1e4d2686f0dc34ed387a9d692b07cbd6c85b8d883e4a09100c08189daacd5fc6640306a532388c8134579f0fef773ca5b5840a0447bc973ecf8b5d165050fefc45a1fd1bbc171aef79cf995254a414d2ac38fddebe786638997ba623074b34736a8ce696446d9b4ae8d381d26aaf5549bcf7d305759d006650ac9ed839694421b48b858f41f067f51bd6b061245575fc4f9b676871cc59743cb987d520b5fbede628f008b62fd0434e1a231b82b027c7e55b2abac14f8951fd895040d3266a0582dfcf6dffb89f355b933cee5f5b2f65a65fe9cff8a1b1e8515a27b836ae1cd8d543677e1805df33f4bd7a153eb9ed3a01c75f3f53e8161574bb6634859211d4a3e9f84cd06d3a20b311da8e33124253c1ec1b5bfa500b561723f6c3d9e89360dfb3dffe5caf51d28120ed8bf05f81d349aeb492ef37fb62fca18cd79a8b01bce246ef0734047baa6e69b001ef60e52fcb2fa285406a5304d1b8d28b5a1bbbaa788d9921b2d66a4b801eaac74bf9f67d4de408444f6aa86a5e1012a6d93a589f5a03a1cc75847656a4c3d4a57ebadb0cc56ad67945890ec392bcf3524d6fca2a362dc4610d2843d54afcff2aa0de7a1f1c7f2b47c91cd7799211f9517aaa4feaba917059906b6b71096585c4c8eaf44d915b74a7617494cbce8f5a8d8fb901ed9868880ddac1734b85612b44e442a894d881611a267a5b60106d87e68da0f3c8bd67d351765bd173d98cf24e2ab653c7c63d697c2a8e43c454230a4b78de883a40ad147c9ac7cec5b07923c70ae9a503fc4cfcc0b3ad79a57eff4119be6414232e754fef8d077a8fe553da913b933c827d600887951c54f0b4907f68ac261b818c23689dd442157566309efdb59caae82ce1ab3840352f1c9a93ab189c79a0ee88046d222f640aea58f108e7336a71c881167e4c802b9c4cc2a5688a6e153488874e6d659051e3e1b8c989a42bc918dad8799dbdbafdd45ffd64635e6c40466e04b1578207e26ade528c67286dda8fcb41a835016468008c65644647cd53a173da62be8bbdc20703ff3248468f396bb24391a467fb44ed70040796803882a0b5753ed606b00e911ea551fc5fbdb8c1a4ecfc9138925da9387d0a0f10676ad73ca19e3493a1083b4c778f9e20d597cb5a1cd3cf1c97a290589718d67f253287989f68c251b76a6c11dfd05d53f87c334bf2a29c0b9921228ead1e7b3d406ade420567597c8b99b8e60b6f53427b6ad340e9cb62c40fffa8cbb1dce040908a2e75f334dd5f7837167e27dc85371312ba614979db19ca3e2057e93fbaa04f3acec27604b61862c9bb0b68d71fcbfd805cbffe7e8e241c8e4bc4577e5022f57addb4c4bb18e6146f8553a6f9828f2bfd73f284f90d7de452454b2e752cf6e216ed599cc6612d5410440d326c66a919d5ac13c2d580c56c1ce71d0ff80845a88ca78d2826d14700ebd403405302e1079b10db8166481c80bac2d24509405ab37ea79f783f48b5fead295dfa8b71f8f6fce9be8cd16f4d72bed75a1d57b6c28ed91f8443a3ef23b918485630004eeb211ea2a0d32e5dafeabe542c19e70e35a8c317eb4b7e297f143031b7b06a79f51219fb24dd36f46ed8e0ad4101ed70bc66346ecc08003821e02be7f1153c64c249a99ffb9be3f27538a338ecea9b77fd735485c2756dbfb824ba84be40c5248e93694cecfcbcb3b7e53f082d179c27355e88df525bbfa34809f11acdadffa948a305c8331a1921df58a2c4e365b3599b1843d19fe14e4893006ada1dcce4ef75fc284b339bb327ce5c518c572d5112bc8995f693eb8a9914378b738fa8ae00382cac2067e496f5c2159a34b97fb8a47d162fad13727cd4c998fad545a332fd60a473ae525872da88599979d43d4745f5def16af6bcc6fc84665d6c2564389382d238b41fb31e6b8d9187c9a97ed5a5ace1a31d2d9ec010fe8b3ffbcc926ede9e3e396a94bc03183e10c89f153eb5e3a1989de0ca852949f3b49b5b6cc4e4849137fd324515c73d7370424ceebcc9278f227095f16b6045a412afe9133230c87473280d679ccebe84b120bb50858b0b9b5c15078e756bce4d88c75109f011648b35868ddc3c76906396ba208de8434fce1c2e327793435fe37944a0172b7fe42d1b07229004db830da87e248c8c5f438c4c1f689656e1c35587e388d255f8136103a4202e1bdc6791e20d70e7b2c9b29f489f7a204c48c7f70f8724da036aad45c65e040e6fe3bc5873c177f25cb2fa21ba6c8ccf3b02c4d1430b19cd09156259c5b92c599d2a0bb292c1cb01575c26c1511735ba2867b890b80b26aaeaeac068e0d7f9378518ef31fdab081b6657e5905e1e6b3aac1cb794401b342994577dc6f9d820c640527f031ad2ff1921fdd4abda8cb6be680c28b10ea75a69902e3b261f7d939ea6328d105a914771f26b753bff35da737c2b72317a6edd633b989a12c53086d35cbc477f6ac8ac9177e0678db425681a5335e3bccaec64d4175a6127e6b9099c23daab46f6353f4f10e3cb06137a8dc63ad1bd39d2e10165d93b7ae1b26dca53050ab769ba4e265999859d0689a3ece9e9e22fb0adb9fc133ebcbed4731efa004de469d025e2510568b956367f2db23112ac3162d0f45d654777c641ad78ae36203a675854869c73704d60b4ad46cb70cf13a83a6551f5a0288c5c8f45c46b988417a1e033604e3e5408adedc5e2a7aa2269f7b36e8b6ecfd6e94a8970c9b1734bd9d8a9255dbd2e7bd2b418a14a2b3f4a0b23122c52c1ffef984b2d6ccbb0237a417726866716907a572b2f7a8f48234ad3382e6202d13f66d06ec8d8cd7fb23e956fb79fb2d2b0760719262c7b7a60c98d266f1e9d4c4869aeec781dab8851fb7b493775267bbc8d0a856ed1e2a5965cbe709e9c2698ef21546c08471d8f9d9c0efc46cdeeda641ccb5d6e3230dd4b698f44ac1660387cd81cb0b136f389ab51711c5a259bcff77bfc3db23d20143ac767268771bffd68b4435bdea191e769fac76d1fdf80009b88954c46ed6b71c9c7f298397e53fb1e48fc5ec3ded55790dcbbb5a0c62adb673ef6cf46139c469d459811d3ae97be803cd645bf61dda187af44f7eac7d7bf281aae0d0a2d1a4e0e5e379240c195c04d6b327295ffb6c423d222499dbf92eada6501dd2c74ba5144167680d5c66527578f3f5083b529f191af3db7fa87375f3cebad492d5ee7e4df82dcf9c650b732c7de6be5c4e7170cb152a4b662661cef0ba37ef9475e5e6d7559618555e60223adc71bf123ec42b019855d0c6d09c2beebd41d4e781391839c4d1f12e54ee720d3c1ba6384387f97945546a23b227ff8513f7e86b203d9273d8348b11fa9af64b7664fea554c801a52625bfe8321dd5ae2ab35ceaf478e9c5c526ad15ca95ec696b8229bcab0fbede17237e5300f6989fe4bed1475b737129ddd6dc32a6f27f08f76c6c740f542a9d25fa59861f0ac7d96bd6c092fcc2dbdbb974be83938ae7d67d360b3a9c4b791b95d464f00df1e9e6c56fbccb18f9f9cdebfbc4c5f6071652049c32741799f44556fcaefc134eb5c2f65f4d42f83d17349e2ca42060cb6cde600c5e4079e1ec83c86861a2ccd3df0a386fd614cb46f934a45b40d09e31905e1c8f6cbe4a92f23f1dbb1b2378665217365f6497b9f6fc04052ae95d23e2a08ff091433ac700af5d7cba6630fe8cafade67966e4cc5f3fe3c3b674baee949ed819a46fe704cff13b839e66bbb9e3e87ebb90fd7aec44397e4f0c497a764cd9e93babadfa287f83963fc34ca4b292ff41edf90d45476b7048aa71905186c0e6e61a375de1143926d55ea81b7c846beedff2bb8175e23544c6184bec2100b79d8a54eed9e8713e9550431711064dc1e1e7fcaa29a9815338e251d50e6758176e460166401ca7f3bdcca74b8eba499649b0b15abf6adb1828cd1135d8c3cebb2ed8bece97eda686a41c95dbaa4e41b5b9a51a63bca7e36ee3a612e270c266f4de1af74bcf2fd26a534b44d5e357301258a812d8dfdbb1daf1516c1adffb37013a8897c7ce7ca974f1051f598bf0a8772a637c08962f72e7a90ea40bc426e561d1963d490118ea270a5ed0e6bc55467d229314a96257ac17c4b4a571523412a0c644f1ccdf105cfc1cb75c51bef32e267a87ed37394ef85cc13aa6b00f287cbca777ec00fdcdcb54e5b6d1d1faff59915eaff3f2979e8a5758b84ca4d04aafb0465270ecbdafb81d644da2d0dfaebeca76729037ec5ee297c662c8e0e8dab8b78751a045e04e7bc5c5bd3dcabc5354a232b7313da892d34a31738b6bdfdb6d432509e7538a11b3606dc1ba570e47df8732d07feea91a2cdd66386a38bc6c111248ea7bc2c25542245778cd7b194cfa689f1c91ac96bcacfbbf8d51b6d8dad42b82087486fc0c620d1981eedae9e5f0b4f7b73b6588fc51ffb97005a0edbc492b0e791b48293132b31748c03d8b03e50e563385c5c528191393d6f17999c5b8388b4ee6ff95ba49b28a6b4046feb088f23050008056e446385a7d1fbd3c8c63084ad2308bf82bc304b8c5192cfd98296ee118cebfec260448c8409f9c58307d1e17cca73c1e8aa3a7cbb71ebec8c7398f3f5ceab9e4a31bc93e284b44ddcb429dcb197b488b5729b8932490afaa8412c16bcf6d0a9c00bd3c4a25b99610b546aa91beb424479a0a911c104492af288d3ea967c3dd41dafc3e1561fb774812c723ae36302914d622ae2f52abfde26e9ef33775977b7cc62f3db269b7d69b6d2592c71f52c20cf318c0c0b7252a6b948fdfc0204bef769981ce9c1616bf2fd5280462c894904baeaf5b4f6f63a1be9cdeb77d051cf19535b7db3dbc4d2e3a5d6b8f13fee6f8f2165fd13d42d5a8bae3c62482453c09a858a8c9e251396ddc815b23fc1bd3cf639d8f7d6d17c6b4c1bfdeafc0bb8d08370931e3dc23ab513a1c1aa6bd3c31698adb72f0388615c1c2ace5ba7ca8b6b5be6aa40982dfddcc496d566e59f9d37a5415e3ee66b222c9397c0d9ab3e38656c6c41ee73e7303e2260771ee0326343e91f3d90737cb03e9a02c261b73093fb9f93d0bc012703f654864e5fd8a068e36ca4acdedface33f0d6d778350506dd5c32fc467d8f1c379f8372f5e26e8b100105c434fd72e86eb1824cc9d2755798477adede491fcf76f343ace0fc2fcf37e62fd905768e9e0889262bddfbdb055ddf6aabfb5db4e9f25eba6dd331c2c480c82e3c469251e0b409c96cba63d53feee3b8d79083f99063ad553a583c3e7f9cc1e5828f7183da40adecfe61ada0734273563dc17e022e6f0bb1f04385d4108483a75ee9d94ea5291307adae7a005ee8584b6b0ef54ff24668631a72a6419496fcb82911257c893b1f9d1527f52cb22f9e77562314231b88be18756c02cf64aed32596ac441ff7bc6dae4b258889735a39e0183e52d89abb4ffb68577d2362f0a8a11606cd5cb15bc41a07c82a4825a3aa045c8e5739da30a677263ab87985f5bcb9c916af8c7af11457f50eafaf80c005123cd1e6d3794e967d9ed675fe26bd8a22c19a3e3f7b7eef483a71acb044a0ef9d04c446336a31834775ce131305f98923a29d674e48d906986037c37b10a733252fe5aff7a23120771f8776225310cbe1c304fb3f47bc75885d1643daa7053cae100073ef013cd3d5adef26fe1777a5a4097ae5745630296c42c9d07c82a14a8c1eccfc0057efbe7dbf7888d963402efe8bfc94857cf172e7085f1e21c93fa5366f46c013d267d1b50b9e198a188d8e5b042c2c54028831c26d3c0ae7dcae374a2ccd0ef3ee1364a34ad02956ebd5b33b2e6965a74a72e66794587d5fe461ea8c539a7f2ed28986712bdd9c97914ba07c07aa743eed406efea4cc9cd7fd506a665421e07ad237c42d02f2155f2155c800c21366f9d55f9e2ced7e8d2081e16d2b336c1339183043d56c4486cfece256f2ea0ba65cb621567869473eea8c93758bf5073c30b43b64c0f8c678d9979917806deacf89a87e095452cd645270ac7be7f30716a12d4acaa3075d9f0f3d3452582fd3b01d80b4091da14bf9da2616fb96b8c29c8ae34d7758edcf4ca3058bcae2274240035c0435ce866446b81255cd964bb48a8c3a0f51ff4a9d513fba8c57f187090d8a54eb1deaf281a56ace0ef362854be8e81cfcd434da08377609bb600c5b6a77c6d50e3641c987eeb478fab1ca00711c65753f10f3b72b846ff050cc59a5f8629cfcd0bc099f6f379a7a7a95d03f265be3ba803195ce4e8b5077fee4ffc86ab9b9eed744229057fa61c7e71b6c33232195d6400ae6c11c2321c01b0fc15ef6a045346dca6450f384102faddc7893b4c26055abb816ac5aab16d7eaf2f197e04df48f81562b19fd5c8637e87fd85df48127128b04697a92d4fac709a703859e1beeee48718b11426305be638ab797345603774480af24c3d2e4616ae24f217f88211a5b41a2aaf221850943b4993832cf0d8baffb860a0db93e119bc49a849ef8ed195a78a0ef2487da1e5b4ddd3acabd6860fcfdc12535a80467a513df55eec226f3aff2d13e9034f0582c333026117c8e02de3410384765384eaf715e4a356e6a30f263ae8183b812589d78160963b1554c5474617f4c2e89e768420cf976fbbda8e3dcbcedb24b82d3d2c875a71c9741798cf50b140014dd4912256cea18d1f6863af7189ef321118612dd9a136ee238f4c9a69f5edecaf4da015a333ed2ec03711de0dcc167846031dd52b4fba246072794b3feb255a5f3a1cca8fa2701ddbf9f88b44eb67b150747ca5362869410d1db2882d0fb5924a95065371a010ecf5c9f470579586dea37c888dfc153d307f8f60847b5af5317a82ec242768732295ff546e18b2519f273ae9043a7b51acebec594fb4104f7c2b1750c39d4a255ade828e410add92bdb7ff69897529dea29806d0548cc098c1187427724de3cf1af6777589953cb6197e1cbb546cfee0f3ade960c93c00bdebd89fb49c64eb512b8633d1b62e12158dd97d06ff0e90efdaefb5ffc62689ff68033c6c6b1a52d7c912a65fa451f1d4b5ad0deec7122cfcb566c1f9030afb4a2d135674a693e8c59268ad1ed26057d2bf84ea0c57eb61582e1a2ff56de22e6ff4130b648466a89cb3eec59ada3a73c654c708f7de467fe98499f07103708e2f509985e49f1db473173cb478ff2bc4fbe095da7b754e73aad71d2b78f63a3de5203900412d7a9e29f3d7760dce265c923715f8801ad68ca9d9ab519afa7badcc54b7a735c5fb121ff0dee8021fa0b441d4cdf8190e8844a86d1d065f8b116eb6b052be9cf1a8bee469b0c40f7ace60d51c81a217befd315677dd65687eb69803c5700abf2bc49a1610c7b137c117319a3602cc99b53390f426257a0eca7bab70b5b08bd7904e48735c297be2fe0df927e62ed29d9ff74d693914bb7ca821111e235d10ac50c6baad9246e99f1c3fa4f87d06b54b91300e78d4babcb97a68e7e514eb3ceef62b1c4342b7cc295bec14eef340a0ffaeb71211ac576e884153cce61aa69eca9cc7883d7d44e1e5170dde4fee9580c550d33070ef7a546588abe29c2c181cffc4096c5f7181a5fcc2fe8c28440cba49baa98402f74596387eb52235d8b701e36db1c7a3bcfa98d167bee05cf6bdb309f94b959fe0700cbafab0c42c28df00176a28e3c7bfd9332bac6fd6bfde90a14655b2e455b577b5d8f71a183538959d7d2b88de4e2ce6bef72decec76efb6626f54b4095ce86c9b475338a1238d8a9f25abd55166c11ec34d63fd203e7df43b9e1af0915d06f2845c5634d861d274ed868034516b9a65147376b6d2c7afc4270c229bb58979841581a2809db4757bd54a952a77df2257250c6d9896bbc101498e447f77a41a069c2f6021f0ddcaddf78c20f1f63aa2417955c79b03b3c7237ae1577033eee19f0408e272e7f684cec932bb014a1fb4c99130be5524e340b8eea875d8e66321ec8bf2b4630f1ccdd675208b3ced00126876bcdf53bf279e1ae75127ccda63fd8fe4ef2b9d1ffb4efa638ac07b363d40f65c10e84ea507b2cca899f4e0ec61b7fa6648230ee5a112eb8ab55de67cf02d456df174ff1865c3f7c7c585796458aa37146bc622ad287737378322c02301f5d15072f84335bbaf60623a33fb5ed45b7e948f8cc38cacfeb8836e3a48484e5fc31b9e93fd61b6953f1dd70222670ed506de6c4edafcd883c1babc36d2e6d9b2c74e207d5e67b8f18c8517e0b05df1d0b2739e02696613acf25bdfdbc03aa580b9286ae6641299cc1711a06d87738acb0f94f2f5722539481e156ac096e52a18b1e0abf2ef14069b031db1b493bfc592bd5fd8c4531c111e8a933faa8419de7082fe5157b89f1f0425fd36c52dcca5fd4335dd0fe45751aa58f0c19740707ebebe7b85bc4cae816ebcb3475c4a4b71734dcfd62a34b1304aba536165745053d24c5f7d5c33308ee4c3198c4bde9ce247764569a02e73cc4a5b7c3543cdfc17da5ef09284d9a46e0443802c17daf47cd04f42ec91a982c9d4cff9023d933f835ffcca7501bf7a17c96e56f74a088b9d938bc102e28df59026dd87534ccdc47e7b5e7afc3493ef19ded453e7f704f16df1fe2056c600cd7e35a5455544d33188815fda3b4bd559ac27db21ae2ebe8ebf18f95fa270a5138abeb4f7c28811a37dc0cba123963306923439d831497d561a7c48ee6c48374f9251691e3f1c7ad21fd37224d733ac027d975f744b70cf6d970ed86f8c0996039274b3b598531084cb783ccb32588129622cedd4a77d7ce5b928decea2e13356eec89e0b111cb0a849576326e0a32dd201474354200b3bb4fbeb23a9bcfac7dd923c79380305de8fc4f164cde141ac4dca37704509e0b83d4298e7bc6fc7e7f8a67203edc9a9913c44f46bb1948dea4122c0766ecc4a6e3dc554c5014088fe9ea37f313a2e2449d43ae3f9c44216cbf36872d2568895ce38e8a541d7574df24fbe1a9bced81ca102a8940318b426722539df5570dee0648a3c98625da1eac38751945c7a3b790523aa64b5603f712dfa406b8559adc416cf4786d0c9fe236db3fc5510dcdecf280ed854c226c74666093f25eebdaf76ccf969665a8805d5da1427f4cd2958420f4f77cbe9b030890b43eac00072aa9003d74a6d90037ebbfe4190e30c98a6cbfee53f6af433b8dc9122113103b1ed64db023b94ce3b86bf3168376be6bf04c54c2fcd8c9899a8028e0e322e4093496ebb01a8382edb1f6b666b06fa8ea93199cece8408b1c41be53b2c4e7a36db80fd09fe31fe8838721d8b173fddc44cf7de5fe9cdb219b09ede1f18b29894d6dd593e26a5e4dda846dad0c0d7fa28daf00280d7241dc03a8b8c6b54eb69f82b438b079ad14b5e484cb9f64fc2900410152798cf48fd213ce250da179e9e271c25f50b644acdc5fd10dd64e1500a92858341ec802db0ce7b297bc39cfb03760ed0944e2677e2c2e12e1d9e2d1400500162a85cfe28adaf0354c50c5664513d1e6de18f63c86bdead711836a4d5ca5cecd49c3549988538b66dd3a7d320d3b6681b03835b9cee6851bbdf81a86014b615855b2ed969ae971bacd3f9048cb45b8e4d1f56f18025cbede03ef29f2b2d35ca17a62f096146e78a2241b52990f598344ffd12faca4daf505362f2b2108b33d4dcf078fefb2dc541dbaa97a9b5835af29be7bcc550e7ac5375fa053f61505362fc149159dd03b9a4a3901cecc9313c5a6aa17f3d6c23090ef3bb498760d8ffca388c3ab8d8a559dc68da315acfcbc7185f5fdff00dfc81cd6fda54e847b9160d322ebf70804ce5de1be37971a801422419150355ce5569d602213bffb1635c4b7d1e9871c110317ae4d2ec8646a8e04f417ca3dcb4030d1ebfef95de5bdbe846733af4f4b300952041747f57e41e67e1e6568a6a42642436eee37147cebbea2628adbde44622642f1f8ec8ac60c3551c0104d8e15a02fbcad4140f55252951ba8f1f95396669577eacdc7439fcf8c388655a68ad0ca4cd301aed7f50c45b7f8af689ee646aae4975d226675e67358d5233bd721c9326ce671c2132f7c87e37ed2b80469ad711150ca6f9616a733791a2cb8c49356cfe5735ba9d8a852b2c0e49e3525876cf8ed556612974602b4b12694d378fa583a06bb46b57b7259a56ac0abedc2598eb97ffe7d89c493ae34fb286ee40f8f69291bc1c79de5d4a1b17b343435facf915aa96030ade1f3957c149fccc6b488e1ef0c25adc2efacea68d7f51b9b863f5811e197d4bfe93835f42d2e9455d631bc6131568f21e02e48119ed14e1758d39fc5f9753b18cf478fc34e1768eeb3c30e248a1ddfc5db7bdd328230d3cea6d27b80a7700da55a55de33bd01c4a47570dd99b1a36490cd04835dd447b50d6f20480509b92c2478b41e31c87eb06767890bf67f196c7f8c580153a39a7b8b2521e9f9cb33cd20aee6d38e5ddf2066b4372d4ee9fc565039392549bce0a36109dc1e6cc30fd84d71739ad22a74bbc795b4fa2204b75956ce2995f9d8d89840843e94f29e8411b36d75d41384e5292f588803c4cf04712df977646beabd2af77e9af1b8f73bdfa41bd61b410ea4defccf8471e5d75eec9138f0866572065784ee6f2d36453917374c5cbaee70611c5dbfc62ca9fd6e09f2e4d7ef6d19e70c0e9d7f6046cde5c92b771a083d2a8c23f287f8b2cb405e8015e6786c8385ff15ffba3bb5ed95f0830e94881bfea19f75484feddc574e6bd04048debdf40cf0aa4999346498267f1a41f6f39d8ef3b41d6df7302287e976df4db9633c98d65a85bf6f9197d206298bca55ade29f6d8fe654a817ec1861a2fe00868954c2bdef578cc9cdda7426428db18c8436613250488d1183e15447e4d987773b1942d778bbeece50004ebe99175e3d6f7533f623548ae3d208948a99445b8a08bc038075ee25c188e441011c6301fff3ca4e5ff939bd344bed9f29b8e811f81387d9da1c8f214382ec2c63b4fc4a82d31f26d1ceef94d1fc54b2e71644c9c5b72178b723ba8cef92cd42855b900bc3557a3549c02316bde3bdd7a2eaada26a2b17d4c80631f45b8e9d663054dd8f7cff1d3822e0c31caf0453039f05d40cf91dbe14fe61c5ad92e32713ca8d4d8225dd1e2d3729ceb0a604bd27e7b0a97f0c52cb4a57a7c21b8f7fbb67f265667bab774d897b96154d0efbc34e11a6f393ccedf35326c68485829cfb7a471035c864e574b80cf635faa45a12ca7b5bdbb1ad14ff90b484a303fee271176ab9c3fc248f0e267e99304b76270bbb3b520d0ea95477df162697cb5f5d57264e6b5bed92d449435fe1e48ceb2b2b57fca45e74b418b3865249f1c6418c7934146d76b4f1ad974046f7eda77102f47463804ff54847efe7995dfbcb3ff2c5eeba5e5077247b0204772cb35c35057752e4394e7d84c3233f2517661cae49886a5838086c53b13e895fe9d8c93474ef4a5dd8738f1075956a73e672475fdfc7edc79dd95283fb028de27c701c7cdbd33bd13693f5e654baba3b7900c7b0bc6e0cc91d01899014973f65b31215cfd6eb06847e709116ef4771cef710781e21154cbc80eaaf3c603de443dd32773732727705ff1124c59f9014ca461a6f64f1cbe06c35f34e0bec43aa425d9bd3e90d41e3dafac8030d913323069b392dc49ddc74bce83fb9f9a6174fa311bea85d1536139f752337a4254ddc53b000735dd9ef01c57c0604639cc08d4643940c75318fb6f0482daf7acc6c6e91d437950e6bc0ff5fb4f3ef57591964cd55c02e41520493cfcc6a2e4460987ea6f3f7f6a6f6cdb67bf556188a890958c4a7762fa7af112f85454f4e2dafae3308422050c4545da73115edde54b803a4feff8cce0d75db40ca2f838e7e1abff35bd26315f06fe496154b43d3e213f2ef5511ce3f5603a2f0aa1ea991587af892d7d91ccda642f5dd471f2a8bc2d6f74d0ea5f0f688716eaf8dc0c9457c22afce8e97b9a57ccaef9850e5f80c6e5c2cbe3d14d5d0deaeb26d3b429af187e2763c373f537b136d5cc76415e0deb750f6bb5998fc9fdf081acb742cff6d4b2d7db2f41a5be3c32bf6ddf0cb98ea56329553bcfbeaa0a5462c84690c8265a2fe7c6cbae82185997e469fed793b705b088efa3248e86f4dd7a80205605e6255e38a732e5fce3f0bbe20468037ce3247c29461d97691a4529e9277a7cf622805dcc37f3a4fd737c46d589e0e0414bcb37e938c77e42f19df494f99218ddecd41f639fbfa9963515ab6f69e0479cb839389572c6aa32016efe0c5fa06eed7001d543e886acc30a82c6594e5e5e8d2c57e01c9f9f7a3983edf56cd50cc48d9b753845f8d876ca8aa505f038ddba869040d2e05e0a0e8cbf5e7bf9e97ea6369ebfa2c617bcac9a54def4795cca5ea96f71532c030980ee6700f9955b17402a42877634c1ba43d7db10131fa3691b6aa9f72bfa8a98435e2fa1a0bc7e11eef62d7eafd7e5c1d8c8c8d22dccf46cceb038cdebdba9c28dcfe94392fa3b4e5b7a48c51468507814dd7b4445548c7cf5e84ad5d823a8bf6197da47037ce0fd937cf7294c6127c5746448926f6312de853cb4b7db92da238dd2f8263b39eb1c96a7567679874e14297062417cd53ac0597f78b8f87b10022b693355d01a62d742f15343391cee36bbd7e1c697cc5b29c2d56987331dfcbfc08d2dcfcb390754f31e815658f2f9a24aa1eff72641fe6990858922a0c567d0f2385cc0dcca051a68688347a10250fe4afc37e8836ee95cc679f03d8573750c93a2ef63fc5d6d20f07615d9a7bc1fb38f537c98a34635dfe362bd491626fa20ecd0c8ef7ba8db175118973319fd283a7c21b0c83ae24406b724e623dfe89fcdb1382f57576949bccbbd4a98e4b42571a1d3a3cdeafac85d785fdeadfdc5016fd0d37ad281723cff7f679efc0b95f5138b287bb3df229141ee7ff9a9b6f428bbc506502da20ed7dc5a4fe563991c6b4419ef5a1aee7b4e6938f7451425b8209cfcb8ed147ec6dd5ecda1977568e40650960d9b8dcfabab5f31761c2edc20863201c01d910337857003a9f939f647b9697e3c16e19f966aa5073250c51e1303a80e65a2d20450f0b9caef7069fbd3ca4d1c6ff0f5b646b97781a285d4a53e5ee89c679d1fb650a84670891dcb6e826fc7675290efa0cd9f867444a97387fa15fc8e839d9ea4ef59c31080ef49e7e555b9979ef6e999cb51c4b4459bc9609517aa4a83e7f112c0a5bd8dfea6a6043ea4b9f6ad8c42f29da5b4eeb7fe20a686e4a2ac10e0b038cb65bc4e75f1d0fecad3ac770c01689537b3a4242b737a6d65c870502517b591b8e44368e5414ea31161b2fd4227af4b81ad19d9fde298f7c744ac6c704959e0e61c4ba5bb2434eb2771905e5043a8614282301e071bc35b2867b01abfaf099d65d8802c54ebb3ac256bd7ef1c0025ad23af1079e9c8864042cbd25922e2099c0ff37288d115659d91b8db53f77037003c7565ebd1a516b76be6b4d0c5a39226b6e67bb750a0161934a3213ad1cd7abc8ec6a15b2d96b10a7ad68bba8711a9170b49213e651c67da7fda1291e4d474fcfba16d14562e9fe600aea774ac739d0acce15b72438f14aea4523eedf76277523162441802445df34472843a26983df929a760f5ff0514abc4360a250209da8d6860226e437f5eac6c9977577a15a1cfa26d4fd00e6b34ed4d0db0d9aec12b691585c3e18e60067e30f1e7cbbc277ab2a2e7985b56f105d8b233d54dcd7fe235858562b8f1f200bfba815af0cae5e2de17b88523224203688b288cfac7b60ac416c86eb716c819d95436e5f959d98e6bb4b2375a0d21adfe4183b85df6b3c274495972e86d1e61b0ada63e0d9051db0849d19912be238a216a23b38906426fb534bb1267419111bd3d0f48a49526888ca5f779b078e1319c3733964e55e651befb163dff2a9e6afe90322fae8f3688f9dcd2a4c476bc4c28c03fcb7851e29a14139db73ee105e44e42f9e61d908aa8d7ecca99019a4535e399b9a9138c4f9d537b07c93698b5e22deee2272ceb9e4cff01ac16491d4ee704d07da3a28c6aedb028e5a6770e9a3be7b95349378502ac4ea06b824aa516095d00906622f6ee23af2d408549338e5756571f3ec5ece4f3173a507ce8e02b32ac5c89f736d805cec48508d03fd21a275cd7e4fdda33694a0fe6537cbb5f1ef43d4d10572cafa5a8d0a4396e1a69271a69807363d5bcf0d7a8c1f3db0b752ea82eed5215f4e901c0768ec78685f1428cb5551843b654de74874b2405dbb7cb3c10e28540079cdb093218fa328f1b8cdc21a7c87c9d1e7373124b3489e99b0e760957ff8484e30eccd532ed66eb98e2442faca3cfa1da6beaf1f317061246b7356fb2f81843a05be2f9af5a06b906bb58d4fe5f48fe0118c54d9580a0d271c4c3b08cae6d79a8e545b6546820e0107af83af70bc888a129e57698cbf2fbdd98d977375df239848e4462e6b66100e457e787326608979160506dcf0ed4afb297aab6fb62ef4210ac1273cd14337f497447280ac58c1a6f1246dce17846c8824c01e46c97985165cf0f614045f1ac352514a62267c24250d895c0b7ccf7e3a81cdc543f583783e07ae89e1447f561cd70bf7faa40a68e2416af44a9afa41d75d887e5681c06620f92130b1d43296ae3d991539e9961452978ace26e67fcc3ca374045ee577607cd2711be067328ca578082487848265c79024ae5b5e0bac9c78a9c445da9fdc987c498f3e036f43099837a70200c20e406c1decde843e323fe8641e66ea39869623a3a4b2bd71785709fb43eadb602f6003415c9094c32391f245966d46e4944ac53225c25c0741df7520790ea651e0e7cf67e061c477f90957359995453e058062f5bcb049623a49fe455de1893f7df6b8673ce50214dfa8044faa296d152a894396805c9df2138e30122f256dcc158d3bd15f1d9b7db816295068b75771ee13c7110b37c3b8e0579815f8c519342b33801c6bd94e7ad112bd3a1c7b8d8c73a508ccf6710c207e3be20d252a68336a98c14d8e5ce09a178ec992e3e70d1daf1c6c747f0c938eaeacb6abb4d292b70c53db2b8856957d416f8229cf8879f9cd9a5053df0d17919d11d289d4cea9981ae42a81b0287e4c6862964988a9983af02be004bb52df22fe7997d9a03897e57411289632fd8ba7413b0298c4b161b0b8c235d2ba7fe3caadb90bda1962c44e90e93b460d6ad3daa8ebe47366745e015f5e228cb92b1d0e1d6ee5247dec30d3241507e0a122efb89e6aba7ffb53a2b77a58c1526757dc935f3029e2e7133d421e294c34603c4ea4a2885e1db6424783d3e0413bb41a3a95554e40c2c155d45bdda256e2d61f4af97a9792ca5755a154bf6a4e5bf9ad8b9367bbb0679e17fae0804573a7712cdd47edee6bc855969eed6a9f4d744c7ee9e3f81fb1bf2f4fd36be604d2f7c48be8e94b422b75b305f2b2782aaa2e25f58bbba70e2b5dfc1756680707c464221c4fcbdb3b7fdf4cfef753aff8db93abc24f0f2369698c928a9288b0a61020efb8655a6d75e141a360c1a7b67d319f4de53b04c7d0ddb718a47fe36bd01ca4d808f19f1f292dc0ac5ced81306d3768ab3a91c901343c5cca35648b2fd915ee2a4ddbd7bf8267770b1141c7f7b52f2bff447d6fcd54bf617685cb35efe7999b26c189a687c6f01cc22702a16bae5c6a862f016b7bf33ebb68f0d6e6a57f8980b94ddd7e06532b085d9dbb57881e939ba8b6ae91ab8ed2cec1060132354ede0648529b68c930c6aecb4c002e0d21828890f64f4b52a979c7fd5d0c62e59b42dbbfd46d28300980e14abe292770d7b47384c69e2bc2d82912be1cfa24fc02909871c5407e49e8ad8030995383d2b2fdfbe267613662a0973554a14b4f0b096a7481d3262019850b647167f8c89215dbe1bb25009edc29190249fa6e9a4ab25cb732890160b8306b88cc1ea98bb796e9a3566bd8231900414a107790737c5eb8cb682371bf2e3609da0c47f355620aff0c7a811b813856b9e66ea78db49799c85b2c9444044d903a0661d8fd9952197caf3def24dcb3e076ba0181f8e10b41acefad3b76aa7d84db46f7f70ef452799715aa4337159919e0634647f3655cdf37757cc69182f29a6687449e363df67ed80aa2f374ccf6c6f7a7d5ae80ee39931eeb1b381ee6bc4ebb7ac006246912d61e5e304adedca2c0886c35e82099edce8f68b9195c666e9913b30a50d65775b5b215975b7e901fd8d27d6240cb9d4ed0e365ded46114fad6cbd694a0046df239777bd3d6b61a29ce223f23f8277e642945d8266d72f51371214ef32d31a9c1e395ca9ef71c9755a5a65f072d53a6d49e8fe440cc410f040034be14dafddc2ff6419bec6ac647634dbb8bbfdcf97997f71cfb20a6d7e41f22b81418294bc58c9bb4dd45dc26db31f68b36367f132257fb889ae6ea7b83a4fa4cbe079f8e7076cd9c44fb21154c286b992cac013abb0027671f50509d1a01d1ba20c29c4853f7c8d978be6866acb4b40df8f90d4c538442d109ec90efd9d162e71e3d82cfd8b9afae7d97da034b84c7b27c0ee59ff6c4975714ff5c982241b3036f7719d83dd111ce3039e7dd748306078d6507930881c6361f14364b39baf125f700d68934fa8339e16837785f0b1a7907e38f6c7951d8d703b340228963b90213dd6c472f689fc913bd138f03853f483ce9bdfcdf5574d7491bb8c25b467d1287ad275d7cfb3e96a8c5d2fcdaf0702c1361f3e07474a1fcd494698f3de83387d29d3988b588a20aa9237ac0146558db8a8aa78c6319dff9c21713b6a41ff9e40cde65db5b13e792748dd2e73c4f8887998d999d88245b2307ab0c5e551bc40b8a9f28f7e4012ba05599cc84cead168afd1f05ad78fd1e37d8746453bccef630d103d4e514ab0a35e8c9595d2cb815bca51dac3d3e411ad662d976241b30fe07b91ffcf50fdabb2712d92f136459c4223ff85e1d26da34868257a2fadc43aabc39300f54ef829b4c5337279ee143d5e8be675ed2d996ffa1ace354083eb6ab3250e2399a02c161ce6b3850c3a7acf13865f1c1acfa7b829575da5e1d008440f7ca915e4ccf0dcae836e4ba1b02e4281f43b13744323e80ef6efa3a0fdf4050b026e55410f456ab37604496cedbc0d97ca1e84dc2ad2e66d18c6efd3963288a92bbb525d87c9abde5de41d777da86a1d47332bb52f528cb9243a137f58a19cb20c8dba324bb2546c0e0c4124afc2d12565b4c857e29c19a8e065b36ab3005a72137c5da2aaf379b4bab5e155f2acee315697433d4380e7372dddd1f2a9dcc81c16c59dff5dbf36fd9619d745fce857de7badd7d92b1dfc8b69273a38eaa227454a3000dfccaf9b4bb279550d8851759dec3de231d1517f5227b0687731c3d4504923d6db9c746e6b3525e48c8a4823852d372b6250744f9ab285264ee3f97dea5c7b643a6a52862a70c8a8d45253eccecd2065188968aea1dd4f5378db4118635ccecc48905e249ee78447b16b6a9f456f07d535e8a274d2729890bb2e319f62353fd3cf2cb32a090727e249ee7b225c39d9915127f5d9c967f074c70a65964b4ca991e41f12a5bacf3e1f2e60f13a714a725f6775970d13353285580239bd3d2b4d19f28271840b14fd448d490a34b2cf8bbd9575b4f5fcc69ea1b9bd63bb44b6a5d45c519f287bb9d6e6977779b35e9d8d78cf6453fb7f377b8f49a35c2f94dc2b9bbb09c1450a0ef2983b9a001c4437facff8d553dcae180518c2e8f68ba1c7d52962846ab1c288dce82d5b633edb257973a3d987fc7dada3cf5d5876857210eda589046ed20ecaae170360ef89bfc528a0b95bebf68d97017dcf91dc5541b450a27001e5e1b04bc05c0d6d646deb564f01bb78dd0e84989b3d59c6f6078f7398ec8201e0ee675c1e3bc270ff74130083f0175f9231dc29f3d9b42f601c8c0b13877a638c4d748ff6e2d9081d1020e6323fb47d16d8bb454e959004265a0e8caf10d20c5d80fb233806ede5fdbecdf19268d80f701ebf2d00ea7e289a614cdaf193384b4090315af0e0f0734873de13f894aeccf10715464c03f03a47c19f520df3f7943ce08aec15302227d06097a15737283ae33ca0bc6452c0629b5751081db907ad096eeba4b8d3440f5b0071c2c4621e584267e3c864888b72cea016a32a51e4f36c5bad681c32e018ddb3e79805899b3e759bfebdf835cb5437d4907639fe5a83525f43327e62ff1a9037804a08f5ea197f3e7e38b9cba58e67f3a94a182c3782beff8b8f159928fc114557700eadb346876b7247b4ffc8ee9d42716938885df0d2bab371f5a830920a4206ab949a1d423c4c8a1b8494bb1c83e766777edcdb2c39b4db6e4c1aaf6214f217eec2679803b6143aa15cf4f66f69565c7d5155badd519e775ab0023c3941eb5095f7ae13c562bc1a486d3588b470387e78ead01b856837c219ba9def6a32ff67bf449d8aa56c791f5648711e545250823acf93e175797751030262f6186bb7a4402932b4e83bc225e5bd49979a0ffffdf9af8e1e87a68570b707cc7d1fb9fb8d7eda801474d92efc21d269af6d9e2880414ec28973f4c23d6ac5ba7c922ef743a9aea6c7fa60cbb5794504d6b67f94b55c47a7e2b33fd83498eedcafd16da5cea63b2965c3208bec6a76ec2a48db181e146dd22f4a22871a12e32f66e85c5cf4a3c522fa65c9c5e6e8e760b7fa9d9a90bf804c529cc7d0379042a3cb35afb814b89716883159683d74a94be09546c39c2b5505f10d35f45ed8d9879f9742facdb28a6b0248d682797b87ec13b9e0b4e8a906efdb9312d706d9d7f97e14ba3fabd86585dc9dcb00d07306615c403d990dca0b65f76e97e69793971bb12c745d177d463146da114bac8b4bd8978c18211f0fc97749f794161e39fa5df2108c94996c4dcad8a3c943d7751614c171d42b1c503befa33c3146d34620926b261479a590a610ef31b398532e20726c0e71dfc50d1ce6c975a77e4200c8ba0f5ed267dc844c22816e58097184cc6501dae05f2e08844fb6717669f80c761c7d2f150572f088ebb62047a6e2d71d8c9e145d032e529770b9119931b4cc5a5585c37b8244b86ea828a15070848ac4399962b4cea6d23feadd67f5652df11d97788b8c59020b11d41da27b4a6685caa73ae4aacde3add1fceda0aa1ca6047d19232d766361b0dc463d1066c3823ec51bcb4421ed99ed1a39c7fb5d336f9c2f88165314e7b9112b99ff3b2b9d068a34a2b3779cd97f7e96f6de5d721df4650f9946383cd44c0720044b989e32c2b043c907edd5b62af5fee2026fa0f1f50aab226ab60f481eec90a7c3b438f47a3b728793d18284a2996940c4025c9644cd463605f10213f8d63f0a69c356399d914c7606eb173724bc3be1083963958835a6d4ba99c6c6c684ad2ac0b315a7de4c69c9262cf4407202633f2bfd768568defbad53fefa54badcd4f8bd9b7f8c3b0bf811d7abfc60c19e08b42862e8cee50ecc50717b67e6c8d016c1b4cf8bc009c2f1f597a71865492ca849b08f8169f28b79eb1f26a5abb68fc257229ee4981c18028290b962a09bc3191adb6053ec7f27d7f53652879fe964bd0360daca9d7f29b2b72178824c5db6eec4aa874b1ed122a0ecbb8de2fbcfc666afb93fc8a7151108d11eddc2da4c741461f64d1918670a3a05c0cc281acdbe7fba869e29fd43e3eec467a7c449ac2d63aa48e1e49a77e89baecf0c99879e52ce4285c086ebcdbef2abb340b4ac38d86c2d164c9e55af9540b715b7a9512108861b66d7504906b68c2738d0fb92ca45e288f84402eb307346dfe6b5079809a6a6870fd98ef746e91488dbab3e83ba25c87140f01d5593fdda7bbbb5d0476dd6483e0a7b66da32c6ab05ff4d18442fb6ba2c2137c6a0611c65a0b5abe08d7ca447fe344d356ae0354fb019ca79f92d350dd58b5faa5fb796d1e651c91af27791985d4c79e1c3203c070fd3714036ea7920a14b43bdf46f20db7050cc285be70b3302af46d7793b31abf703b530d50b7bc5c4e6d43b774d0108ffa5fc76d7856edb6727a6da37a803ed84ee4ef38047093caccd5b70f2ef4b7537eefbfde733065c43fc5adc6f74355c3d128778146133661c9965fb257259583755bca3282acb1efbac4f16da6234912085575659d52cd358c9f0b6bd0cf3b0b423401935bab7fb3abaf32ae5d746c5df15f7503c4ad178a275d89d1a652993463e6ec1427a82fd6f671711938bc7a0ac8492e0d6cc5200b95fd200f6ea64cbd6a394294f5699e385259307f7ed6102bf984a788daccc5b5c9da342ef857541f9f6c26b9993ca20927b4a8e66aa14f8940e5a261230613f77426421187d121ac845ae4fa16806f92fd86671dcee2e4e711c17879365c7c25bccb1377db58643328132f5718ca6e6a2bf063a0bc72e9fc4fcd0be549cbe830a0b0878b872abe2e1e2fb2db0813e8288e38ed9654c2ee89758448ed8df34296f18ec44cb4a80e5eb222aff1b8a4219f14cb316fee9df81bfc8870a87da40f92cf5e735f5efc3717b736fd39da6104f53d37d7e8bcde0cee5c61b89bbe5ae6c293d272cf03872cbb9c62d7f5facd661b3e98e998c8442df4232c1a622d947ccb8e9db859fa6dcd09ac8de9322dcd1428fe578baebedf069b797e7827e733d169739c9be8be91cee100280e7773506a226e3c98157a2e48f266f3776c919c4110544509dbdb75903af1fdf80a9a79c9c1514d93e48db19e6bde4586fc66b3111941bbf8d45a4da1251f5446760e7c8023ca75e51ae48f6999e57649b797bfb18b9df8b2981f5b13d83d5a554ee14ccc556254af95e8f146423670c4a2b29c466d83f7d68af49367640e2055a20de8621eb7c10aed007398ca62a4266c8097990947f889f586e5ee6320d823de989fecb4fb8ba5992a30f68debd500fd6444040827fa0d8de46e86bdf0916537ded84879884d65c0000c390dc879df3d7d59007f36f338f0939bbaf4727cea46b88beebe032163d41b65d57ef4fc4a8be2cc449695a48fea0211f05cfa827a5688f87bdcf08f4e4ce5e1ddbef9876c6dd3c31d17e38aada140516e8b558f59143501a5db5f0218e88a54c56d55f85e0d6f544689a372d5ec8c88b0b2a98b5012166260a6e02f7e7669d47218cf920fd80188b1e871dddd526ef64dcafbbe48ba48b7ae68f5b0f57a82208a101ee83ad0bb5ac14f205f7da6194bb1511f1d4cc53869bcb9c141155fa0f5bce57e4491d833d09acac63ac21f294b50c713273f9355725fc6614477079fa87a8d941d2cc43347f8a96e1d4dfc5ae6f608ab77a7dce110e66c305551212954e87921c5fcb9d129708adfccfe4f50d280cfde6d530b78308796da1cc9fcfd2c609e4568b1f9b2ec759c1f8b91f345b774a40838806a295886297ee1d76476ea680f858528c901fad178a1e80808b3b533f3d02520d7997ece8df5bfb48bb60e32e01a59a01d9d8910c41488980ea80fd12696fbeb742bd0a0082bdb2a9b55667acc3083a433fce84c25f499f5716d7e93dce4dbecc863f0c8487aeb377a56b409b1ffa8efd37b5886c15a2fc4622ac14c917c761cf2bd2490b1d2d66daacb8ce2b620071bdad3811c8ba92276e3f31f3c5052ff8f878b6632ec84b279a134767f15c3958dfbddc39e1d53e193de05cabde650fc115cd145315b0ff539e81422033a04808e39dbb3f23555be6c00d453d8b6e20ec78e08a82e588de80cc77f81d89586ac1c2968389f155af0a1025d51fa6da7d667219bd99e2edd413179911fbc521b16aa0afab035d3df5d56a08d60a314e7c1b2ccb73472ec8dcdd279cc752955bbdbd3bbe8e620d38f77d1cb2d6abd4cd54e828291e29ef20ff624a8042dd3709d464cb2180b1ff7a619149f0d86484fa8f435eb5cb353cb47f57b94f9a24e8cef25f1529958b30478f5aad90c5491d8aa56f5fcdf8c95a29118935a40c24f6ca08194eec9d58dbac9471e7b9903f167d2dd387d6b16b46bb6d50b0fb70154c5185de81dfbd75ef79d71e9b7a4317ca42a7ed603b8494a928414498663f9db14d6af2077749fbfcd6672c68f8b4f56145b87ef8556f176b069e9ec2e18073143e1b75552536100e5ab75644c48a0b1256e1297f18ab6c1083efb3f42ef93b3f12900b411b3925a3c3ebde516c62f8bd74fd9c1b31f64459ff669098128fdc52a856a8e6f224c7604f4e59e0083cf22c149081c6aa1774fd74def6dc1029969ae67dc97f5ccf263390b8a99b30d381a9399edfad546968e1cc6df23bcf5c4ced15e726d9d86caf96c354a32362bb46631c319a4abd9ef71eaa5cc9de575179ffd013edfa1d5501d3d72c7d0f6da655e984dee9a90b40276fee04e05a57e0fcceeff3be1524ddc9c8133d61c9e87a643928ceb4dd6f95005e519f8ab399b0c4fd00c25fc7e393f692e5ed0f4eab1b83d271bc57f5859a08715f5f608701d62ad2fbd4dbe9073b52a85d841d508ddd50fa895cc2b2f16e689869ee2d140e0f0e0b2dfdbde448106d72ab3bc29bfcc41b93b50e7ed2fa296126c1fdb5a4439cc660e534d696ce9f3d71a353be95ffcfcf50a98c34ace9b6b9d4c5987c1227d88df434057875460e36cc7ca69b9f4ecf4681f745fa067353556daa8e06d315be93f8708fcc82b4c1e42e539bc1dd406350c4ab656484821f22ad7b05c4f56ca0316dc31bbe63a91060ead10f209181a25059dacd6677b10c1a5497d87acdba786dbbdd740b5e28384c35c58135eed044e51d06beedb1081c997e64796ba56c0eef39c8d8c5a2dd44970f43c7225e9db0ffb3a876df289e7bcc2abd2027a3148c86ad8cc0c58d66f19be8df92a85e821f3d83f0b97320748d11b1577434149729289946c1b037fc402ba0516ca63ca61b5bafbb037192132f101c0c121f6f843e9e3ba95bb7a20e439db8b4835e99413c9806f8d276da098dd4e0920d0c47391d2f3d17995f89630ff6b669c30eee1a3ab9ceb7c503b074db6c00577e863dde7f2977144c0b7ddd4de02cb95648f4be939243ece44286de438677efff51fba65d4997dd8a95ebecb98bbe56635c46668c9aa57bb1486596964e1ffd3b4590c1217737594fb9846a50ea7aaf34a01740c0de0a09258b0e734bdd05db7cc2d404e33f3ab43124b88dc606dbc12df7de4eaf21e614ecb0bb35c208644ec900c8cae77db943f713d124d61f654a153559f88e48d9b8946557d8723e3c3331e6a79f0a75ede9a6b0b9edcc9f50171ea62445a76e043d82c98128bec41f7055b5bb830815d8c7e6d2dc2e3ef5515d89bda1cc0abae8bb77024b643d65e2221d6496c8179eef60a608939deeecfee8bf889e83ba6b4ad77b2292843a5f1189912e83548af8b0a4a725e013cb0b7292b9fb84e1e5fc65d26d4d8526d8c72d457e71d1d1cd535fe3af4ea91b868932114f391d814998af11caed48b1d6ac630c45adba66ab0a4d85d4e5bef754d54c16af564c01f00c8f9f1b5dd1f72127294bdc0909e10b798aedd728d573a8f52f84f4d360b057db7386f90192271803dffd0acb137c4d22f7d59ddbf6adb068cd5f2dd088d898381f867d43a3acee55356009e3c1b8d1f7ea6de6c1af045195ae822ff58505994166b7b7fe78d6745b4888347b153fe143c63e5ed37766bf0a714b89604f4ff35ff972e364b462adc0335085de69cfc8033d23e2d950b174251af88d2c65c745e603621fd621793a604f8045ccedac31d8b8c945e2f9611fece860ca506e1e813acd8f68e57d37d35cecef3fe1e8b752e2cafd017cfce0bcb3be18eb8be845fb0f36d0284259e5e17ee43efdb3b68c868c4e2b8b710817e658f88257ff75ec44331fa0170c0102d5e827f2d5100c4cf84d771e0f210bcf567f5b77b8aebeec69b449aefa6133dcd4cf2394ce227e77f368a36015cc9e22414bcd54da5a615689ca08b8685289f590c6899a098fb26c6a7fd6fd188a6f2e6b2e40ebc3cdb72a23ea71e5389bb12b2a326ab06018b647fd833a112d8b4e2b6b87eff0eddd42f4a7dfff71e22d6c58e6d36c6db5bd6c9c9c4ba95785749e1b2d2999d4484878accc6e062fd5431db2e13ba7c564d12dd72870fbf679e36207da8bdd66d8c240071ffda7d6fff5400f6b4474e5ba4ecd806ac9bf24d5f1904529944bf80a15c5a5c3063ecdd722480fd9501490f35ebc742391f1fe14079743b1cf5e71c4242b8c3715652c28241763cd43c7eebcd8bc0d7da22fef60541a21ff55d7646b20597dded3ac03229637c86f19deea18970df22fa0dcb0017b3963bcea10f377c2f8a71eb1d0fb5bdabf3d8e7a4facf41e5f7f477954693ae6080a2f5c721244835557851139efca7fa5a88dba050673536caa59fbc8a1d5bf2d2fd5b4863c47f37a369352dd4a330b88d0683df6da382f2084545a1c60cb62d5df50c881451cdd36fe524aa6db8e537ab2628488f51c04eef490eed19cf8fff71e9a1941a3b4a7fb97edae34be9e1927c65f644b1a42260a7c03a45a938748e45907293d27aae8c04bcd11e569785695f440fb1bc8a1f70c89720b4bc8ef90c013c553a2eaa54f3f597c29f8a3101b5a6d157e8382b8c64eadf438f9d4a2263e8a04ce02792ca7ac591e7c69a94cd7586fabf2841747148e34e6a0dd166bcfc3713f982bde932fe4e13c79ec5757e80a9bf9764359c659ed2f20482ebfaca08924196d746236dd0d0fd1a09468cba3be795c4377518909df2d1881e4da736f9a382e10fbb03c131f96fefb6f39e17d548f3c72df232c09089e505c25bb3d044d7580e0baedf85416dfe4a8aef94944abea0d4d687c4e1761373b02abe61fc8db19e2edf560c5922f8e3cd418f2f582d3025356b2d9f48686fc260b5a725e1fb4ad838ba5865503a0de03c962eace7679f7b777d6af270a5a89e2a3b7cd78308d805d0c1ef39cfbe093498c99c8df0ee09176d9ff33fbf4f046771fe89bdd25a828d58f8de87cd09242c4889b42b1e9c9a5eef72e2612f382e44e79090209bea52b382fa0e591aede64b51af3916a9dce1140d0853b30b6bded30d2760bd73de886539514e48a5e68f3f1dc15dd46edbb1c773812a7c5da61a66ec47ea42e7383abdf7ac4d41115b57bb92f82afac3e4972848792b11bf3d1964337e77613e78d5c5d79ea9e036968b7cf7f7ac4594c603289761b24da6a3bdb088abb2f68059a1477bdbd51adeba223332673c72e81618e15f025a025e9b4847352d0313819ed037ad011b81d649c13a413e86b693760f9f7830cddcbcbd7dd13c9d7adf603229aac37270b4c3f4cb4b48de4ec9a23a2819480dfe988f8a33822155ed0c6838367f2670e3715f79d63075ca571a3431e15f0bdce3db1247f5eefa49e3cd63b140fe20ce7f9c895e346a392adbf44c9aa43dc7ee361816a4fe380980f88a70d3ae4cccd2a86f0129e83c9b478acd6631bb0494db5a5f4b760db96f0c958451209083132043683bb15119a98f6736d0b3fd3099906bc90fe3be9915345112ed959463cfda39c96236b412ca8c960f49b2075b0886234cd23ae1d62f22a142afae8f0d56c72b6d679b1a9105a6a09843f0ceed93f2f18686d766b92799e4bddedbea236e611d84e29c6d39e9cf0b2836157cd0469b9f681578631d921aaf0bee8bd42a514ea1e1041bbfd8cf5f4bd5c216b93dd2b8bb074cf3bad3ea9b148f6f5505d0445d63e5759f00ada6622830ccf5726c7201fd324bb7fc74c1cbacf32a3527aa98188098d53d868854fca7257a923506cf7f6e377c3c635d9d119f0cbe1262585e97a2e63ce1f05e4c13f98d191ff3a35f6eccaa2da57f3645560fabfa379adaba9898035ab97218d16d848368155def703e6c9d16d797ff9c43fed1141e765cc58c5af3c34d9fb35e010a626d7fa28e728f8dbad57e1f13e566771570e88dca7e39b623c21649d54be331c45d754eeaefc5614d648f16c2d0797bca3d42d4d172e237731f72a1e27c00796382f295fa23196dd581f149d0ccfdcefa7807f3cfc57d070a9181fc2257ef3c64ec3ef8be52a8b4845d3a1420c6ecb3193f78e08e9ec50a8681a3ee99bad9966c8f1fddf092970ade85cea3e3c341ddb0a0f840b3b9408db4a59ed7c65704f7c73108afa4e8bbf3724e68f5256525201618c0e7d4ca4d2c2d336c04266894dee5dce37e992a0bc4ea2656aca856b29715ac98c5d750d8ee3e3daef9bc609a6207a7376beebf201a49a615d3a58bd07d49bd3735eb1328567bc5835dcec3cf2df659efb0f6abf9c8b67d482d5f9b89451693e605071ee9efb213f61b200b38ef74cec40f78b3f4ce45985121ccc6449ba435d9e3c292072a7f97b43c5d6d4fe182181193353c92d912bedb0a880303101fdf5094dd537a93a1298ee68791025f4c245c65097654ae69852749bfbe9be72eb388816935de9dd16870868e01ba3d8b42ace35782debd8b79f6dc45b721216e69712f987a73f22538363b8b887b3e3c877415382fedfd8e2cdb4a5f5615b797267e12471d72dd63bf6f113dfff88b6ab28ad13886a22e326dd83d543843d238295b5d3be035805d25dde617db840aaa2cc04d864bd2932ce3dde308e044acbba80d491eca1c872fad0012173f78441a510fd25f99f55a70b77af9513c5adf0a06b554abcad61eed906b01f8961a81d7283e5f1fc7fe7c98557e03f642e06993bc97441a7a2e1836670e5dbb4cb162debc627569f6241169517cb9624e3f638969f8c5343e874f2de8d2cbf2deaa5b24544ba6d7ff32cbf396a3c3c6194dfbed1a56231f9f6c0a771e7514a6edeae8bf1d542a5626c4c6b865a1db068c54fd793f7653ed91386470def506a2d1ccf1648436020d8abf7bcda1649fb1199f01683c8fb227e28647d279a40b95e17dc9df5958ca4ff20c6b937338a94eec9655e2d6bdb6cd7a958ea01e3f914ea7c260619ecd5f93b9ef4061ada58bbdb3c9d862bccb652d941285c2219a2bffcca57ec29f85865aab1795987f75be1531c9a0579b7d4dfe6652d1fcd5d3ec57d6ad9b634d4496b017d5f4aa320382ec66071bbea69ef1e80a618b888f4729e1e1b1a13742715f3213e94cebd1bbcbdfae780d49eeda6e10d5deef17b595cb33178af0394afcd0a622227cce466d29f21f04220838658dbb8df35f941d97d6728d7179e0252f9c19370a5dec1018ec92ba36ede5574f0b1980da49187057c7a686265f3c8cd2dd2f19ac2deb5b391a1f7bc3ffa2ff07255cc62c21924dfd301e5b92859074a98f80ee71096fc2626b6e8e2f0c9cfc1391b05a69d0cf27ccbc447313c27325578cbb57427134b9742783d08f6c955371c794390470745e624752b83d760d5d95c7c8ac19089311199bb375b86a5bfd5847dbf1724e4406f735b29f8aded4bdf1569a2b3dfe90d8f48340897ccf149211fbbc8aaf4cd20f20e0fa92893bc1ebef7ff92a5920eafc517dcd003ac54af78179d2a8811a7189a41d071f1783b10cb3084567050c3ea2676a192d86ece19a837c13b41e2a3cb336238165d3c59e1e2c5d7fb666724c2371f2e52aac2268fa1cfaf076f23e1bd6a719dbf9fb2e93089be55159d5e8346a5026c806fedb95577430fbb165c54ccd2e26934a5be25a32261ccf36054de57307ea53bd5b8438cd1877d3be80ed0812a91e86c991c718d06f8144d76706bceda1ae69899d185f8b4fbc7e10e3acc63250ffd7a86a78289270ae53d58dda611f0e646d444e7845645effe82736e67551d434b57028180de191b33ca8c9f88cd68c152c2a040b06e44d626410d734985430e9d6d511b079842966c89f81cce686ef7e3d87bd3283550576239914f05f64bbc839d10069bb65fe202aafb2ba343d42a2d675672eb4d77b556afd973c5b0b35705d69adab4fd8b01f2c26caf0bdb05c44d576928c6eb44ae3d5d29ee0e2f7132f0a37c48934f5ac3aa5bcdf8176920630e146f3a8e04f4a9e9c1672da60ee7953c47f1748337da85bffe5823aaca62dd53ce71f67dcc15ad26136f70fc8d24249c9097e678fe859d3c9f39e72302cd2c44f8a199486fd56dc13be5490ab9d8875ae77c7e4ce864b663609e11f5c64372fd29967fb26df05c4ab1cfedd765da17c40fa4a56bb25c2e898c739af4957afd501d549b3756b821c99385fb20f520a931832546ac1f30eae5ca8725bac408626e4f8473d25a9264ece8409476163fd5d86cfed9142c711c80854c29f2133d62cb63dbf1bb93e6a31183d00307284c60e3ddfade9f0bf058380d0a3a1dad848b1c4c04773261aa4b0763b1f8c5e7b102461f06f9c3aa5cc5372b7addbc4c7c5947fd72c07770a9558150272604e8da44483465f3e220fede6ef935fdff2f29b670e9dad6eaf7a1a7bc27acdea77a70b68dd5736592b81a5f3e50866502696890ba98a0db0b0d45c6f6ce84fde8d3bfd67d1d9f9a795097467012348bb7143e2bf9a8202c71986e3c9d9aee80be13428eee66e29d249e91147a9d5f465545132108cfce89c59cd64ec55eb6e59920bb556760221f384649ff364db26d8313924c271e2b35b095cb30764d391cf496259f0ecffdf296612d9938a8e78175f140aa8e372526d60be3f968d1e4e6fd810131a7069cfeed048705a527c75ed22c5936241c10cefe41f0613bc41a3b4edfedafdc6fddfacca3d8be585c7f7d7e588a1afe88ecb0f8b01158b665dd5bb70752613fb852edc68c47935adc637e0dcc77f5165a1d25321fdcd51f96c0cb12b900856af7a66c96261278160c1d562fea1c7ac9aba7163cc53f1a98cbaaa7a9e3cb414942d91f42090f398f0b75d126b4da3e84dd2e35505da5ff180f2e0167d3ab8f76d807798ca163df5b1890d5d0ea841aa0eb62738da8167d963e6fc3648fc44a786a1e4f45c70fa8660b5a328ee516e55469d0eff06355066f43b1ef71d5db7c2842faa5b809caa0b574cdf1e1c8d77c2938fc07869b5902b1845d19c13fbf5052a5e851bb86dd50702935cdb2cffb08557e325e4c58a9cdfc73505db53e583e3e39e134af66ca9f7fa375551d856735ab5d21b8901ccd580e783aa3449371b883f599ea7a693ab4e42ee10ec3a5c7f7ef7be311c909eebc927ab4bc96f0595d49db173bb9e592be71bb6fa54c94c5111afa80d4c9c6facce5b85ea61e5af692349df7c42f6fc7b275e4e521fad0329bd7082fca239c0bcb92414b05541c5ce2c5f5a4d533800d104acb4d4895fa3048f4551cc61569d9a7211cf1209fa72d008a3af9332e92d4cc4e8b558fe3137678bf0265081702ed7ed30d69fa76d97b98234632f93a6ed5c25cc1d9f9bd855908cb27cccf2d53e245e6300e0d611bc5e4c679aeee1a6560e8dd67fb279b077a5f32c8cfb9ad6e76fca18ed184bd418e6a5c6d7e5e92c8c88061e35d5796a2a85436ff8f976cf0d7fa24c231ad7dc9c8ec1b83e0d6a54a421e5b5e9e9f8b8871487ef500a08ea2f7d1fae5ccbb44117020b5d0b547b63b4b433c8ce33bf38f022a903e89bac2f90cfb8c56f1065e34953aabb6b69b19a644c3e7e126113d1b4301253c401c0e2109f7a15492d43d6892c3a056f972f34b2fda626bd23590870306272886b73c29bd277d814e56c32566009faac198747e151f9fef48d959038b4b22e394f3c757c23d2b52a606cf5a6baca4bf4d677f511a5093a60ac70169a148ecbd468967e66747c2158f77c0b0fcd045f9e8a4a79bdc91602bc8459d078d9a070d7def334dd109f5c1a7a474b5f5724494fa38bb7df82461a98a914f4686e6bad5d44247850dc7a28d0d3dcdb06bef0556ef68354fd750ccd8a125ad059dc5b52ed91f10526c4fa410bb359837b9aedd8f275e75b55736b948455ba665d914e3d0e6e2f26e1cdce925bfa5847f66895a35ca12178912f6aeca48d60429342910b94dfdff3a7e2e283203929097945113d82a4094a0bbd6a765d1ae849e3a09b94e4bcc6a76c2f27d95e9c4e23c4960b8f9f11ea440ddabc0175c01bb8575ad8416b86d89b57364dde43ce90d954fa596f81735f248ebd1fa29912bb7354d673a355ed42019e755f1fb4b772ca49f545052ec5ba934ad4f2ad4e83ff2311eccbd2705afd3c80206b4dfb0f5c37369541f82ed9e6646caa322847d86cb930efbeaa866f9ac5f5223ea61db49b7af8d793c407fd271d5631fd923c53406bd9c2be3e657b7e7eab1096b5ce5fa664384c7b4099df390a3df267443953f8d6f24dce5e730ee5170542f44e9e8dc3b4d0eed904ea022335165bf287b41435ec998208682dea710f8071f0ee43312208c8994150e8192c3a0d2fadadc561ecce4868954a915eaf19f5df487de6fb2981a0d28b8632d03df0bb3b17968b88f6c8a5cc999396e9b3b95b527d3de25f1c681cf06cfdfadafd945bf59896ef4d5e9662aadc9bffe981b82b155c3fb9caf909203b03f3ecfd6b3f8a73f1f3fd76a3eae7515511ece17313c384bcbe509e571affb8218bb038506f92f0418cb83d87a8c50ccaf0c3581d3dce64cf4915c11be1aec525555e1955aa35160dce0aad07fa1eb8ee0e8d7d1fcc04de79cc7d08486b79485c93e9543ccbbda9106c47508ec268ccf50db78ac85666b129455576f3e36c601fd0592b9eba50cc4588a13a575ad5de88b9d532a6c3c5e0d89ff60a5df5e75c8f89a40b7242404d7ee17a0d0c3ad9fa2307c254a2515f121c68542786b438cd5ca7f8d5d5fa0b3e1f5590da989cf0fd75ddc12e9e2c51a1017c434336f6a958fa9253169b494108ea8ce6db16dbb114cc65228acee356306cc5603a7caf49ffbb662fec375e34b46eb6ea6531b0f7c512cb708a674fe2318370dea1e191e0b81a5a6766e068a6046c082608269ccd31f301d5e69e49d9d3134cfc1e9f83766bb7680bbf1fd7c835c4e7cd8ac62da517bcc65c1d4c2d09c82ee1a1f15073f0846194ed037f4b5d7e903622ad69992d3e625ee54c815506784b3ca757f3b49f152e15f976fc84a1f6efed818651be7fdab6a20168f354ecef2d716d48b4447dc58dac8793df50f26f526747940ba4881bb9b902d351fb39b49f7ccbde811dede1a32090b268b36e1ae883b90ee55345ce163a5930366b4bc142ea1d6941131220ba7dcd992d87e1737da585f12632d7f9a3c09f62a324e13aff0671ce774485b6edc28a810e3108490558fed236fc6621f3c6a675c4fe6a9abb008f13d3e8c38327837866f1fee5bc08848472baf1b49a21c22564ba77a3f75de1a28dfc1ba7eb443788b9dee65d869ec795e757beeb101df109bddc12dbe20b5ab38893b59f49180e23bb74eb342299040fc8b5fac85f7b8f223bc3077648553a88cd15dfb4a88ca37a49116e15633935711a8212894e6241f3f20435e396cb4882fde47d059bf70eb44af4e068be6473f6183c067dfe40f30ceeb612126e3981eb17a69ca69dc17ed6f946eaf77359e45e2bfd60f9dc7862cddae45488bebdae7db689a371196effb7244f34806bbd647418a9a23958f9166a9e6ac44a2627a7874cddde84719ebb99aa0519acff9a2630ab5bf69e94c90ec64a1415d1066b276d77d2c7d9ebc5e140a63969c52ed7412702c66fb0c504d647dec6a42f83665abc8f4e5adcd958910d3108e99f82b7e7b1a90f92354bfe97b3f9f921cefd765153bc3092904065d6bed1a1d2523e8879952a1ad2720d7494a8ba5f95721d4e56022eb4d24295dea9cc7dd81d3a63ed512d05b7d1eb8dc619c999fa61ddbdd529b9ef3545a255a8068fde6c09349748d711aa14caedd974fddc3ac91a9d5da6b059dc1d109daeece6dfa0ca28fe5952782aa5e2643749f116935495862b42b1fcc49ac5b56c6abf21c06fce33bd8b0aa17bd41d3ae6fe55f84c1319de776ec1efa0bbf36d2797d4b5e033c2f639e251d9f5a16e0ac546a8ebe78e35f7eadb9b5deafda0ff237ed703e43cf0543a41d866a0d3204e67207b112854e418ca268fad81ddca626f0696807d85049a47a98b847461c53a62e6312110a7eb615c8e806ef41d9e9a2319c212ca59c47af362ae896aa5f5b23579bf56e66f2b9195e55aa17b41b611e980f34939f0b388b65c7a926ca29d6957cb734c3381fb41b1da50159fc5a4c0f79726e153e19c1fef61179d70ec6c8a947aa4eacea83062616f7ec19a43cec14eb9da757596bf4195ccd740f27fd41c44ec0eaa758fb1e7167f4d729f978c37f4492bc4e3f0d331ea82ee41948df4042c7f5c15bd5df3762a25ba0b6e812917427f3801f51f4279231c80fff70e92150a2d51a60e74ff8e997e91161adb8226aa4014891e997d601bc2e9def81247447a811afccc6f1def990b29b178d7953ec623730d8c281c26ad334a548c786d8a428a1367c2163c167255daaa90b150e6969ea03297523aed482ee71337f888fab1b680e9f0ac1c66473767d413b1bd9416484d61038ec79374152de68e370a190aacd82c99ae9f485a7117f7836ade1ce279ac2af32ecdf17e183d46642c9eae54a40f0a503d57a1c0020bfc1008524b8f7d06248525ff079b3792b6056698061cbdb9670883a90001b0e5ff6004cc9e25ff2471e1d111d43c4aa14edebd1a26b40430fd53d236ea16cf3e673d593d1bc96cf82be7ac9485ffb757f249e2a4c764cbb2e374f24a03ddeb7b9649f74e5a0782967cbb83b362ec87d320744faf9487a8c04a6a732a1314ee7e0dfa3af671a7eb0c3ec6e2be35ed467798c5e29f3b68b1b77061b540b6159eb15992f39901339977984dcd92c1e7a39bcb349c97984f2a95f0b088156fcdc51832639cfbb106564bb579e8797e227f1ea803867b919c80edcffd0c7da29a718d80fd865d5d4d35d18254baf2a82f9d00ea4cc97127aca9e9e26b5df973e14f4e2462307613df6055d914640942243b153926500e199ee8f0d83af18659a4bc092092c04e616f5ef07ca26ee14a6f7fcc409f7330072dbb6c6130ffd405880d1d58d91b7eab8f6677c711fea0b63c1fc6b3ef0fddd1e8b6ed9f5b507e2e68f69bf092cf1a68088e9cc42488c1bb676adb6ecdfcde460ceab348050a264fa0fd02fd933e0bf28190cf4407228d01beccefc3d00c118317e50d6dd31909db140ac10acf9ff36a52a57ca24177b8dbf05e3b1587278ca9aac18b5336ec31be59e06d4d6fed4c9f45cf8c8f6fd006a2d68f774320449ded48e9a75e0127d5558c8de7ce316a2c20564d48fdfbe1faf96bb8d011088a83035bdb28086d3a911d217e72cee20fed62d9c622ebd7804d552aacc97176f9cb607ff46e6a326ef54ad6b717dbb764fff1e960302b25bd5f16cddb3bce0c37b82adb63750ccb694502b9de9950484cfddca3f64433ca5de5ef93b7ba962edae3fa48d78ca37ec21fee0dbc38f8563d44e28669166e7a1e4b80bbe278fc40f5a9acc5f07072045c64e1c1dc5b148b93231115e3414654024cf91bbfb7936100a98e579ebb8ada9e40e563c8012b58accc6e543ef15b7eb47a35f63f26c1fbbd55ca5af1f65f7423865de1fe15b188ded8bf738684fe8216b84db30b6e4a5c29f127095e45ef9d2d307f08bdbbe79ad4942dad7c00e97fcfa00321733955c5a2df3df8fb1ee29f2b3f3697c97a552433fc3b39ee04fbfdb43ff264b36d60f67e2a1758f69fe054e400480a26acdbb84467752f3eb8edbd8cfa49aef31b0c327b0c69211c8f7e1ea43eca60aa921152da5c6f25432920e429d6885168eb022a051240889314928d6ac22d5ffdcca2a9e51afc5f9ba11371deb1b0931e82a3ebe017f7d9d9fdc35ebd9dc23c22608e7ad5deb6f3c17ba1729c54ecaaff8bdd30c7dcab7ca59b07367b267ea4bd5d225c8c7cedd26d4322f48dba0cfd166469f2eb57c0c421b5d7b7fc060132408a94d74ce7c5a29e6a7d7b213264e5277b2199c7b773abb4909d4f0125ac364d8ddcac99d2182bb2c3c3acc48c3f63df3421328df8a5bb1c8e41ce714918825f574b5c4da79f495486f7bd35900f0839b762f27936d35374d293d75c22f602f04d1b663a70bb344631d8fe66b64794152abfd18f522346dee0e998d4a5c1bae985a388b739a0e493821bf68276097c2c3288ab2fb3d9ef667643451e076b98bc30088005cf6863dca69d9b67f61c3c9dac2c2cfca9d2834cde9a76b5bfae7471485bffcc5d5b26c4db27e95e4db292d2f1f5fe0c502d723850a5232e40a5496eb8e6a9f35fc97932246c517053c8c0d2294c5ddfe70f566f323122c76478d3bee92ee740274d47e034df8884cdcad44e2ee5d7b12977aac79cf285d523d59736e47f0cdbf7a235b6fb3ddb8a0388476572dcfe92ed565efe91aced6fd7b05132a1e44767082b0b1cea08e91a075988bf6a44da0b381e4bcc6a956dcefc31e218f964078640fcfa63d57853afc3f07808fc436d3b81b5a384ed0e793e0da87e0a3dc5eace6b4f4aa04dba110470bf50007f7c359e6d1d46662d62dae2f845fa7e14c1f84b637673155679b91839578af4e476a1b6c23f3babdcda25c6cf77c7e94aa8b6a714270a9cc39b2447a259930b1b8daaf0b7c6a84f119e21ff57968610754541296c1b60c04a52c7f42a310a9aa447079666680e07a7106001b6b0c8423d0486aa70a580426a390d6c8407d5057576980d8ccb6c12dfa2adeb8ce8104870d3dbf766acb3a856ce30aea51b86b080e136d4336800cdd09fc1fa89386d0be30bf5cd87bf8b943417fe98ba8e858485d4eefaa3dcf51fd4c0c8750bf8ef5d41d1ac266713d8ed1cfc0606c0f5acf7506279589e6645ec88344cb5a7273fd019bb150d3fd80ea50c72663ba1918f77d2247d55dfdea070079a5eee19d498d61942265f9173302f9b7e9e2e635565f3e4d979e79ea6257573222c97220273b326141191b38ccf194ee3cd7930a3d3ad89ee93694105e1cab50597b248ed81f960cb2b7590b06ca30efe0ccee520b248a69efe5d00d1efd76ba5f0ba951a863e089002237f8c3c124961e7a16ae87e8ac98728ec50f872251ea73527030c1d8502719f94fac0730b10b323ef91e161709ead22d6fa0f8a5f6debf696a199695d793483e5cefec37cb2886477a66d990b2e696a7da16d0728328306766094a791c6b36c9e19128f1fd119885cc91476051746e204d86c5f33ccd66c0433668049d28f709fcc4918138ca67e984c72cf377734dc742447e5a05fa225e7c290509c928e0615fd632e90ca48a71fae51706979eee0e73f16bf70f7eb3b6559d07120f3400fd7f5f256a3c89277f3aae9836bc3bd0386e8ec462a7ab50a65751cbedd96af08a9e1e0dec3fb899977ccfea7df3e769e6a640ab24ac1da095ea2364c76e7e330c6788eafa8804dbdd030f18f80195b918cb8c7923770978f8850823c967f01fdbcc7cbaede491942b6ef0e0f05f47046c4186629cde0e2ff1e24215cb7f59f85a6aedd7a89813560de13c71126f49cf492f163c64953bab8f6ad927f4b8d54eef7d537a6e734735996b59309c5c4f190523dc1021f0a07db5978923cc9b4a8ba18810b6fedb737b360852f87d8d1bbdc3c33c22b15e88869ea7932f866c6a62807e1bed9d8f2ad090204aff97e17c958d381b7b743da1b75fa684965d69a278b924571a68f6760c6006c7529b059aa0b9da72b69f0666b41a8ef2e9df46998e6b5c23bd6a4e3929e7f89dbb3d63e985371dcbed949449a3966a0ecfec915980af345752ad9e280cec84f5ad7b03089f398601bbbddbad14dc9ff710d7c631521399d52d42aea9bd1a17b36e787a6c227f07103aab8ad6f80e7f5bccad0960be1f8f8136c851190f4c9814fd50af54bdab51f533354d23d3ab5ac0854146bc63d36584463656eb2c0eace73aa663adc026576d9a494690bca78fb2169122a5f437e5aa3936b661456cec40988d793ffc6c2e2419dd8b37ffed919c2434db1c1e0047035febb53d33ba19692574e74d07439aa60e36a886cd93adbcb81b83ca7f91ffb602a224165bf00abe14459522667b0380b4e260ff1ac4f9a2bffbc3c7f4d7fef4fffbd7ac3d84f2bdc5a97cbab877d167d0309223ce917a1d4acb5c68c83e4d7a9ef90d9da85dc173bc8d8b0acb6dbc7fad76f1f3db03d04b1c74b09fbf294fe2b04b94b346864f040f6d10b47c583167ecfdf343dd54e3369d6faa41b6c6c0a5cacc75be9b05ca85e9b6a44b840037fe279a0654f781c9cf2ce969a03e359445a188dec90b545d082fa53c32657e9c246ddde059b38ade9b821df6a11410247fc9d47ad0530cb84ba45d275e9ade0bd47a4d84ff5812cc4f60c93a28dc400f85e80d340be55f655e2fbb21edaecbbc45dfbb485b6d55bca6f8db55721aea2471689136d32bf57526971b52265a3cf30c20a0a606a357c28cc8b69f3df930faa2589d1d771b065431b45deaabe17ee30b5901ea4f8fedb9cb088b562bcc9f5b4151e5dc2b0a09a7ff53d4f82d01757adee2c04f0b44492010ac854325d672aa1b8cc32b2e3e20106bb26291e021c5a8d14fe64e020fe8e0abf9bd85d071dec6ba2fc52b9fd041eb4982ce985d89f1418859e5d1a943094fb7ffa2f3dd76758611150d12610ffd8f2c78f6b8e69a7e8dbfed5584f3d0f04d419757ddf0ebea762c1b365caa2ec35f24a2aab6973fe2359fce66b3912d11f31fd08d8dd58af298416762f06286ade4614ec72fc30ad47247bf9dd0d259d5c2ad907b3986832fadb4e77bc12c583ae1f04f3f97d972b05bf6ce03ba92d376e43c77b3386c18c22bec69ae800214a4ebb32ae134374c402fb564dac74c5a48c2e1e692bfdf7b6538bb2847d28ca515039993079db667a3b006cac64713f633c12f4708a372c695b3882dd86a65aa9d5e3ec67538ab4b90fddc37253937b0f99f2a6ec4fc10355f6af5847de9f546aadc2a8f468c0128b9313889bd510140cdc89135db108476ad9172040c3f8b7858925abc36e63b22c13e5a689a555c5b47f7abdad8699e3f1c47dacc088b5dc7887545968473911c3614dd4867410fe1f8c47e31630a4e144b4f82fe3e4764e8ee745749c805a7b33cb811c3d0add925ca6288807c858949cc925251f067d565a77b4d78adf51a473ffe7824898d519555a51d402b32ef4c8e301ed6651ada8d72835c7f68c728db268cf609fc17ba7faaa9d7f383534248e6134ee4b255f9a69a1451ece054567ec3a16bcb7b03d8bd2dce1f5ea5a30553d55522effb037beeb62885fa5adcfdb61fa1653773a215214c0f8bcca786b77f9deaf0e39a525e7d8df52a11c280f44d8f030af666c6799dc316d8f16d82549e87e05dd896bede7d1da654f5bc08ad18558d0d75ebed71def2d43a597f375efbd183e3de120e600123cc3024713d881b3cf70a9948b996556559eff2c5d02304b1afd547de7adda218b7330bfd7eaa84e55746d5e3550caea313240d35f07d972e2ef0a1c2ab33d5ec98cf5d0be1e705d73d4a4e0b835e4d26bd4cb6fb9cab07b3766c65b1bcdf5ee9a2b8f747d1fdfa01d250d454250fae2cdd65be152abfdd10b988bc30b79685b1ab7f5a96b8b9e15f357a3740103ef809305d07072874d3c44521e097c918c7ef9aeed5fa1abe9252e949e4c6db562e3d8d126a2d6a1e0e765e7cda645d5ab48ab33c9989efb4ab3dfaac67f11800c4ac3d9d840d19101653d8b3cf31536b8021cf3413edb70ae1e93f3c1ecc47fbc4e04cf1b49170edaa04c23fb970f7ab21abebb9fe93220a76f5a8a4d0d758a635afe0fdce73d01b82e7337bfd182e317f99cfbcf2e624b6c02ff3a25dc45f957563b49a92461adbf50e46a4985f7cbe864e13b13d5138f6a1df59d5fa6275dd3b4ec037d93e18644644c2b5f5f9cd3580a7e81bc473355ac386744fe2fd886b958054a1cebc833cf199862fbd69039c329c566b8be2d42fddf2d9681180aa1bc0e44e49408244e9cc16e8df51ad70ff41b76d85d6cd4bdf560ebbb02f2f6026c07e787bbfeca2ae20b6d63d41bb33064b7d2b80e332448f1ebeab260618e31dcd78373f06766f46578b486ad58f17585a9045c327b86201df4ad1b6945e10c6d22838dd18cc1ee411077be7e10670fe53cd7857cbf547b3d6e0421add6216f39b5348f5050264087d57c4493a6f3537ef1bf58951199c9b8d3c1619289f71df60583d79a233f9d59e3b204c7d61c029b6e0cae316aaaf9a26a4b5ff83cd6784a3e3d5f00d24ee56445831fb7ed95f114d9c38066c47d596fd7923dcb23740e26798db15b180af88762f3951247d88001ba69bc2b5e6803ff7add583dc6c24968111df2993bac6d3baf919f1ecc6703162723f6394fddf7460ecd37ca1c555b3f9f55c3a5ce409aeb52451b685b59a68342ebc774fc7ea714ed6ab1d3efc527f49b7fdc958c4e60b380b2740e339d3b4f10fa085d9bc03057280523501194e4be595570ce66abcc654228be8f79783b8436245ecbdd7218697c89d9c829b28a8ffab850e26c9a29eea079c1bfa0d1b34a7a5a6803694f6263e99f3e95c3bcf769b7606b6219d8136295959f510448f3ee38b9d5628ca56ddc3d901462d8cced0e9cb880770135fa07df2883a24486f48731b2e3e3d248ccc17dc24aedb0c265a5f9e47aceef061a8233cf495e4addea5e5afb3db6ab623acde7609fa5dbff1935cb8f9ab017324d421ca8b805959607add926d1e80734147d958887b209655ae07a0d7de7c1a522def47317df75cdd5b9a29b00ea23f7042478f906993578f91b90d1eb3ea89b12fef9f30edba592d42bb8681ed230a409fc44c8eb2618ee659ab28cbdf4eb72257805f93e3d64b7c73b4973c8e8a8db21f1865895d14ebe794f235208f13a49cd79e51899e4e6be57953128b03203af79a7377f77ab698f543671ebbefb3e31086cf5605c0983f004fb62d9ab97ffc1cc48ac772825c2467e0a6fb8399eb6fd4aa36250e44d053dc0dbac20ebd5a4dd5d81b62cc0a41e900967ecc1ec4abcb484c79bb0d81c327248bac1e34cefb0cb7cc88eba5eea7ad787b3a444a4f347bfae68fb51e266ad8218c5381e15d539d73e7f51ca09c10086638e175669cc342a07e32df91e08a564378aa7da832df43161fedabf7797ea931ba0d0327b3c34a69ad445b2f64ea3ad2233babbc01c8d464ce1ae919ae706bef467817fee3c48d094ac63c9471d67bb599750bcca0203fcbf34fcb7e11f8e291522c436d5e520ccf45bb7a9dcca3f87bc11a45bc890c92489b7d6b6fbf123499f5bce0ffbbc062bc4afe25215df312729da1525ff7a7d7735360c1dd875cec3257a86af7257c79872b598281dc1774adc89833e57f17830c1b0418ad926447eb97671a5b40eb3d632adff74be9a83843ccf0b7d5b829d1f718ab68178b0449e0691dc8a9745664dc609409dbc31caeecb540b73f5414c25e260d2dfc4baa95e58843febd94361b3d67f0f9cce308180dfeb3283c8fff235a8c2d1e98ce9b615666a0f2cc6a28b89205a689373483c5930d803aacb04a0a63d8d3c9c4435c1965e6ea33a1d32547843a41d6c855eef7da188fe879844e8c8016a61cbb7aa905816544af9fd62ecc00f44155c77a0e30a8e8687d682103fd56df6bd6539605845534d99c03c93cc3b377cd44558aa966e1db159eb76494b5c2292a5ab0c49b556edada69e9e0693cd714be59e96883f3643549eb8eb60b0c84d52db285ed945aa42f6ac910ddc0e673a38384eaed738c766a5e5d8b65286b542d8f7a89a9e209ab0bb3b7bcbe67619dacac7b1f954ea384fa1540d5d5b003f06019445e9f26c6dbd082fc8347b037fb27fe76420f3cc1864549a728ef86fc483283bf54c03d8ef81a75ee7002e2fbda5ba9b14da07d931bae214d96da4b52ca0e059ea49d7304fdfbc8f5e69b3c2ea9067235ab5824b5eb99506d367b0bf65e431fed7cd83a051b83e5cc2a32a7634fc15752b8939dda3948f37eb0a7142132d6d542ec40742ec8e408dd786b6016b80a3a18116898f2c65459a5e0416cdd76b1d369cfe11da4d0e24b1a8eb6e0b621fab5076565a0103ed9bb272ba20198fb523b95355d258e10fa0d3467ac805736a3b82b88182170d5fa543dfa5ba00a6c111c32e9134006e1f16c0f43d620bac56fcdb804fd7de950ab7e8d1fb43e930caf7934c972ff12c2659debf7e51b3a0f2d09dd8f3e805cd34c8fa31d5649fdd13bb5ae7fbf2f566ab8c09fd0f3bec555279f795c197ff38434d58d8aca2d34e9ff74861024b978c55328d85399b7c6890dedbcb008d06d71563a108f6ba40ee5ba9dadbfd4c533d2be1341c85976db2c51d2c81fa4687269100c479fe64af58ec8db207662a8920da7a411c77258cb3a84899b8d86e84fcb16f2760e949735c2e1d080498ff6236b9682e506e82e684cc8f37c3f85bcf467a81ba78b1103f70315e8c89863e18e217d6ed674224779873d6d4a9d8ce3b28b98d7f6ab90a1e4de2f3caace3bcdca26139854e541e073ad047f01e2f24aa8ae34a3f726cb4fc6c39ca357e6902fff3087221cf4ba8524c4a51ad18107ed500f45f36f3c52f8107790a15918c914ca528a0d6e404d81b5eb4378bc1fb0b90d6c0800904046f4ca284db5119d1e3578f10d3c5f8634884eac7b217058ccae56541a3325c89a552514bd50c3f48f0d924982fa0e200066eaf2f727497d5abf38ad6844979512ff975c70110b3c7d553aea087a7d6ddcf4529c8d061859bafdb9bf1b8b5f1f157d8ede2763e2eca65a304ba9781a09f9876a8370311d37e73c99321eb014d3d692cc1de4e5a19ca6850ea72c507c605bcd9ee3117125f80eba2b3b20a6d7f918f3a45312c6f44f79d8d59740e0138fe647bf8173dcfc0cf2dc5905142d6e133fec6e72eab73d30ab3841186b3bc71f142a12c5734e8bb055c335566fcc0f5fb321a457d9668988edba15b9b84bcd831162fc69297102a8b096129a394f47d06af28a1e637b093e40e453e6a39df35fda4ad1ca9419107ec576fc274b80ccf6d73f552b1bcadc797fc5a4eb6d8602c4ae151d3e1ae5961c01c9e514f8f624f502faf51a3db65b5af43beb2ee5d117706bf89a213cc9651b2314a6eae3a5f98783e97f79b9ff63b1adde7d13f22e257158796ae8eecbc6e597effde1f280f5ef3bbbcdbdc37ea19d5c512ef49b452e695ca61d73e2aa7ffa5661331aa85418f0155b647c9a532cf0b79be2a851921c8e3c72d679fe9f44628df42d04d815bf3661212f1d498c623a548b2bbbc622dec98e4e8ef436de33efbf4bf9353f950cd2cfed80d10c00ae706b58caa45baf7e31ad28acc6a3b0d3ceeb2a92fba122d991c19dfefb156447867243b906b2d64e7ae654cd90d160b55e7c542244621eab7b6a01095875683126aaa674bb1d426bccdfd2a7bdaa3f7e1a0b7a8909de972afa89b8fedd28325a37bc1283df60e7d2fd2c3f3166236f1c5e9775e10beb802d873ee944802f2dae0bb9dfa127451cd9a5c189f0ec147ccbfe43c83d63703b9c3a93662780a5b3958b26dd835b62eb23830e779bafafaa3f23e1eb1baf6c95c1817fac2ae64c229843e34804a93610fd306e3559eddc5aa781528ea4abf94669158f3fc141a3151b7d1ca641241cc1535c63bc611a21316be0fe95eaa147aa5645707e12a8493811460607f03deaee70569130cc9b732537750022b0bd268a1f0f6424e1428bf2ddb5e7f90337115e034ad733004084af486a145131cbc69340ec4e933eef10356ee17340da281dce013df72bbc718b7aad8ba2265773e9fac54597382a1a255e5288a4d1fd5f39f027cee1467e3f9a29a48eff395259f9dec779d74915e35e32ebdeaa3e13b59beeca66f71862da21bebc6c3ac0646995ad180699d20a90565ebf47ce6a509c771ed251b9f37929ab4551f91cbd95358d3bca89bc81fb0440699b0cf82bcb36156af122faf24469be578ce8f4f73ef288cbe90f58674317fcf13578a5b022418502d8f76f07aad1b0702bfa302078b90a8f19987034678033f848111f4d5d5a2257f795d89562486d61f2f450d0123c0f26637db625f3c0135e46d5cfee961816b24ef1c740e949671f1818643931c9eb0941c6ab76e37a9437779a6b581088d3958256573ff8ecd28beb3a79127ad8b15f8c2a9bf7625f2123ad31e1f41871a770aef09f34f65f66c033a5c9ccac53c6ad9a0b6119d92ac8f78de02312edb2fc94728bbcd7c2a7cb74d49f6c625dcb8218678f22e562d260defad745e8e349da0e406ccfe1fde6f05f952835099613e09902bc6c512f45e1befc81c59ce3de8c7af202075118b79b519381856f4b448d8c4922c43582b0ca3dee58e7a42eea98c900df9c0fb1ad09613cb352aa74e537306ed2848f748ad99e927754a11954deded06142d66a21f9ca5464b3478d35bb851606c45cd5a1f8504cb9f1894d190d6cc870ddd00015277456e4802f74314315e81df7026aac7afd1c7b8fe0fa37390c822cd45dfe457ee17930a2cb96f1ca47adbb9c8fda10cdf9ff6ae56809845b6224999bd6538ddba35880e86eb5c9a2da0b835d38f33937a0e98e6c9fdec10c1009a2dd8d02a7aef30d85003b9de9c38ebb28d2a38c7b7aa1352e760a9a2c21e2697ccef6c975598d086aaca951de6728d168cdec8a2289a77cdeae00a85b1041af1b4f1ec6bf3f7ee8d1a4b2a06fe2bcb2561018896924c57fb6a0ba1a4b7e3a097c4dfbed62c687f75f69d97936c7fd0c14fc5050914ee9f3789f6f45d0dfc9615c44047c26f80ffc9b0a5eb5d96af062137150c893025c85753ef64d6b3e03523e744daae295fc73fafeb7b97ce67a40d4ec6ae306f5df435c819c1e1e8a7f4ef042698d76ca617d20db76f8d91ec522ad6d830f7b61ec1228dc240737d9a698a40caf78ed969871b5cc7649aa5e5ea46841a44d18d0f5ecb527083059a3c9d823c8a97625078ff79a10b4a4d037016fb14e31d47b352d6c0bdcd83f67e91e1aa44785c1127f1d5b9f54257d8eee4dcd0e4d270b381ace5172b8c706fb0369f1928d153f281ca00411f50187abca01bab1940d63637eb2b00c8751526977b48ce0d91e487e988cb754d7e23497395c39de04d27618f15cb662d20736e3c54605bd2a50f5c0fbb727779a2eb266984ef7b616c7c369bced28d50235eabe8956c749c5c11a2422c950c42ac8e098b8bbeb7b2c6b1811ef17f5ddeb98a021c13778f73237bf0ad9ed9e6c0ac213edf1b55e66f275b933ab5198b92b093b6bebbd0122364f493e5277158a1c4c409b788316570d44625870ebe55338a7198f8e39214196df858ae78095b55645de3e8d52e4396e9a5c4a949197e2ba03491f94243dfb12714d56577939bc09a76927ff08327fe2ace2bff5eca8cd62dd87b376de92a10a9e664923957e767423b764f284b8e39fef9f13e14973a983ada0808cf32992ca00e9bf44f2e02a53cc3f4126b39d879d208c852816e1db8fee8fb419a86dd2bbadd8ce71ecbad310e13fc018445774582a94db15e8ce66eeb93b5504ca52e2566aa73f96ae6fb71a474748a8542377aa80ad11d390adebc64fa6dc0b7fed9e82ca610f9e3e7c6f96bb390bd98935b7d4ec28e9caa7b9b3a5835a98348a0e5dc48b3b04a9313ae00b463fc6f288401f5a0f324ef6b328963e0d9d3fe280b488e679eb9623d64c4cb15d7da438965e5e566ab499370ed7dc4fc5c2c4622e293655bb9c287e070c1758bc8405d1402930d98fb25d50607379c831f7b1b7dfd9e00ca4de8a7b89adeb1caea15c0a99842d8fb336716914531b89343548a46a3b0ada7a18ef74dd5c1368ec190de41dea7edc30ebc65b84d6e54662dfb33f2f98f6958702a94a8f33948f8e277709568dc9649088e87af1130f5793cb0948d875f3f3ea7c4bb42000b8993b072498d7ef8c51c986fb1de0bb493554da4f274c6c33ce88eb127f0dad5a108c2deb5d123762fe0cb26c050ad264373b93527a0face6eab037de0960626803287d9f6f0e8ef8d633d1808d7e9d4aa22453dd1a224ff9d0a63c857344ef4859a289c0af3ece82a8ebc32a7de084e83b04b5c6ada0713a45de7ece70ad5deb3a92c703f6fd23aad77a8512f7ef1a7b27b925e78bd56fc2ec360a48fb87b49d12f187b65fc5326bfd73ab5c9bd082f14b04196b990fdecd49c2a9358db8e5a29d68ddf867008582cfff4a379819d3f0b46d7364722deac4e12ceef8f1bc3aefb8902aa7c2c971511eac7eba440716716f8a929204978e3f8185995b071b03f839d163c12f2599237a5a488a1df356deb31278633ecdb7b994f5b98952fd8d34043d18aebb8d65954195218243f0651fac3e889dec4a9cdae49d9c91237684e2116effa5bd83028828495f10b3ecd37c0f73896e74a1b55e1cc9fe2a7599993aeac14b44daa5016e0c1ccc8874e9451ffdea4ae44ddc576da1e4c67f08f4f4d06cb8a99285c493ca4944247643e85d57413ba9c7d28e833a344a8290d566a8ed150d0d6ed2606d5dfd7944447fd12c065848b4992bdedb8d8eef77349606ed6513eb637cb6392b5b5711fdfe53a7ae96e7e091086a6cee510fc7d46a809e98edb93b75e5949ea1f5fc146f2d1e4dae283075a37458e381cc14a6865b98a6f498794562cc89dd579a942f8296e2352e54264018abb48d06dfa5399acc7f6b5b0df1cd51bd51e56311414f9bca4ceacafd3278b838979f00ac7cef7ed42a97e2f79412c5805a6fdf051c5728e7b1403a39b6ec50933b4309b53c3dc722d67d33008cde06586fca17d53470e31f6ef516487e108d0486e2f8bfc6bf5e16ebae08b2323c2ccc5ddbdd6bf6482b64199254806621da2ede2498308a7ae4ba4688f43030ff891285779d29c8073b31ac6ba7bb403385f2ac80940c1a8e971585c9809f5ad264bdc3c12f04701f9ef0e11cee3b424b8e502109e59af22612fa1631ba1da9ab6b4ecb672067d747d5e7f5ae6c2f7cdf9dc5c23d2b618192ee76584f113678d2dc4c55bf08153f7c4dc14def87b1b9fc707c92cd785342785f0d868549ea4c3a471ac4e9a3ff35e1fdd5424381ce68dba197917f2d9d6a4eac6b17ae580b4be5200dfc55fda1450ed8fcc5892eb0863d93fbe5cc0a5b0e1c094d0ea78be1367c341d0c5b323de1bbda19f6a4b5ab39e5032141d950f7724d581207cbe22909e0afaeb16172f6c69c4cc5d0550506bbc644acb7c8a147aea3be877ae0b8f970767adb4cd4d5e93c367d33bfb0ab61de415965a75b63d1bc474d32529517c5e9203e259c077aeaed38b6296d3d878a613e1f4becf3d3c1b009e5aa3f6c533b3d807bcce5bad991792d97f32df5bb9d714f51933cf050be7f6cf2c1b891b10941f2092f3de320a8c50c5750fc515633500579cdd5d13061c7e1ea5ea2823499c537e08f46605195f13071f3bda72c37bf18316065a7388cedc8102ed4a6021827c152c292e9d7613582b5e28a0208ae993ea3b4581bbb4096b1c22e22bae5b39f4275d57accd3440b275ab8d80d5c95fc6fa4004a4744ea0d7e96cf01063a12a5d5fe474cb99422de9d254e6dfda0cd6c5c91dea59d0c2e290c2e9786d637b2ef7ba0dc8d44d047b9a74911b1cf8b01c745e475c7e05c0357b0f9f5c071b0432c1318f5d4b61816d7a2a63a24d5962680193fa0e89ad130ef0c60efacf7ffe6da835b06d89781f8d2c78071b76c0ed30ae85211fed62d5982ce68ac6c3160f98722700d89bf7f8da98bb8e5090bb34dccfa796739b62fd9246624eb2b07082ecb5491b337182686adb9506abc20be9116c07c88edd2a7321d7459829f056b2de08a420a97bfca16183401d5567099d1328b59d93bcac8820aa6dbc74265d0c59a34e9f7e014f4c9ea264a114649aa467682ebba50527433f16bd081affddf9cee3114915300c2b3888fa5e161b75946b6151abe393d60fcf5dd99684ea97bf32be266eb745b9bf911b7e4a0c28951c7c0b5b997c3830d2413e227a72160ede453c7e84268d2deaa845139ff6a3a6b209cc52dda25b266cbfa79aca476cf11f3b42dffb249638986de16ba9e7f13e77dd837aac04ea542db69329d3a143e4466bf84fa1d32253d4da3d14f7df112e2ebcea6ff3231cac9ab052c8a06cc0af098616b9756ae72c9be7737ec8a6e64bee5b4cf7546fb5c9dc146f658cd0e06f36e16e34447666a8f14cf52afaf5d6c78ac8853d33369651fd46773043a4e00a3027fcb600f1c3059657132582ecd341212927a08072b54c1f824808eac25cf5a91216937e25ffe8e1f80b89369039ae48027075f24fb1d8fd0b8e88acad85e97ed525122065fe94be43fb6cfa02cd0b09e5e84c0da492356e984d256a8af680276c5626aabd84999373cc72dba9403c45544e567bc51dcb654405f8d9d95585cf2fc4176f56b647e8f4f36d708f95323d95416732ba35148738d84381fea00e18296156dabab5ea7bceba492648bd62889e13cf01cb3992a47aec215f9fa7cb501e1aced81ebffb6d9924340cbaad1277894b0bc1a76f1eb4786ea073c6b215612dc7a2fd9417f62ea43b96fc70680a6d75dc0ffc948709b4ba9fa8ad49afa58c91c5301311256d2842160ac2b7eab6d7674d17be2704e4faec4b5dc2cdcfc64a36d1282686ff87ed649ae108ee26cf1d480447996f9a531928bbe11b310481b148d04c169bbf861e812b282697b46cc35846be8806ed10c8d5eec3393d1a5da1e2e5477b5c5ab8e5ef333f480dce2de08ef563ef678f83f83cfb1ce0bb1b1cd855e49ddb9f61fe9bbbd55e2dfa141c37ce06cff653e03742586ceab349905cca6227367d4f683c681264da1f01a279ae74e23e8bb015ebf240916abd8903c4c257a7b9c9a39892db53c06effdb1910e2aeced5288534b9e8a025b73175bb2ea7ec88372f589c13edb6524166fb61769384816c6f26a4ea976d4abe8629cfed5feccc8421538794c05f3370dd2f7aed0419c6f50a06e4636f02992e7950d5032246e9b830d40a370e2d7070547ec135e727b8dd6ad89c7d8a010515dd02ca0a5c39dc0a9fee5d2f4bc03bd6d3821a548930faf9ff35f54498299788b6cf98c4fc7855fceda270e2645ed6861df8d1cda79db5ab4dae3774d78cda21b0dde4f336d7c7063f0b39eb0e8818786ace5a94818e623fadfd591f6b07fc634f33e5deef8a06fed00847e90703ac5c5540c73db6bf808b347531ad794dfd93cefdbe1bc3492976064a4e8b5dff3d7fd699db2460373ea92b52a7953dae3449a78f88bee8407952406edec614ea850f410a51a7ed2d43aa6c4a645ac161c6eac80fbd50fc9e3e4a14bcf4947ccf1b26b4124ec16ad95eeaac82f0bc209f9d6c2f8dc595914e48905ea1d6e3e1287410b955761154a2d3b7d3c3b8a7ae24dd27994acc8cce4e93666223476033c8207fe41c3be8ded7f58a77442271640548b7ed5eef57461f76c4d944f1f1df8afd1bd39aa2e7a62fca7bcb8499c137bbb3b0130223365cc6c56bef59e0395f5b2fc20b3d57a179aab2dfc0794d43b8ef2b6643bf1b4ad677d27b72411d590ca72f7736040238a55d1545c4b5e5e358214ae93fd443667a871adbc5b185f1f3f32b40f4e99beae1aa5d30b65dee0df704cffdf067c1f542ca9486f6ce539a1df201b6c8dc7cf2453b4310bb734b31228e658cd5770661fdf00eaf7b4a18dfdb85b2c58bfe3a2062f5562cf616f9cee5121c6548fa8c10f15df658fa5e3e4003cd94be620d7af34f02d6ceb4eb2fc29d100f06ac170b6d2ee5f95169bb9afb9e5eaa8b915244b25fa05621a22754b776f69876bf29961b91dcb3657febe202483e3159ae1951cca90303f07d76b399067b50510e71eea2474c8d20bbaacbf6017d70ed40ed3dcae6bef178bc06483e6f0753a52374a026137af21373bb9a8a0c59badeba003a3ed35bc7d8fabf765d7ae50a791f6c074b73adb9da55d00883893dd7727997466d96881b6c7945279a091d774eee26c6e2b6f773bacbd103f8b740f42aea36fe4c3d09e87d67c0faca542bf9655b9c0418de7cda3aa17a6656a8a6d46dffbc9016d87e27e7bd156bc491c8c48ad87296311c94aa1cef1fd1db0331d4763247c6bf7bde2d6d2e77b97b61d73c5f195bc28ac3293a0c3b2911af2910d5bedd33474fb6d670a904264622f9d0b820952d918402ad64227c9d44c307bfdb3d126f56b7c63f3455ebc59a03e3af6c1446cbbb2436188130782c37c1f52a789aab146000fd8c495686988ce8b74e2a51b6f7dd374f4cdef4fdb68a1c400c80bb6f42fd6edcc879b70bc2e6ef7259f50b7a89b78b00f69184db5fa7b6878313aa441259f0223870a89a11c214c878953b797bb21ad1a6b189cc49c276a85ec740cd35a10b6002fa4eb58b234d2c87ea1e4ceaa67ea29afa7f63ce924f7144ed35949f06b9378ab261772197e90df652af628d166a27af36ad71025411d622975df1aa6c33555da4caa90aa30b130f482cf553ce53c6437a4b279fcb2e076c0ec9ad6829c0d224a19162c7a96dfdd9c14b727cbda7d738c5cfac92b0eb85c13f67ccb564b4058ad64782f2cd2cc540b1ed31eb1f1237151c5054c46e24c62cfdd8c90f2c5f1d1385a41b601715c2a49681680d0042364233d63c998e0abbc82ff1347de322479aa55a753ed4e2d20f8d26d588b6a7f35c955ef1c1300d1a6eba37356c6e3b9d26a8240eb8cb89430232d993f355c2898e0439838d1aa91e414c93aafabd952256199fdf9efb979f543236c41bb0abd77201d3505ada8befe4eaa9712ea4b747225e22e5b31011d71a0a9f0f45e99cb8dba4108ea037c273e04faf8ff368f222d96e581ef04de776428fa5450018849725e0f8ed30eef26e99a2b13c32b32316ca04f6756c85e0399d870861b81801293fc9e6ff905c1c2d25da6cce72c30ff9df53289ff88a61c7378a6c0ac54602bdc48574485e47f3894847e80049e33e2710ec9bda96d51c5dc1e2e131cc6e2acafe6111bb21cb98eeab252e18d1bac88e89827e85b24293c218eb2a105f61bab090d64fc3c8d7e4dd5aa6d97807c28c9533dc38fcf6f48f4f87527a649111513b48f72a505e72c8964c4685f0d1d6381f1c465db2df2f988b466bab1dc4e1318f53766b919a89b10847597c9d2b4de6ffe6561e9d21a0e24c56ea772769e89c7286b0d79fc001765dce2e382d43cc9f2c70b3df340902eb8b42bfd20b3991006973fe3d209ce6243a989b2733d2ecc7cfa9d141604c96295aa23fbb005c6501e2aa2ca6183e5471e0eaaa9a4e9eb359c551c785b1cae84cdbbd2c1420e7d8bb66fffe10e06b3887199053ea3f941e93bce9733c00864af4ed3e3c2232bfc2188bb9ff0f4066ac97f9271978c5a00d25f99a86b6eca4fa02688b8f114344b19f8b7c84f767d283182c3c50e97625e77101c92e0b0095f6c3e7dc8cdcd1026216ec285f1d60171d004c6a8469b7f75012634c9cf26dc7bf0cced47722a3271c41302e34890d96480e5509aa4646763245566728caa2a5db25a1fc8336ad982257e900a5b3481e7f690b343604aa540f89a43669a016371d67cb854b221cbfb3dc35a9c8175ff88c18fa1abc5ffde4f2c76c57d25e407c5711aaa318ea208e2bef534e4bb967fd22860b2fc14e6e6dd8fca1ea76a715d9df3b6a94dcdf99c8d383855e4673c627b523f6374505c60a50a3207917fee5768fae3dfb10518d4b5118ac8a6fcf12ad362ef637317005a4660831e3e421ba44d0d7579f90c6e6adbf4c7d9873502b363898c82913d29ad428e1c932963b2427ba267b75825b013637110d21b6247933eb786ae602944ad77e10eee6c41155201423aeb35be644910369bdcdc1c8fc4f9814c31981cb929e83793e4c3fa5467aed21ffc66f32b2cc16c3755fdc37b2e9c37bb9dca8c3f4a2d8dfa7621e8f51248962b21acec3b56558d2a276dcb1d47c8eacf7283c4e7172a6e08b85947f09f55f22d8a55fddebfb3aadedfdd582e3bda3b1bc70bb796bde7f4b3f9e371887036365a3b59aac827145c807dce07de4a93d5916a34b9b1f7cfd1f6b40387659a08fc468394ef4db1416898dbfaa3244e9b67386a03380dc66ba9bc5d4792d87431e28b4f5859193571434c3f95f008d19292d4fa2e61cfccdaed774a3221096ba8648f4daf83e142c05fc8a00aca70432685d3e506e254cc511038ff1120bb1dd5177744b5380d0d019f718390312de9ba37d5ef791173e7b5f869c4382fc3e8c18726a18f3b6bc0f925b39ddbad505907016324f3c27069076459f6e4167045eab3879ee85292ba37919197ef2a3af154252769b9751d368275fcbdd13464c227ba0ec8a522a29483775c2d68009d7a302bdeeea43976dce24995d262be204cdb647130f5a13585ba06735bca5399a583050b221617dd8ce7ada88bd185e6c5d278b893b46eaafe5e779469598da596b54aef825cafca15fc6df3f2cddc4888b977bb0e0471d6accb13a8c8b8c34cd1c2de845474522c4ad6ed0bf7da4c06c57a801a7ba041ebf27b11880713bbf4777b53363e9410440add422d3b4d5437d8bdf86761a0a66f61006d478ac79a951d476eadb58e88f1df9d1dcaf31b8a794e22592b61e78f7a88d2ebd299510ee2bcee961d7aad3bcd3f1e0813c21aa8389d753eb9aade9da2077cd3c620cb78fb0ee70ba552cd87d92da9a50f211556b1d40fdad7b2d9d0f5fd949b07e8f6adc9e5d8f02746c9020f1484624af7712ba9274774e26513e1df34865d3a1ea2a1eb4f374821b64a8fb5dcafe11e239d56ca7248f9b69a5b70ee81ecd485fc85f44110dcf7543a04d93aa36e0534aa708b441fa561794d56a4afc0a2cc1476882306781b0510e3a112448c0f0c81a334c68e511ad5ca0e0f032f2532b9805e36ab686f4512b3240d5aff90ea2e3df0f4f3436fe7e839b9b2f612d19b5bd61b15550a11a85566240284ded204f7a92a54dfb4a1f7c3686e46600135dbe1da30e2c8682bdeeb94886c75cb8e8510654901189a9e2c75ae082027297850c6502b3d83bb79f4d5acd07b4f50803fbcad8d2a35f9094bd274e5930e4465463c8164ef0bd2b4982bd1ac1b644a5127816b83d307894fc28fbcfcf26d59eeb335c6e327ff271576a2dbdc772786787da91f7a622f32e94b7dcfc076078c5b082344f21b5c7df0a044e6627f6f6d6b8b600c7e33429977a1d0e3462b89311a38bb86207a81cae1e341a5872b66efc84e793ab193ff6a19d4d78e49f8638554349595c5b587bca40a668ac8070863228ecbe8400dc416b38e51e976e99d34ade2385a4e143fcbad6662d7534246964eeecfe1f217119f2633c2c6e8db5ce466c02ddb14f36482136d5ea43b8b1456dc852150b8d6c8f042885e390e86fe6597889a804d4a45d4e2d078cb5056fbf9086d7ae575e0f731fa5cedf550bd2a3abdb8127c14841398320d594aa0510ce424ab107af78b03cb5d9db7c99db40ec81a6ea121fe1b555aa70665f925b4501f1aa2355f89c8ee0988846cfa152bb53329bd9457324f41595f1fa5b349572ac5c45c9a6fd0cb7476b72fa20b18dbe4ad2e3ebd2424d9af6897714dc6af2c4ea118b7491f369af8e4097894099f7b537e2c30322b0f7333bb79b3d4d1f0e612df42e90e042cc6e33aac8c9f248a5248b3890fdffed18e3f0fb9ccfbee77227e310495f010b07e90756a4afea36d319a754429c49956dd1fbaff1da085c1b91b54b71b083483ca83f4da2db7e61efcec03da08bb51076ca82057aa8ae1a66bb777979eb7c5ee5d86446c4d4101c99a4a6ea8d0d8cc3178dd529f3a241798a161972e23271bf78ff13fd0013b3a4dc835e9ea63f54cd475c0a1523abfc7a601fa98780117d1723cf9a97e62b6077cedb349632b4051cc826fc1c19cfdbf72ffb450037be67763f08090f5c9cbbbea9e97d15d31105509aa783c0220da1187798543f8ab7f529f1403e1dcfd1d82d1d111b925fa2cd2a88c658ef23d7a45c8c593c119d54237559e663d24be6229cfbd933996d8cc1051c06caf88733e1584ed5d8bd01a56f71b6bd705f8b2191a202596e7113615081204bc25b898850097319678275ab9abceb4cb048aaaec95fe75b926147e267c3fa7e4447d5a283b6fd142ef3391ae14f9a1b4495ab971cc8ef6e3a8b373158a0a168f2db4ca31d2c08b2d082aa0d71a37614e182699898307b2b55f11208924be8672429142321da4d29a9bb260c0c29fead7755c732b40778a87e7b58a4a44836365cdf6c358ec365af295655f5800a88e073d044c7b648ea30f98a79d929f4c1e010eb0d3d829c25c1248a72f21b687d3b8de0d24a500e95c0c4c660b3f0427c3851486fed8040677b05ccf150d4052fd3621b9bf638d4563daafca3997cc84e709a9ba6c31e0310cc4c551041431cf88289769356ad6439654a070ed8e342f55676637cc0661c35a3ea84788bbccfcdceb42685400fc381266c5e36d8cc9c63963b92018f421a269df244d781cce7eff1f39bb0fc8de2f5cf71232e16fee19a4187071b4929aa2cabcb8ace92a4acc091c1c8aad981aab323d1f99193fc30177d11e2cab94f08d42c4bc39ef7a164613862181628a5814210176aac8705594a60be155f3f7c35c47d3d8db63165354bdd0f40f4c40394e4966679a6282aa60938f74efa547db07a382037aef383b99a0b365a55986c410b40511fff5d0bd2838258155d2e802c314762559c152da4cec97eb2e39d56b621dae55a4416479b7141071e878e1cdfab14b18f1c5f62a1e37feb1ab66f404cd61c2dfcd4b3509df6a230527dc37ad536eb38dfe31765ab2ae2165fba38fedd665d4d44d1fcaa1ffb199108067bbb34cce5cf547ce42eb13a54b0b5300fac74112a77d4fa983b1defc80edf78103d31379bbc6c006544d807f45a67c812f86ec17a3e7901dcd9c1a37818049dea03252288ed9db37df28dad64190b8261495aad54d4f2375d9543a11d939c5895780943629ecd02a56987d53c1fcace2d2fc8aefcc9f6327e17ec977234adeec86092871ba33134743ff5a879183f630f43206794d2a2fab5de3aadd89d919510c4b7cb1d1801e1aa2c2c6d3b6063284a423a5aa7256905f6d887ebb5b86d16bad3cc982a9e671d387829737c7bf85eec0e60aca21763cb86337d3df0d8346b0c52f297bf1f732375968014a0a23211c1af77c680da4beefe9c30bdc4e2b40d6628b9a4736345d010fa7079dbfabdf14893fb76803b5bfb6c574e4441be7c6db4bd06cb54e4bd08999ba3d487b29a3b0cb47a9ed18613cce353ca8568f97c1883142f1e449d023ce47da33afd9378bd2f7842abca93b4e0b224fec187235d0edb788b524a7795b8267c5f3e35f11e130afc2395d43a2ea156db6fc639402b9ed18c9113872e19e4970e12958d00b0c66d3f694a63a71b9c3fa50d2122edff88e99342e1e4c60730621031cc4c918a23679cdb861784d7cf3b38c468adcc9b7f1f64be8c00f62d17302d2fe493b40bbeb43bd0810eec61f1b8155dfa4e7f770720e09a8c3ce44eb9fb5a8d6936f5f19f56a3ee1459cff8f2be7a64372bf76ad03a3afa2e2b7aa77f2e778a8e8266c062c80164153887d5ebff91f877d3fb5ffb6301d6086f76a9ce89354a876173bb1ba7c00f60e22f7b01e575c8f309332428ee5d23fc677821836fd4f2a95de9e2e135fbb281dac41a60bc8d8810ef32402e89cbcb114aa9dc5660b8d0ff4054a5b769f10156bad8ea04c3a0afc2a1052b9863d667fb034bb81d7f58493f02d06018c4c1363887beabd3a5b1e2be8747099d1180ec60cd2468defba5c967719fba80ca480dd6b104a882381f642e52bcab399387ebf7771cb60b0cf4cbc661af64956082dbc29971d36c1662615778f5f4756373c16f54ff1bb9812e9059cff594a95966e5638aed76cfcb9994f89ce415e29efa7f96239bf1f619a476e0c2658e3ae36be6c7c1fc80c49952c27371a7b620376d91907287b2b34ae02c16aa27e9496183d080b2074e409e90885226a4d83fe19c6d4aef9022391f1741ae709f323a25d62aa6f5aed03333ea04378a1e4a6ed45246c1e67aaaed52737697e5b62696844dbb05b841e0571d8c42c6f2eca73530ddfad25b37369af4b8d3ff864f6ac80e0f5be43bb3a968866d824c07867c7dabb7db55d7d3c69a4e4b5db81e1f58400e1f477a276ccd39ca4a8a3d15243fe9fa75efbb52e4eb89002da7a2a59dff5403a9f0335baca760728d648d9cb89ad86afae53cb2d68300f8bbc5a5e4db57fcf04464eecb779eca821aa5f326d91d2c38736a306e62bcc3238f7ecf65c568391069a69605420b6d5dbaee275df3101f7f0414a4801840450081bae77c7d531aff81edff61201f0f0d2fda8331c7a813d97384afaa92d3d0e97834c6c9d79699406b4a482296af375d6cb81aa6639f4acd25a1f78a1d1f03b9e585406381dacadbdeca752b3720baf911a0a376095dffe15998ed7204a57a817eac6ef7cc9ad75a5694293aae0fd225afb5c09906bf1a647e8433cf4eb1a30e21bfb0fdd94f76a0f5ad880df7194790b6a67f6f91394414aa42a39d66ddf3756445074d370ea78f58c053b3afb9f3d875318ac2852e086f9520bb93765a60d62b312cd56f1ce74cf043cdae87298dd1a8fc062467a2bbe8c8d9e7cf34e8cf1fd27d1bef9a0bce6a8a5b4d75a0915eb85384088628566ab488c76999d099ffbfaf89444637cdf6ff6396b34fbdef086d09135aaa40557366f4ae635e97062137dc1df3acd4a9018e13548a58084ea61bfafda34a4d3245cd4ac3b18067f6081384fcac46b96f881a1d155653043bd80194fd2fe6417d736a0d942d092f8629970dd58039ee12f5d9803aa5123215a9fd90a97c0608d8a0a0bcfb5bd029d37235b5859cc1175c3cf91f6fbf3b1e84613db745fae5466d20141695ea9c42c60f092158bcbecda586e9619f4fc8ed11d60aec585ad2f7c61276301cd55fbf2dbb080609950bcbf3bbd30652250a8ac72ce250fdfe59f05c86a9634ce4bdf245afa06adea196426abe4ed824f1a5183dcf8ea5d8e1879a986d75dc9a36c1b7c9f0c5be257009e2d7404b4cf8966817953cafcb9a342b2f4a4406cc8359aeeaff1fa479b69e8827eb21bbf65f14a191d53184a89f2c585b14e8644940a44b1dfae6a703ddaa102c654b67c00ca28513005359e194d443510c2cf94a27a963e2b9ca1c7fb5ed440b8cfa1894ca280e908549c4b17c4425cbfd014c22feec666daef4ef6230fdea009ea429bb7174b48ad43b801d674011bf05a728598eaab4b915e0af9c2312d29a4dcbaf2e038e7853b34d60b9b4886fc8907aa750d4e07a8bc23bffd11d865f92b73b3c1cad4736cbbb8de96c9216c80f01a43a0e1edd071a79cbe73262c4049c08ba835f5fa4c10189a7a876ca8cd880aa7dc28b0e41f487b2b7ee530f19a317e6f373add45864ca75026ba85d30ee7ce4abb728601ff4b2f3f6b8f4f4e13ab52b0ed0aa64744e87262a8ae44937527020ab874715d996d7c91c6e3ae42ec402b46299188dd84f1cb74764343f8aff397d67523723a322782d64f2a6a495e48c0584570a17592f145818a2ff35306a0eb01957af78c9f7c8abcf4347e23d44305829cbec486feda0d2322f92e99232870d11ebeb674b266a77706b1f27e615ec8c700392b6f19e2ee7aaee2ae118d8a2023590eb97da2d5c57f2b39cdb4b4114271c77cce042bd0d9bbcfc3ff3f13c0f51360819f48dc3f6c0bf50e2755ebe52eeab6e5e1d5cdf0b0170d6216a1c9b5f532a84e26df94c613b013673aa424d5c5be4e603ea7d5dd9812cb10b03bb414eb5280f66a0e89728cdbe7aa57a48bbe3189bfee9e44d2c80cec416a8406788216c3aa38cc21bd3dff4163ce355256250fbb0eb0abcf82bc9a71a891c02bd5b8fc1b7c86287921d115e46e28319dace442971be8055fc24726eef7ebe0c3e1a4d29958eb427f2fe725afe48905b429401f493eca6b02dcb038fa44a9531edce185b26cd9c193536dbf4993ef1ef04974cf2d8e98d657129951db9e30a0d35ca6c8cc242c2fb016a239ac83e2f1e5646d0acaf243540b38965e826aafca68fc8dfe09d6190e5a6b526ff70f35ae0409f0c864125a0b7fcc98c3085b5dfa88530edda8c65e7fe5a10a680880e45d82ce8725b8cd44a6a7a57cba5283dad21c18f962322d38e6853ce8fa2a1c5ab77da693e8b9cb1e1a44d66d2bb0f8cd9c1815f601d5a91c07d6824487f617f8d7eab48c45ebf732b72053a130841ab82f0f216022ca0a04cece4d5efadececa2ba2ac588a456171849c34de47fb71134d04a6cf7a1679319656861ff1113a324e1c70059556bda9e34dd547b719cdba5594301dda8feb95c6ce6e7f1c919328b8e7575ffe35923aa27d3d28d3c9165e0f7f4acc547de8c4c0cb9fc36a8c2e3909a85a0e9d698676e08280de2182b01661f63d1fde0648615ad82cff1a34132f4368761005bba184f2c2dec226b8c7cbef973a4dae7e14c6304bfdfb3820b036553726fec4cf52111c18dce16a516edc388080a60a0418a046e46c702a7f08cef53450b346f231df094630b8171d2065651555b8861ade3b278fe53fabcb8fac5085bf0413e06cf21b999c6ba4a93d8cd64727061d310b8bb92fb73dca9d34027c6941ec1aa42e403da3f1800b43004782c00672793d7ba983d7493deca72eb4577b8d8e6a7d778b24cdeece0cad4174f18308e0e22afb1789da94ec8ffdfcf22dccc1c9724b2a3704da4023821d8fee379c2cd1190f88ee8bac6bf277218e9fce19bc37a844d692b95e5a3e0926ea5e7a705974d68e5cc2dcf568716b59b05dc75633830508ffec467e83bba2f0d7b9ce51528e5513a6c414afe3b2d5482c089c9bfd796852fa6033b2b22758f617a00f04e957f3481bb28c0153fdedf0133a833dd2a68bf87cc71359ed59e51f47525c24d300eee793d9fc2555cc81270e00475d76ef95fb4646779c722b9dc7fe27f196cd9798faff8ba365d877dd78698d8dc9a1773c48c70647430dd0db2bc693f04ffb71f9e4150e9eeb796086ded176fd3cc6577aa2132d4cd7170312217d13ca661ebb3653fbcc27a00bffa0405dce92a8ae3c1961039f3bcb480869576c3f96a38f9114d3fd897326367bc7f56a10e00e818c4c24e86f405dfa61bfc98db3021b1931b99bc75fb8dbed643d8d33a03162263c7e247fba8c2a874abd622da9e197f2f81bab04d49ed08d4eb44827148bfb5e8c96b91ad301e2e3ee063cbb66a0ca2dece92bc47ffa30abc01e21b709e0ee576debb5cb04c1ba1463679491c7cc8acdeb14aafbb78e7063b542a88e9fec8f17eb5b496113f095213fbe6a39904a2e8afab82c646dc2307a5da616a714d1aa4ea398647523cd6ecc11e6bccc181c7f9d2085b339549a9b5f0f62f6f266648fb75710754c8ca66d50443fd87d71169c264e6e705ddeadc2f63890654ac3980d8887c8451d491cdeb5dbbc2652bcd134064256c4feb67773a115a98ff482b0927e3ce7034b3a37ac125113cb99292ef57fa47be1a1f8d07cc3a28e2972d9ab642564eda96bcb67fef1d8b30ea1202c7607b1c040af9cf42323a56c6a5c1fd32e9900672ad705534f115deab3ef56a4d1e1ea5c0585edcdea2877b99a54dd1804c1ddc532e60db270d8dc6009982bfb9b0faa9b1b41102a843221b2a7ab335bfe3877b1510ebcb59be45087c35419d9d65d202f37a5845932dab6eae1cb77db149fef8213382ccc454ba25c12ae8f9d22ee457752a23752ef114c9dd505244baecdd184abef9b69167a27c3f00a8ee1d861c8e3fcf788ea45b8a296860d8d5bd8543c32487236aa4d32c89f270a487c6bad7e82919304b788661b80d89b4d5ae2d53d984632b6c5acbfbdab8ae06117120d93dfa0164d08b1bec71730654145ea25f18d93a2740bda0677bab6684333b0e59ec554d73db7d57d87eb46aaec848638dd12f2ca9778456e49f8ca5b1ec4194ed25e3eb01b8b7336677b3dcbd6fabc1e25d5450a808775f5a881be3065093a86ee54ad841ef91769c2c23b12311bfa9e6e2f014153b4200c0bc9d7102621c9537f898c9c67f0c8d49025cda4cc91e1cc4b87857e0524bb6f96395ae6d68c5764ed1935ec288d38818a255f0d6783b7161c71377e5eb1b4fe7abb5af99d82dfc790463ef2950a54ebac0f40700e411cf46c23d19c2c5a7d7086ed2eb989383790da6c322160faedfa8e9be6c6577f5711190cf12540e722ec2056691f053adb43b72623295aaa87ba6d4d88863568237880e1aa738ce54126aeb3eada1bc7b682834f418c33291dae751fc8c1d8dbcb9926778ed25da35b3edf18450da896da2058d6ff81cef4f23f673115388de55cfe8956aaa3c28837637e438eaa232afb75c3b4d13724d9b80d6d7a1b89d7825976ca21be2b3d2e51d680c681b27e9561bebf66a6a75d483426b0ff40b3826e252d9264400ca713b6718e5fac6a3226a3ade618b00e62ebc38607de2dc4f6498a0596fcb5ac98b4531557752261883f1aec6110010f794199a271a8d91099df5dad4b9ba851afdbb5a852c6a2e146e91287a9d3bd2be968c504f30d0d90841d571bf6f81ce0af082ba0beda3b03ff1a87cfbffcbf2f64627fc0df216733f8037974c5acae7ce57be167d959d69027b8f56710c321d6fac2e08bd48145c14f699a6a923eba97ad18d296d8e0a873d5dae1f235b3a6c59dbc5f903099dda1c9a19b0ceaab92a90b33e983ef54ea7eef6ba922162550a2435697b283bf7bec71b8ee0c41e3598b93e433026d5742ecaaf0058f0ec93dce2c4aaaeb35f23270320d70e413917f06c83e2eb51a950ad09f176a7e9b5b0db07073a6884bd753af1f71c8caabc3fb7e23dc68dcd6d5ccc9a5ca13b20a5205443900934dded378ae94dd47c32db0ead99e3cdde07cf2a4db22d92c7839168b543a290615b581d338cdddb495155ae3238003c6596492611e9b2bead4490e77a43ab389de68d17fef3e164d8b9855f3296ab60974b2d40f50cb6c7f995f770554e0037d0e5b357266a04a1d3559a8b8a4249c15387b893bb116a64c3f99dc662247ad1b6a5c112251ee7cfa7821d8fab1263eacbded2c520695a350001b29225dd1c79bd22c902d9ce021ce2ae79d075aa8add81b843a1ee888ebe214b1699c8cd495c1e7d22a290151cf3b2b9816702b51a38e720038ef11ebacfc2113cdf85e609e40af958e3410863c444a834cab5a58bb018ea173698d7dab894be9c3238c8338926fb9db8b30bf8cc2992d0bbf6678872f1203ed6bae41cc6e8a7ac7597d66d4501c29c32ada910c92f2e269a9fd1f07d85f781b4f5dc5c927f1004cd8130c3e2f90a666f080abe50e37406bfac00c2cc9e97366420f3d55fed8780c396cbe8b2aa7d4c15487249a1b5c72308ef3fda74025e9632c34c3ce8151691eba0c849e0e9820392bb2870fe4777173856c7b4c08b095d376efb7bc91c89b4da70c466703ea6aa8790343a2de9613a26d9989c64bf10290fe72ae8dc101d7d13b8f31a06ad1b15fcce9c0f28009ed1a46f6dce413f84fa01f9986ad98aef47c07edd66bf2f8b5c34de6c23c4e60d2c7323beb34ad052eda1c6137139ec9e38dc1b263ec410987a3810355d7c0549c2194392fc2dc78fd278496b4b7eca42cb512975f57e143a77ff22165005a4e6b687ce6dc6c67491f9cbd4dca2872082240ab8d3afc9f2261761ddc59d0d99348df871eed121ed87818a7f5297f087392c8591809a3f57e4739f5432417b4f16343c5c5bea8b981fcc8ff89c829e8b956d1b33d83d2c7ac035470d4f6a1d40209f88120f5800abd89441f239df9659b8ac440e43a6e5e96c48f7b3b3df741bf173ed728cb5c3f84d12d3696eac6f54917d57049f52ca126022a53f689768db644b24784aca12ba4323fccf9e8e674dc0666993dc80902d0479a73de12061e5e2c57d0d50beb89a2a9494845fdea402dc7b8c5a0e6b164de9fb243e39f74cca3f845538a6906bc3ed1e6dbd1994593cea54e7ed9654fd1fd40109368f5db698db058b76b4b9507a2ea9cfc6fc76ed7406713db763f7ca9e2ce77733db99ba369c7af7cc495492132f80c9b99d06b36af63b9f37f16a028ce15115e666b66c79e53c85e2d8bdf60064ea9011f688f862bc04faaeb5da108940742ba77c563827dd7458408f47f162efe3e6a527f72101532fc346b20c1fb26ead6d5bfe9411bd242328e073a5ff395d6c494f3eddec192c90b8b84fba169409948c7a1eeb82d38b332e83925128b76c2b9c9bc3d942bb80cb015d149d3170908e57edd44ad78286580f41b4e6c075d79a7193a72bfee6ab64859b431bce904b92123c623dcfddd925bb79f4d18bc17eee66447787d5ac238501c53b5742107d918dbe198073e1860a57a00e6b2b7617266115d155968a494aefdec55b3606ee1228c74f7d2d4ee82913d0d1bc893dd3fcba5cf63aaba8340b80130fd926494086386a0f5b42d02fb81b09d1d0e15c55c9a8dcf912947b017e3b5ea08d0de0015bd106f6481ef567c9d335e0f1ca91be50db2a47050928ac3329fc2a6a3a18062c622411899aff91f4199ff1e9064e9581f26b0ce2b5728c0582d77cbff0ae34311702bbac9e78353a5a94b15e7216640f938cd74cda7629d1700c66ea4fb76db0f8d7c63b6dd96377f1c15ab2ee91c0d6c861043d33bd568e515c78f12296ffd1be9ce674e449867c8475d527d07e7267f761ee9b4418db1d860608ba9eb27043b41628b7ab58cad8f6c2c8e67d74138938107c8f1dad1f64871e9a6606951826181874de815b5001d396390f798782311d62e96fc63b4824cae8f2c1a602783d28501f46d02f16272eda896df99baa2de1e01caf888fb037c9a6dd8f0c843f434f0e30cdd6d79ca55c7e2cc53ca0549fac6be589aaa499249c0c4fc4a06ba4332b3934f26869c5d068ec3cee6469ec25d5cfb41d9d38d8ec6e9da379db31984bdf7c5092d7107f6fa88ad2ad62812e45f4c23ebdbff5a1bb643c359c6200b34d716a56f1b0bd1c200f56e21e24567a4a64b7a41682f3c9bddb45c10dc75c0cc1f00aaf384b942cab7b80b87c8b7edc72f63317707ca843376f0637c336cf6bc84374262800d60d442ea67076ae457af53d861c4bc1e32ae3249a57d153bff59546e3454ae7f9c9c86cefde42de7d306cba2c51ca21387eb14dab6e4f48089084d67dbfca163c9aefe8301321782b038f64d24acfac45eb1024082157789a1a57281fcb997bef8621af4c1e15cc7eea4b46cf50d334f5c055fbe1301558e3850a6a8e8ed9b99d509799fcaff8368c5ce08289d580298683d0652cd936c26c940718d42b2ecc5b4e43a43f40e34856939b36266d31faed6ba19be7494550cbc015bd15306b1caa2de6369371574aae6f4aa64acde396c071cb7b5c021879a75637c7b32621b6b7bcb68a9dba9b6a0bbf329a447ea9d12cbbed14894d69db672b27d1bcbc0cdece7f0680b84d36baaf4db331ab55e60bb8fe9d957da5de53d6e996de754409a904833c85a413a72c9b2fe822e774db206db8c86b14386a33bd443eb5a0178c66b47d32317b4242c28eb90586f83e04f684dd20992b22383010402f04a8e6602f44937e12b35839318863d6559e0a2c0830162ceb62beefd586050225d77ddf2bc063933f57018b636756c19e7e5adf32357bd75e795663a5dd7182394b5f0a6996f30e971a7886ca7e6a703cd0bd025921ff1c9ea1d9342f7670169c88e37623889533f59960beaaa860361159667df25b2b7996d8b19229e9449cc69aefd82f57531d2a5bb3b0d6995036449cfabac4539daec59e2e403af9e405334ac4b6af851327e2e4af5a1299af4a500390c130d9276d654b31ff719d471eff1a41c745ad3a5904622b240408b6b43b6293ad184231809ab9aa2893d17cf2dcc02a147b0490bdbea9d072f7a4020ed2160a11e2e08875c95f8fcacdc7d7fd2e488b2c875452706671658bea7c55886d033bf74ab3e34c60a822bafc306fef097f13fd2ca6bbbc50add539bf070a6e387af1560bcb0a1207aa3c5560214977c2d51272de0dae265bfdf72dab560dc7d44d82326579f91282a85fa9aca2fa4fe89dea92072c05b30fdb4cc94c2cb4ebfd4e35fee5c71f8341090376fbd278ea178c5af8b04c44a5cd275e6e0ffa17c2089b612dabb37ba2d4a36c331ce3a2b50f760e3b53c8f6f375a44cd23b11e26e7800cd975acb325ee01662c2ca4203d42930eb1c22d7848487d9b17f6a05101d0617ccd88da3af6565c6d43f01f1d9dd11f004374adb64e2d4911de00e604787b8b24748fd8c4aea24c3792bc2917319eb3677b648e41670b3f31d8a9a3a046e796b506bb09820829620a50293e24b2ea7f1d8ac576d2646d793e5d2ce2d0be2b26a039866cccfef7f711379fb13387e3fea9db76aa74191d62f7ac771557e2fbfe4ac89ee4eaaca814ac2c70e9b75fe4e5f477b5f5833f88f95721008dba5163ad261ab1048067b84f96849a30a4e0cca7edfb15d16126c4b931722bc66a168af9c01c2b3b043c806d79c9e7149682406a0e6812fd9c1c5bc8d88995c378d34946edee1f22872e8ec44dc1f5e0f8977fab4539da5f5125045b4423c795b8f7a30e52e94c02c3085b581162060c2fe80d89bdade0f792862882a3da0762549bd02fa6b65054ae7b287b32407499bc73b0c0c253499567379f39d9a275f6f822f29eb56964d949cad2406287590945f9a64a0f88a13b5edd6b7ead7b9c18a95383b07a034984f03405f9a8ef4d587394dc80c2b49c88e9c335b80e14ded808c3e8ca2ad8808b913ea1446129d723b82c9d61762c1ff6e7796ffa8cb523dad63fb10d9bdd97b50d596b0dd856323cbe6bbfc2655d8e19f169ff95f056422f159d860f00bc55ca5b157f8ad562cf7e0e1cd90624d66c0cf4be84919b600ec2643c88dcc75c6a79019f04d2e37b0145633761334910de7a34267979182f8e8d2b0adde22d39a2dba22fae9fe2edc4f16251c7a5033808a2c041cdb246601c3fad965dd60ded258cde413388a3bf88f0d9d74d267c78d9a78e42424cffe05faa4bf25fb5f7849601814754d7f47ffa39d42d3ffe6b458f2ae5d0bd3f2a819c4dab0c26cfdcc90258b6e82f2108ed2fe9e44b013ffcef875ab4c88641c6419a23ba13842a3b2f029fe608ca6aded9040ed30e64433b85f3cb68cf1cb9a61f4e2dca6a909377a1b424e5dd4e476b816ae2bc35ae9811fa5a634f1c51450e5f9a96050b60e562c771a3338d72758ce67463fd64ac469e58333a321fbfdca8ce4115159cb1f4dc5ee57868a2641d0d21da0d5f5b8e75b54afb42bc5bde731969b81323f4bff5b79617d5ed15ad8067ecf23e4e8604b37f5e6b3bc97eb65c1b9352e6416ec32cdc478edebab0ce62799545bfbabc01f137f64861132d886a8271805d57589958078370a6735d08a09c0ba5cc487937829b24aac268a53c5e7f962fc76d72d1c4ccd6e1a30e89d33b66a2c11b7d88868b3c7d51261ebcda56a902ce2d4cb021a7143727faae7b6a77022e170175f1938cfd84446c9eac19bb3bf372d33742396f772f280fe3eabea942c1414c09b0de73a0b6ebf83fa6deebc93b1e6876cbdea5b6f661b6499e185db54053018b5f891b1c6c0755bc56522aee59a7140b32ab6a53432177fd0e628dead472477058351f1f77f3a93267d78489ebb6d4574c6a63221dddb0a211746202d9942ea0f4f6b50184008f9acefaa015350ed69d9cf36a8ef292ad10a1b6f1f710e46f6b1c6a05d3e8e3b926041fc76f37708ba65d3ee647a1ce4a5141ca5e5ec53973d6e690d99707ddae01729da6bb3853e6017c4d919e24c147d6f0ba518c215f515d3dab52c9efd032b209b09b5488fc70d5f9380fb1420851ad29dc879b5f0d0a5df4c4a5535fe1a78bd4e91b4e555a1285a420bb77edaab5c2858139ead60ce605cd21ee6e0dfaa5e1ff1ae637cae87b78b75607b11369b64c49381d48c6c2b09173a6e39c98e4a8a67639bf038ec56910966f2ac497863e054698c0e11d375bcd4b9650db107ad024de696e4f4600d3a1f8c294d682f00ca8df0c205b7ebf9b65043f3469431b62f612972a1e57afe9a5e02eaf934c25f2b9cf6c77846ed80d03a5a7de383863814c5031ab34acdd0a8171bff467a3152da8941f0bcf9cc9f44e971424f8586cdbc993808eabcba30f8b00ce2782fdbd666236ced116d67d7f8669ce3eb5b58d1f8bf3d8f17ec00d60c9557cb0d32612922430bf852e3056b96f8256bb06f7dbeceb7db6e1f4231be08e90d041950af2119d5b90ed0b37083e5294892cfca6e4ec094594c064c1831cf961d5d465037ea003da0697b8ad1fd96e4f5cd432347a14fda1f5e2e41eb3619a9c4bdb370d7f42beb368703e3c22de5ecc8e1a43c1c35f867718d659660322c8ab6f1d28522515b815e8d51803379be608566c911908940830d8126d1cb029d0e0cd1319d3c4c9550438333a2dc6fa5c43b90e0e7d277a78f64df4e0167788c8f7c186017b8362f33136c3c3c3aeefe43da7d7103c9d35b24634e094fa9f7df0bfdba14122bc30c89dab08528f600cc32fdf0d30de681c41f25b034890775435d69a8ec76e95c10b608255c01ea9f08b34c823e39c1da9d17c92155cecf207ca2076262f3b3771172faa89d711008f6109907a6e60035f3238dc90e013edab1af3dee9e75e48a66a664256ca23d317d9612284522829ca866e55e4b23f400a73c72abc243261e6ce03c7beb80920a4746bd9defa87deaf8ab520984f656d9147d0dd1a1fd5f4f312fb769a41844bc29e7bf2284b6b956892936a6274744d7cc2ad5a74d100e29589b026bb47935d8516cc9eaee4bc54a45a7fe8acf9dff7a0e72582ef77f0b13bbea1d855faf1bbd03b5f9f2497bd291b413984ec99395db7b1a314973e1364bb448359cd310358969f84e4c3a07abcd1b008ec8e03effa39815e7e0400042e0b143af8c8d37a4d86a6389547e6756c83636f84aa92cee361a2b44992e82a04e4063988b682663b67aa1534372590c8ba481cdc263f67058c1ca8105faa150a6825636a50f48ff9ded797e163d3b1efec57084a5256f1946c2960d46e09a50c8fc9e116a7363e4bf630c9c661abb857987939fa39f2021b5ca5f8f4c0e1601048d0944b6d8ea482584c50d0558cdfe122d5de7e4186de6ec9405e806549cf1bcdd8c72c34f45dc5cf302b40c5ea4954b3c6fa4bff0237d01a26357f61ad01b18c215344f99d31594b2bfb0b892b524000896580cce7aa0d9f848f5870703409b9b0f0fc3cef92bddf2f057c88ffca0354e5ae9f2c6696547d18087c2be9db6e9f0591299066c50343c768d6f88bcfa29034181283fd72607591514f3ba7ea050e301194a8fd791bdedc5785e3a232dc31f65e2215132c6334790c3897bcf13289104df19c8ee4214ee2499d655b5e4ceb32758b5dd4ff48c29ebbf9d63ebc1ca50b2f93e18f13b80aefaefd87c9b0a69c3f12dbfce1be1b37195e3e09e6d938c46426a0601607d9f5c55cf2edddfade80f0b9f598b1d2d16decc181d89e60614dca6661c9b24adb05703aa20a5506a348ac221b0984d01d4c821bada55553d95184af24e53c264c9822685966de51dd655a64c1dcdefe3418a9ace7b946b5e6b57fc749a2358fce9361a0ca5914e3eb7213bc794adc416f6bea1a3b305ee22130d66e698e491622fb1e0c56275459217df8b456238dc4b229d9dbb3bff5447e345c402a22a89356915db88c5015ec0d9ca25af8f913f96f89cea6eee80202f71bdea0cfcc85572d8971c55e7ff7e483466e8e6cdd09775a05fe0b607a18376b4878c28ed0a2189ea1c9c42cd7f04c8ca54cda869cae8cc14fcea320765c4064dfe251e7dbe42b8fed363d31f274ef80bf0ed7878f8c89999a0c46f01a47131bde101ed2b69374dfc821994a83423557a25131dd01ba54e88cc8de622720e861e07800038f8a2ecabd7a8e84904f5baad1b2b192cad27d0a7d277f4d540b127f3d8c04ba71bf04aa3bc6bed3644a52edb794f180cbf1a4e3365144026a6b28dc22725ea106a6313e9fea00b37ba7bc99fa9157950b58bbf15437e8b008f3bc2a9bca17ddc1f98454be780e8c05e9c79e857448d750286f42d898b11ab983e23ed1184488e474f79782667caac797f7de8b3d80218a7d29f5335d656379746704501d3fddfd5151551a36c86e2097a9b77148f84cbc62fc8e1c52d7538b6b87105e1fc210e696ae0e9e341455cf88effa297413c60b75002c48d687036770f21b70e2e366ccfe8ca97c9b0fb10b19147c93d8b2b36c2688e4a6241b9bb842bca0f5b5be7cd4aaab926c23c815f544ca1ce39d95ee08d5727ef29aa2d5c59070b129d4812cd1f41b3078586eb4ecf521bd3ed2326ffac7e2dfa2200de58cf77f0285c58efe561ddd41047f9624f02e49362c0f31ad413bc41044bec6bd091a60136d93e60264dde043c4fe1eb0b371d7f0b383f125f895a36d55f797ba7c36d31533e8d153b700771a9375cf79d8c2f333d05a6ef5031a0a05754e6ad8029c9c5e27e1aa5e81029a8c5c8299b72944b341c872ef799d501f87d9261c2931d16d70824493348e76528492ad6acbc205c5a2d981cdb08824cc0fcff8e5b5d85376cfbda5450563b8f927347c8c5d81f05c4e0fe83ac58fe73a679efb9679c1c41a30f2802f071b49c1e2ce92c56420459b174a22d6d2b93f3bc939594edc63dc6ea7e7d291df8327df2dbeefbc63f75d63a83a9609e3ffafcaf057faaffcd207089cdaaf9913d143bd6270130fc7d62546a8ca441fd788d82878b00e455085b6696bcbda2ef1ce8c6e4ac90bdc2a887804679d1ecf79ffc5e814739104b2df2dce29bf76fb53f42490a7a17394a68f812994254ca295b7ca55832dbd6ca6af1486dbf14fbbd10f1960a5ac183f3cffa3262eb753736ad2d8b3babd2a9933a6311326df2e5b828347f519d91b6cb7e4d28fed9be008cb50d690c4df67b2cbbe4e8e5bc8a560454d5e16525e1089e49e3ff0426d68ebf1e23b7ebc41e315d8b3868613b4c019ea59e597b328e1978f373210dc187d22f1b1c539c3602f280eb6edd2fe9c6cfd1dee58b8fbb5230f99b7fcda08feb0bb39b0c3391362e27e6e19efcca3880e8b234044e4bc5f2961977b308a209b44e7864c569432517c8f0772eafc5ac81dc75f6bdb284c0e2a5dcd38df37fe52d44d22fdf1f269fe6f29df863d1c7b610c04443fca630dcb8540a01d83a7fd339ab8c723cee39cf0a37c4e055d2671a4ef2c44c0695a81825cf17edf27e821157c1fc252125d6fb8264533f99fe2e2dd3b3495f8e82ce51c852f6e9a625c24b14ef9ce8849750d4c3dfca57ee37d583764f74f2e1033d51fa95f4200d87449a73a06cd3c4fd78ce46a3477efba44687db1eeb909f9e96e3f7778c076203e514eb86bd84ea0404ffe43a38cf0144dddde3bfc662807e448636fc7b16653fc6959595a7829d19384f0371661d37c599e4853be8ae2deb3bb812d758fc833e0d915b9ac33d19a7db4fceedeaaed3a13165e9a61af33cc136f61848d89b0ea34ed9c3ecb6765c9bb26a2e2c967b0483beb6c7546fe0b21ceba572b0167c1b72c09d4feb83d7581bb242f96f5f39d7a2aeae56810a2bb66109ea01edeca5d38efbf260188169d9fc2defbd7eef8cc1aa437e6801a42a41e60769b616bec40b104b52a77d79467639dd1a65a0fe940a4c7c9a9c58e7a6805ee302afe36698f437ded35480951185a6e15bc6466c7e42b6158bff6a1fb0776eda745d1535f785f7f9fd1ebdca8442904ba8db4842dc3b4fb6c4bfaae9cce74126b36144f550a42d2825ffa8c540f7c0639c9453a390d64d4d00b50333a43bfe06cac2807c35b3887e06a47c3646dd04c121fe801c4c95767dc1c9d06238751a81da09792c277751d0563846554fa4c25ea0938d3b7849a539ff5a44a31de18835c3ae0fd275992ac41b090cb92ac481ca84f9babc0a4b6193090ddf4dc7a573ed5ffd523a0196c8152ce1dc9822ffc7b4cf33cfa43c877da3310377748aade8c91f0a9db3ff5fb13e626fea755679eea908b4d1470424bbff6ae15c60f8e763115b8e4cf1311cd719f0b22871f5fec86b1b276ec7f81fd5ab2fa800a3d46610e1b7a5c97b5e18f00a27c7312360efca4cad3e39b6be6d94bbe50d2ef7c6e70042d26d4c205bbff0814c3a22cf4a5885ad899b969355dae41263d48e41a0f0b3c14479811106d0ad9351b9e59a76ffdb4c203a5eaa29a450d6c8469c4d940574c213eae5d0baad9bd1d1eb5009b09e1b165724946f0ad1f29f8332a2d22122589da3985cc8949050b241387fa4f5551ea94e15d55b19c09ad2f9c210e227c78bcaa93ce4f03b64064b64778ebbcc9b0552fe9fcbb03b57e2b30b1747afcbaa3cfd5c190ca8c650af6529a53a2248eb24de9a9036714a8e878fe1ab7bca65a3f85abbefeb1867963ce55d9e8944bb4edd1074932fce8af99d8d8a62c2fa3afae2c7537d08969fa4b3b9f9bc77a0fc286427e62d6bcb871f7e0b6a3e602f5a1cfd46fb56c2399ee4e79d1d6ed6ceee594846a1006b38b9bd66a3d7690f725c402723bac144bb4edbb33e77eb3413ddcb437ae4b9955b0f797de0252b9932651257abcc21ae564926c2cd10c7434632faf6805f3adb565b2d897cda3f930013a728678ff2ab3d2ac48fed31f520d869441c8fabd9738465c7ec3686c4d076540302b20a1e5f3c3a2eb9ea52fb6fdeb1d31536842d49e9490200d054622ed985a5b444c2914065c2aac83f784bebe008eabf36b3701f8abb556d7f2e4cf3a53e3c9919b79fde1ae9795244710511c02006fad836e092f4389c2f40db2455011e76d25cf37c42164f63975f36c7c0c6107dd6b4191c3e4e269aa59d63681a65ccfb87cb1917921a423babc9fd23b0778d46f3de3e9520784dfdf638893a3171f0af878dfa474c72e2694dba98a974cbc849d14c94d6c228a9a473e5e28f18b3175332f73e96ed6b0a766f4bde9d730100a8dff2c602bab8cb3c7d06dbeb71982de04e3fa0965fccdfe4fdf992b000a52a2ccec650876b78f2551d5ed6fa39c0f8da3c3c888cf10017823d0666252c1f9997257678c9b7126310465017b190da25acccf2103bd1a8d9e7865b446058e83e11e8ae58f5aa1469b21c37ca4fdad094e7159f1176d4075fd81557ac004b84382d858995ee03a274c94cdbcab231133b1275dd9d7f0a7e21096e0168e9c3fe9352d22b875e5bfb947dff7c4fa31fe791994028f8d0d7bfe68c4f5c33ff5835a9a719e6ebfeb708977d0248b368991b79afb7e53a88defbd7dd6e2467f8f3dadf9d9e80d559a5d4ccf3ef309054a6a17f9cb6c9b1a2f3b4677b9a83661a362be6a31613fdc22b65e36789734dbc2181aac007a4a6867138456802c5a1b08b49c965071bb3c0b534b0b80d181ddb6763a8510acddc0ceae125eea7f5a699cf6d36338d4820470862f02ed2632a9a8f65c3fd514d2dabecd93a5ed16d48ed15ae56c399cea804bc8b205be2862f4a1aad935781d3e24a675509dbdf465128e27a05287f91ad58e9e8ce3fd9753076dc3dc5c25598a959ea9b0fdb9296f6e2413d2c930ec20d0c0df1c01dbe23f6c243a57360142ba710a305c9cf88f046aaa74cf4c03b7ad47b125333ee62774f8ac2ced695e399ce2f242535a063519b6d7b64814a08a7cb6401ef70a5ba0deb4e65c01cef012074ceed9c70ca02fab324d0293ba1b09734c0b30f46508a4f72133b3cd8e8ecaef7ae39ecb970db71db54c626d8a9c5eaf509dfd9d95fc150f5cb3754f275067b9bb6f2401696033a9400c74a57c27dcf35c49083c8f7868c5845cc43686877cbba0e6b2a248fe12f179f61b614cc19b3ea8dca8aa4dd2c1dd61df0019137ffa3c56596dcf70ccc9793b8cf96cd347c1df9b3cb3c08f4524b494c38d945e6664730d5bc9ff55a3173fa83893c8a200f9e7ef8ea0ef8970c4a082c1fcdfe5b4533714ab3888976ab53c4eef60dca384d027678d5f92441c26eef73589791a46940d79490568835c60abd1a92b575466a36e1ba3c1952aaf037a9bd1d31f0e01fac17cbcd229fc6b84ec029c80a5c7f0433f38758abb49602fea39f69e7e3d0a2382ea6657de1ee45d86d7a9be0579b7b6a94075665bc6d1c6af317e7d476981e37d87a1e2ceae602acc188b36afffdbb446e2a2b3dc8ba4c3071a7ec77dc97a9f23b2cd768d33c4527986c0768ced306751976a273d4012107557e3cf8eea583012d0e0776171d1d8da5b3fa1b3a2fdc80bf1165269d8482d80d55c28d7f8e6a41b77b91491c7fd8ce678019078028adcdebc57cd38e53c1036664226cc8505c0824108c6bed3f6a489456f64e8946fab1193fe2794f3ccf60d9ba770663469efc1c9f7ff4434c33275dcfba936af824f4ec90dbe89f337f5bfd8ee8b05f7acf4cfdd54f8d3a8011a4d2b57a32076a88c891dac30bda22fa06ebae91502a9c6876cedb91dd7d9c083582d61abefb953a3165a809150ba7aabbf865a7a3937930ad6ae729a45d394e5471890dc9d6077cd652f28ed91ffc6d3aefc96f020c1561dec33cc181898200e7fc6de1395e8337fba0e0f4962b346e3b16d7d3693d21ca0b5b3b69a9b2799df9402647cc63deccefa47ef7d5d2f4c1eb98eaf6203ffd9eb3ffd18cecea0edca95ba3a42d78dc12af7518bc11eef62ab64813f0ae50a36742cd435b4490f936e5676826e156863f70bf62795d0439c66aff1f13b9e9c047f405b2fda5781f59522b2255be2fe9f0867b6ca0665de9c9268e2462caaff3d804e14fc86f4fe0dc49aa229f2f8b0b8e696045f3f40e702c98b842bdf865653443f6e000150b58fe57faf959655fe00b7a6411a33670f93e2f07a49a5139298768410c6b76c8594d0f4508ab956d9f2c3f78b1aee8bc2c27c598611d88d5ebdaabbb6aa83504ec4fe6d0426d1f5e0d07bb0d58c2b3e82cd49a53df1cb447ab2ce984a13b83a4b80f7a6e741c36277c5bef3afb1a057392e0a244a8cea8815dc5c62e5cb0f67ef1c288982e6f6c4825815debb303f20ef005a7f81b2b0cb7ce6c99510634e95b5474c9e91b764429d5ec1c3edb2e3cf6a6c78df0bd1c1c03e64df102611396c67281789ffcac851b388da0b5b3142ed590aad6cb75e0876d87bfff62094618eca5a5603ac9030f97b74f9f9278c8bb0ee31473650c6cebfe928e1853c6ed4585c114ce525fdd540b0f8d63b9fb0e9920db8bc8b4f23c021838634a166d9fd98bce17a5a0403a47c1d46843e7ee6b78e503fdbd5001f6aa53cd07fe424f10f0cebcdea4af795930a05d9a8d253d749d7fb6f0097cce9463cf465dd55d8a42387898a6dbbb73cbed1fb0faa83e7d06a3437eb04478410246950ee8b3c0005613907bbd0a126cb15b26e449749686c6454a92fdce127c654251b0e0e561dc7cb239cd9fa7baaf8d3befccae1a5776f0c74ea1ef96da2938235ef6b7d87b5889681b3c4c5add9c1bf8d63a494ad0e6c75771d52bb1e49e0c837feba2018c1b3889cf1bd01636f727a88719314c1181984dfc7c4228bdfba77769170eb8189b3452d231542708ad70f4f22ec372fa08ffe038488599116cf6b8f8dc8f49b9cfc1ff6d3e63042c95861bf64342371a8cd8cb8acfbeb96bff467434ef1be80daff995eb3875c7a2e38bde71b50997d39c92ba0e94fb4b39e9f9213dc1dbb9ec66d54cf71c664ac5822b21109fccf0026f4c428888d91ecd8a9458287a5b740297f8190af3e5c06ab1002bf994fdb617fc618e37ef280201f65988194782f33f3f3693c150fa059af611662bc8e39dbba16c49b130e4a11fb4abe202bf6dddf251c8d90877ef73bffd9d73522759e511a9b9ca763f44f3b7f93fda105cde8d0f3da5603bdcf9fba51ac09801fd6e0b48a3a6851f62771a5f98aa587447322639d6fc2413e9ecf6f92493481010dbbb4f836a43f50057294815ec19230913301c256cee6713651a5af7683d1c893a7bbc34c015cade34e6e29b37ba1a04bdde09f2a9b92f69c6fcc42ac1f0169e09995b4ed9e472f9eea724417c7c1833f4f1e5e06de4ceb61ce6a61ddff1ec7099a44be13ae844497a018392006bf1927fdeb0cd1e94630a8d7fbee4e79beefc14b6f747b0efbe26ff9add67e3f93d5899c8a3095eef4be7f0c5d5118ce2e76e2e8c75818688a341f1350ecf67b55ac8a4ed251ec5ca26fd98140ead1c64a7ffd8a986ad176c98ef21085bb33bd73f997ac4ac4735f7acc17271fa13366c55728ebba36001b875e05c8debfad7f3966493068e3b4c9161a7ed88c17bf2fb0c8d6edb7ca9cb6a206b72b90c6bd6bf64bb4cb07f2100cbc5ecf13130701a72a2f46d7fd7aa3ce939c1573a646ba82b06978c841aa7132b4e46bc0f4964b34335317f384ca01be173937253d1614d32ae8e8f91d4d88a0651670da32100a7c284d6baf7763679ddd8d03523dccc86a8795469eff3331ea8bc2358cb91216fdd1c1c76c5c3a06ddfd36d7de36c5a019b920e40d16b05cecd3493546d906b42ca9374fd6a65b7b46b82a6b27039f522b93b327abb5eeefd302d2423c3cb578b003fd9409b0f83d7b863621a4c21fa26f8979a901a1e25c71849fdfb3c0071f817155c48d5b9996a0d42b7930a6305feaa91ce28010d71c4582c60a7e0dbf9dd28658aa4076ccd410b6c85a88066d619a0a566e386674a77e4ed5c42eef53f85d0bdd9baaf2f3504ecda97cb6eb7dac921bd1dcf826c881c1c34ebb440706e15ca38a4050b9f8a5851d291deb51f07eb2b0b73222ed20a4c1d191587e5923a6a6460f8f541516a92263321202f2dc80e2218203db9d4d89949cc57516ab9eecd72c1425ca0cbe900c6255ab7bd329cf5b659e72f86780adb09867427c683bd2616bda9bfd7cd7793116ceb40fb07c331e741dc277c56d3f9795a05b1b7d1c7601a427b009405f062ff9f3d757f0fc44495cbf2f7a2bfda2a4890b47586bc3cd427dcf1bb830a54e62bd4eaa6428937c0f4a5c437abf331a8f423a97849245607d90a7b9c4f830440d67e253abaca9702aedf03f28c2ae04ce4a3282d73a97a92928b388a5204a0a8da4c941850c9938ecfab56e2786f63e4debc8fcca9b0781ea3d562ffb6a8b7a66bba6713385f1b7913b22f45e9072ba928a3e4d8700c301e58dd5649de273cdbccd3761511d1c27d88e32482c1c5c919d62da91cb691746e731d285b7eadd9d523e69e08b48762af0d7f543a976f3b5b8b52464332f7ee600e38fceb7c679544927ae41597b935093bbd302ce064b8f3001588c161636b30b040c6175c45b9f6a4b8b6bb7ce85ae5d5e8f9b5b6d755fd3cf8a4254f1499b21dcca34dae4e0274f2e71d7bea24540fdf967eb7d23097e175a6d581fcf629b8dd6dcfc70535f2ae9e8dba9ed96b7cfa45419e869a9e8c84986e83bdb60ed30d596f288a4ee67fc079db25a50b1364c2feeeaa89bf32ad8349734c53619f7b050a3452e3f6cb835f4d72243a47faea8a0175dc246b62f0159687209e79f658fc1a6e87dbd8b17d9a3fa65fa1589fd0c7da88bbcb6311f83a881b3e0d74fda393f753e22c20b62bf428192d5cd5461632fcac105fe650f25d6b3d9c3fde296bf9d762f8cd82dc2e3ef1cd3284f9959c2cef4f8eacc145842700144b854f4b73ac47fa455a5050824cec4c42f57770e6897cc8dbe1063579d0f74bdd44c33edac72f0b507a05dd783ae9ddad94b9515cc148db187a5609004284ffb5d84c67bce21449dec23e71a6c3f27350a50d2d51becf9bf3567b65937077ec2d0e096a759587865a447c9ec45afcdd9952bb167c04159a1893867d3bf5d61770558c6d6065a3866bf8ba92f835f34e705e74c59675b8d59d29b1426921d264fa863396978a2c809d633937bffd942aaf521dda194576db916010cc3ef4d2b363890c620620e6fad727d1d34e95a46ba7e417fdce4abd58eeb263f0e6d8210cb4b193a41697970ec03f3fd8d0c12a94fd0ff8ebc2e33adfd1fe77783a6ebd38c0b8cbad1b56f6dd9b1a5498ac417da12d6e2c5cbf1dd11dec6b4d62bedccb36a083523f181574e9a0e9abd895e1e1aed690181dcd93da18bc7d778ef2e354b51270186c7c138f23a647c3f7d7db67dc4b1ffee170db10a603a9cdfc90a9dd1d0993a6c75dae96c04a11aafc98afe149dd57634c933ae2fed2809170c62784f9892fdc53dd522318750a0b4fc3801d3b2a68b00e82aeb08704fb9f2c42cea5c0ace6d4f2d4baee05f9c232883554dbb00e47b3608c74ad9038af399c463ae102f1ac7210d65d4c4d0882da5532d1ce86b7be4734619dcb1d5b88ef49de0764370337d243d30438a46cb2156399d915e7ece6754d55aaf412c53ebd2cc60276a03597e7eb773ff8014c98604e0f52668df30e7b88c8c063db2a8da8740c01f0cb4a2bbaa2d8b6af0c8209270a15941bdccf1498e10d6d8043e0ba2ca7bd31123aa1a48f1a54393fcb80de5c16930de81e466518b1e41190e6dcca73e35ca69769fec2802ce8b375a7b85623f6cfb226ffb42ef404ba0e6de45f0e945fe2f21f8fc7dee79d258fdfe0cc73e738c43c7c827d48e7a9bde766959941c86ea1649e693b0141f32763a74cb3b75c3356c0aa31d2f1a65dc95da80fb3e5ef735e2909e8b20b495a35d8b866d00e44102b836267bbbdc4e077e514caaca8c83cb6b41e2a4eef7356a839861d8e5b514d65536197de8ec2baca7e15107a3fe3e15b0bc021adb770a240aeb5c881f8570834eb73170631b4ffa1b7c6869d94ef580054fa9bb87fb14159fcad4a3d8c60abfaf4e474e33e8637aea25edb2c9bfc0ca140225444a6b16a7334fe05ecbaf4d499a6dd8f07e03d700e8b8a9426d34e79ef17d3459ca2c0001c7dbaca7c0a22befc2535019426c2412d7584d0517a69b8187f93561a89c301a7c0506c5009e3eb35299213d1df580ecb39d781bc77d85d8a04c8c0d9cd1a262cb0212be00c696f88ac2550cb00804e9c3e9a6abac799962ab426c7a7178b237a05f910b46a31f3a75c1a1d7633c3313edc478169ceb00f168a5208f61e88e8e5e3a13dad7c0bddca92477ac2fa71015aeca13b6b0c47a45cade9307cfc51b47c559141e9255b990e71f84f655d9db6c569efbc98d80f4cc7fa9d8b7fcce7095e98285ccdd1c82c387c88eccadd6b4df97804cb1a975b160815833d8b291776e3eb40c185fec51c1a8c93f665041106bae6a64eedb25a4a07e2fb16410f6d14283647a12f254733d9cc040b87d09f3b898adea01264de03f6b5458cfae0df13dfbbad3cbacc8653296dc6af7cfcf3d03892100903fd516fa84acbdd053117cc919d71b94addb603eb4b12d8285cddf1e31396f5b8532e475e9bdf4876ea2d4a5eaaa9e66c5eed977a50c469836bfcde571e03cf290c28c36469358a927ddc5c50df0aaa8d11b8cc4cfb193d681ab9fcfe79a58a1a61189c3932f23654258b49e4e848d3ce01b3c5382c741b018f18d8ae4fb4ab02822e0a273ea823cd5d3f3f46945349021936770dd1522ac26c8e181b3302277deb28cade4fc98b63c2f573e0320a3fb9e5494fc6d891ccafbd21603cb6eede9557c85a6787e155f8c7012352bff6b56e49256c757d7398189bf81954de92882145900b2dd2cd77e65d84a77e262ac9fce55157260010a7ab0f2efe6d93c3f9bb89752f93c8401ca1c4cc085b33cad2468724c997178e2ad5c4e3f99bddad1b4613710779842075de3e3ab6e9dbedb63f01a48889300d58885a06f40940f48f879f6fefedbe235d206abbce68bf0bcc413cfb67028db726b7dc5818b75d2c9e9e33db34e410c8416e293cdb25e4613c810272af5678cd930542dd45d08f6ab4d425c6cd2ab093248ddff796c12cf4a7506c7fb0972c501007bc84bb42b2f6026d64be98b30fd47bf84d60ab4d2d90c031c96ecea89dfa3421f34610b86f146e716de8f3bef40f04ddc886d4aa3574a3279404a9406bea943a2e42609d07e3bd4ed75330da5ec46d903eb1029ee4a3dab6232867790f31a2dd688b04353ed04a64a4dc64c386016f32e472c7cdd02794be4340e157d101f07115c9a800493183b7bb98047aa867658a6943b9035f961afe6e2874f44194d99d4bd10ff01b26e29e646e7d4665693304b170d6af6e171b0a87a51dcb6c120d2244db5c8f0a683a0b616bbd63f69aac79256c2b9ac852dea3117e175b8c43571e2489a25afa5ee96a1b32dd90339c7faa70077c079cd8faa65f258889f23b0d89ff04cd8905803bca8f87dfccef1f94334492aa068f5579b140f8cd45a4c5a8ba23b9b9ff1d62aef33dc151200461292e19c2b305c80860afeb7815f392d70699db9c4235804fb8b583265a47338d4fcc28034e7ec0846eb3823b47dc801d2e1e4b5ac4d157213e476a1d48b1b9794da0d386cf33137f2953b8206357068478917cc56b0d5c8bdde09a52f3c7fccb1e2d2407d06f2254663e85c6414252503f036a9f99dd2bdf61d7bde78443ed7457f799603d93c0be3db3e63eb6fbe1037db7623f52955a03157af79677d8dd586c5b4394368119209dbdc4d96a9b31a4f332cc351e1d48d3548ba20bb8c3a15c5ea7056b380793814525cfb6d0a9e91eb4373cb489307c7faf41147e299ed7e5011b44944df6efee3a84a03184cd77f29454b0524b97b1c7cbeee06218a268922539a8420b1593013624d862d8539994e2146dd8a027ea67769d998c3d9988bb1d50ce1896d8d0293a4e6862a4dceb0709da7336a2bda71f49afc3b1c2e73eb374b271419dd8f1c0ba08f8b3bea0a41b45d952b1271a07f3cdcb9350e974fc2bcf5a88044bc3f3668b0b830e924c6af12cce227f6eb31b7904f8fb0eb0820e22ede8de3b9c05068d598da1504ac6b09f91f8f54bbe761275719558de03db42493da52d5b7144e99e68b767a3e02540f93a01106d8d8408fd46238a5042d656b72a7532c080773d099a5b0da978e7a5b7cc658a209a3453f3e578fa2dc54d791d7f919d0452e64194dfc0ac629c93c541032d3e12fad4ead55d23518a423e98e0284bc5ec9ac6161a34bf376dc82dde1a5bf54a204c2138e8cc4ca9dd05bf3a64ceeb82d87ef7860b1c543a977484c151ca48ec77ce58e931eeedb9fcbc7b6cf5549301c0165c60e00a3d55456d862e1dd8e9774f10a3359f190817d41e532b508ec6e546a4d1f59494e257feb5ffd8b9c92e002d839b161a22a2ed4cc08d5269d967834b2ca524fbd1ecdb162d8034096cd37aeb421a6264503d60873d4cdbe1b3516025a93d79f1438a38963f25099a32322ff81e9a0b3f0a37e8d2223394eb04ee84111746e5b77cd48263a5f39a6056e97c0bda23801effebba98909036a4438dd1fec8d84fc388773c308b3f73e3d83b7ff241a2c1f8423574aa9b6fb0838ff5d4c3440acd45c5bd55bfb7d4b08766c9dc2a6f8919f897663f31df71b3036d3c4060810f7d6d1e7bbedf995992e69033ea0e49512808376ea3964cb7a383d37eb966078e1116304f7dc5763971ccd7f4ff878b7aa84ee1295fa7a7e3dcab437409a4c8796451f00c4cc1b32c605e789e971f6bb1b5e4cff4b32d484209e4f80d54472517317bb1ae14b7c222c43bd27a9ee70589c4078f2edf6fb79c8f73bce4067c1d0f42bc91b0fe98c2a1c085554afb2e91abe2f7f1a4a0b3efe4b3399a3ba2e595638c2095405a73a3a7a95c429cf1aee614b6329ab2791b877623cd911305789ddf21f20df6f6d9956494f7dafbe59093be7307d61bd857dfc4df56a78223709961b56b7a3f308c1f958c9352c49b4f63c63d302bb80f156f5bd6a79970cbe409ca480dbb478e3236f4a6089e16bc98445b7a294f0d1ebcd5a72f2de50a0bf9b358d4dd2522b8e594580ef5d8878798b6cc54f8f2ebec6cc514c40f7b3d1c58f32bf2610abf13616fb791b4358746857e4dac66bd4aa78fadfa26e5c1c72e5957fd7e7e4696b398e19dfa8aaab0b6cf9232d1ce5b977d1727342041312347d6c6ac6b54f20b31c489379f802b8806b5c30e558ede9ff571ea70d1efa04824fb388b72864fbfd1c7428f6f04d1a7ac46b645f515ad8d1e8152aa30a47fef49ac48ec80b27a3fde03041e85def3ec6d4fc1355e0aad7dedc74de8cf864e99cdf4dd5d273db65f48c45cb07704b7f850176731417435ce253c4a1397e219c8df73e5970a96e8d813f4160cbb58662a129ae881caad39af716557e7a462b49801d4d81516c43c2e6f1e2716bfa7551040dc8ff97db004e4faa17206939ce337849bc3c1a8776c5c89255701aefee3df2c2b1ad12b57f1faf880b453a310e27b649c1d6be8df9b8a2519f47ea7dc32396e929b4cc8bbac87bfc2e112c4878bfb20b187ab1c6321757cc4496e95982f39de326e779df97e0e2503f91436531c068b69a02146fa15d1e876cac0ef852ec37a988b8d91bd9367f87b3a48101f60ee9dc8200f3425e8748918fadc2c186bed2c1c3435a9706565006036c3ded664ca1cd1b771350c317c55915f849b4b422c1981348abf53f6971962cefd313f734dd311ecb77fe1b964d9d11939d9aefde20ff1cf79fea6008cb7618d1af222a7944395d5be802909e5429b4a7be96f5e112476bd41435b4143dd5b93d9ca5bab85d9211e58d01cfc01ea086470e148c536e48ec1c2ab4e8efe6dccf19b992d156b99e32689c53a90fefd10af9b35bed4572067fdc5a80de23395e58df003b7e764de2e063911ec50fe9f15e6bf84278bd1a39093ee891caef7a1e52ec8025301ce64d93f1509596a113018a8dc169fb351530f35b33139124385f9677770cd6a6f1e98064ce41d9da02487f8665d5b8b9e2d92c38316afd9c93a2575a00df498bc4996f340919fd30336fc2d81028af77992518137841d8abb691e1e511ba30c065cb5fef28769b9762f9daee6153d737dfdd0cfaed0293c48d19700288d895a02bdce38a70de3f5302e8eea82e984149c8f09a0bbd27a85f2a1d4ea83c2beda12a2c711fc61944d6805bfa8efab7a62aea4d161fe279c6bd8c262255ba1bf550be3efe083652375ea4a924996d8142e01e4c85abe5d55be3e0f2a3c59b393be9f6af84f79f9d16ce53fdd81e121db8b89ca6a3309e293300ee700f5227ea57b82f2e6fef01a22e4170f0d1cadcf944d2c59d8210c88d24b697c38296fe3ab0a369666c8b013bcdfa7536c85327e032b3308f9d9ddf7dc1fe0a31f4b154319839fccbbd1d19c12084f0f5aefbb3913f5ad89c6ca17877555a051e9e997bbdcbe7b91cc645f648d9c2abdc2535d488f2f34fafd3c3d3a6fc379cd149e3ae9759728a38487d87b63f5fc0769c4dd9832fafa647ec07cc0bd8ba747144fb4a63562589016c21128788d88686bd5b5428c64962cea870325b2b8f4a83dc424eb16df2b0f7bc887aaa711f145d24b60247429434e65330d26a7c59eb2aecaa7df835dbbee142759728a3a7f75d2d0683fa58a3dd7aef97f72148ddceaa7e857db278adf33ccc560463024fe5801db5525e29ce004f782c8aae874f3dea83b374380484a7f61d4b727f80e632fcd80bd1e1ca383b3a625545620c237e7c821c17723dc11a1ad76816b596effed1af377ed01270223bf21ef0c49df4582f410be91cb86b77a89b3acbc5a6d49baaf0387c9373c8bdbc737414882cafa13809bf52cb6e8c770de60e264f582f5b0bdce58dfb93a85cda75dff498cb2aeb5eb79f7dffced4172efb5d60f0a253f6c4e56c04826fa9d6164830bba1e1bd73fe2df2825027f2fa7b73a10a762d690a837be0e5b25e5bb2f48d650e4bc838c709a1bd98b832213b236985792e74878c2d49d6cc4e16dffe516ac6f71797180f39df22213337aff5fc378444febd4978926234ef2790fb88662764ade706d1dcf33e7cf0241fe8570de24dc5b5ca9ae9b86f63d199b0de47b8819fe38211abb9ad9b9c49b8e50c23c9d4809b658ce7c6ef3137d9b404b53399ebd72232ee0c06871c8ea06ca66ee5a4d62e9fa3f201ffbcff76d21f233ead52936c8e206b80f3b1ec13c9a419c2fcd44d6676c6d3489692a06b139fb6f4d1f79aba2ddee1a3000399d966aacefabb89e466dd26c8f0051224cb1f7a0806fdb1e56e0a7e33cc310a3be50843737a202490f714554abbd272db4d8b3b379fa1aea39b27dcd6e11e0b74695da8f3a95088f97d85ef1e3b747229c7f70119334f920ebe496d6f9a949f50a6dbfbfe44283c6f063e12b352bf340e3cc900484ec42a2b1ed7d3ca62d154c2f8dd554808be4612e01ceced538645109e9fc64187f4087983526e5cc9e116acbb3e347f18a5cdaf498e48494af00c65c2ff709804fdc984edaf19d2e836a1318a3084c8e6a0371c1aaf4572b9ff9bbe302c8ec5d165431346238a8e1f881cd90733c5a69bde820de96f2d300c8e4392e4472348c6d02b2f39bf0d1f601e5bed512a7c748bf700f73508afeeab87f04446063ad4cf7aa3094ab593bb137bc7375edfbbfce10ede55bafa4184139432c3475a2717f3ee01b1ae9100bd8cb63d4bab528069d60224fd55da0ee1cde7718bf95566938cfdf0c6593fa25b86f86b1713e956f1883b0bffb16193a1308d0093adc3086ebb102e3a2c7ac803e51377ce611f2d8a9918878fa352c17ccfb605c5e25029fe607f9cb9bf3743155a18c423d0a3f148ab0f55f08f06818a55bd11d00787b7bde002d7f65448c98e2f71493bb961268a5a80905a6e3659ed2d3f9ed4ff7d5d4e0b4be22c535bcb2548118ffa3d3f58942c8b3e55f2abdf5099bed454fbbc82c9359dd63ed5aa2aa41af04ba385113414d443197d4ea82a5898ee0a60cbcc05135852874e7dbbb85813503bd89d6034744496dbab72f232a03d7b1fbd892a325ea9921530cf5b65713d895202a568d8269f37252d1b62879d5fa521ec78c6ab00be8a12ef968e57c42f349ebb445bfccf7e1ec93b89e010dbfc1795f9e03db84063d86a8f0590dd96333a5dbf10bcfcdc0ba197dc1940b23fee91d6e242aa2f36d1410837fe9eebe092b7d0661cab72093a9f19521c9a0e760271dc254fdff448e97b21d0b3b8f344786da752a6af34ff873b3eaf522aa279d4818fcc9cd6b9c0906b04080c269778c96a472adb6bd5ab4913268c43fb282c86e34f0c9d58645a6db01d90dbcc9775f854505c3616e32587a4d646056b42505d147fc6ded3cebc3fca094ad0c6df86c9ad0c2f12a1a035571911cfb7c902f6c6c3e61e3089fb9ab0d2bbe201e60bc9dec87ee30bedcbd0f125a93e7bf7ba2c2e34b08c4b464ffda4d2e8ec1440962ca5b1ae15aca3c96f120b64baf910a43df8eddf257fc4d78852d2ac21d92c92bf48f6a7461484b5a8db31b230c122f11cdb017a39178608bdb1b7db9c0b2ca5d1387e2095c2cede1e122380b791cd4e5ff9673b7e676b0530167ecd0bb25b00aa88111dec7dd67b93ae16d721c4b1d5358becababa865ba693a0c018ed4acf6d26b7d1019cf9d94e8eb1ac77a6de58f524ec2e36b43cdb998757678a2bedd47820a197678e1361f898aa1cdd51567a796ced4b97c942232a47ab2763f995eb172e09f436ace8c494ed6dc04d0495db44303a72df53a9d45c9a1607a69cadf0b066e164ca6b370d0e6519a6520029c73388bac87322f405405e36742a19aeecb3d2a1a97389193dce80a272256324f24f4936eb58781597235b2e22a7d4217f8a5619b6ea956813ecf2b8d2a897a652127c76632ac85dcf3897dd82e8095feee9dff9e11d26efcccdeea2c81e16d418fdadb793457a2871e6caa36fec9679eb4796543261c05301e2cba34a7d9a903de006344131320c5c1ba9b0d124a95cffa00caaa3d971e805c4d77a4b55d3f17c235edb4300aa45a34180f6dd312544a6cb75b4f66ce437cb637327514e6862f326fafebca13660fe39177e829a33430c659ba7dfede1db9129fa5f55cc524045f85eb57f41418d3054d36def7edaf20b308c8e5b5b7dc50354cd91a65ac53e5d574b904c8e62411aaaa6915f4df857bd7af0609e94ae800932e2fcd9727752ed855812f5a5d4f3b3730c1a2a907aef9ec4e456c8be5807c57b7890e567bdf81251347a4b47659958e04d590d37be16f08e94b5e4879973e5e8ff4f1c2431e6f2c4f3943436162aa2dc927c67e39cfcf6f19c25f29544efb29f81d48cf0944a8a10f2f1dab6616c78678a4fcd7c815f2781e4f5d3ea324451d92b8989821386f353a7b6a13446c524e95e62199fb3f8764d4e09c212d585cf1eea9cc1acf21b66c1000d09bea848174195d7b354d0e71bc36a720650c0c85f56ee0e7d80e9ddf9d20a1088140c3fd14d3763553c31c19ba76ff1c96b0f507710d1503f41af02c0d025fd09e0923d3569eeec83e3de22aa4a064071e958feb62c553ea544a44d39ff9aa0cdc6191bebce388b82bedb63c59af0286b20a0421effcbf97bee0c4a7cab8d7b33d64aa45c7dea01480f4e313621f6dff3b87714775f27ad69b2ba75b75dbe242a19194c4603079685c7dd36dc73c80f64b9e9db94ae9914dc877c638f7a10a71b4baca23dc413d02f74e2c771600164246c49b51d026e9018753fe01075cb45cf9576906e794e13529fa03373081f94a7a9f91a4d2ba44ee41eb12e2189417c794190acaf20a7c12e0e7db64492bfd5628765d1ad5c04eca28adfd6aaa98b911924c14331d9f15bc3c014555d4ce36d32b55999e8389ffe4ab1c982aee85ebc9da6a4f1f7791df3e2061c4995e8ae6ceaafe1026cadb9641240ebf752b6840f034661012c7ce66303e7601367492bfc0e9e17c3a8d60759e8daf46959024234057bb3731a79e6d5952a40511e413218aaa91a3a1a5dda014c4b755127821f090ece4fc7f98cb95269ed06d8804713b04a2c4bf777f69bfe27894615f4d20d9cb777b68b6bc4e4a4560814cbb1c2372425607eaa8de244f0098b5d8250acc35aae895fe7286154848c4c644d5b4e8846e369908fa7f8d705ee4c56f25516b1c78b3933c5a4fcaa014964144b964a8a774a9eaa22b8fd06f5e8432ad0e42e2155a188bf95c79719810b547e5cfddc1600c5297526ff1cd0218b7f96aa78b7a4a6a2a5a7367187fe01296eb04a49e19077a48d58bdecbf640690d841f53ac3caff220000038ad86b5455b3709f62ba04f75240585f5fe77b32c483402cb64e236f8e5d5ecf5d3d13df3ada2a3b62b7a64808abb2ddab7d0598a6c0ec04485b33a93a1ed3c81fbb2b2613ed23ceba6f96a7601f25bc56c98dc5c66d240914939665eabbc4bc0d408949d0dc0adcbec358c060a6c5d6e89a688dc621b866364cf4f345553d59eeb37e70b25070c546ae0d20e5900aacd2bb7c9d3ba47f6c18fbb3a52d3c48dc8dd315a87008d8ff37e11d31e78f6f4f322cb4063c0ac7dcafdfa42900b036302fbfe4632f32f59dc68adc223a7b8e9f19255885d4c91c3c4ec159a00654131f5ec761a429abccaae7328b23497ce09f51e25a8ad01bf30a22c0617d6be77d5ced6ccf77f8661c4a8be88a8ade655fc3d92507f27a2f57ef7e12ddc85f9e9b3396c9241e59153084678c5a1b650052a9bc2c300ff718e7ecfd1557b427d6984715fa7c7a88ffddea68bdd19536b0616befce66472c428a30f740c3227bc4906f38f8a7c7fb09eb8f23db281878e7122f93ef57a7f5047509263f67b7bea4ac1eb8e75b8b7e182e3b90f0d3e1fe0de9fec12a384934dac32098e283ede052893893e7f37abe8a479068eaa28c762611250b79c93af3935d897740a273ee08a7f12163ac0d0976437419ba07f6281f5485e3e0f407ca5eb4c3d821ba6557f2d30560e22216ab6cd6b03f48691429d5d2d3f2ce40f18bd8d9421996001729f50375800f2ee59962fe88a72c556cd820598fa045e912f1721b2b1eb52dfa0badb9d553c75d6906b368336de70ec9229e8a1f1da41e46bb4e8de48707b1a96731a74914450bed2d40afb3eab9134552683496290b81c76d7694c6c41d8c7ca7fcda62027dabc3f4a5a23eeb6681b21f36c9fa9de40b8a6e28625624bd96679d1e816e22b5fa1ceba20bb46f63ed1f99b0da6ef881bf8d0c1aebf0490ac3faf97c641f49dfe4423ae05a1cc7d27f4bd952343447768f605f530bcd602fc2d7902209809dbc84335e7f32c0d8a019a8ab5d96acb0a2ef01e28e36693ac7a1b6a99b38887b5779be8971bff89d9b6fbba60f420dbb03b0a0a982cd5cd6496433080ece719c234fc8169633be300bfb84865acd042f960da936d540b60d4c76003dcf899b7f46bc86c2525b7f2691e1bfd14ccbbd8ed4815e4e9ec21a5910edbe023586cdd695bbd36ccfd6aa0e3df90421612b8527fb4ed084eb3cd1f907e4240fb4cb0b0a446d7b100c9a2af8fa64bae1d60eb6a4ccdb82aef4ef503de652bf88fd6ccb76784540f5b86b3e35670ab8e0c0e4bb24b782022af2a8598d24ab66c8f589ef7d9a1a16fbba70dbf23c4968c1fcf554cb2398370e2d52e7689555b92029f3bd6f7ed14033d7c5fe74729f9ad7e491d983e6779b0380aea650d2c364d4804dc49ad0cf8192a5e01f1d2c1c7a4c13eae942e41ca68c573d7ecee18339fbbd67a5ae479342eba07ddaee04937c4d539438e305950bf0d7fd151ce2fe0a05a76e81aa5acabb8f592b120387e36bf29b01582d1e85b65baa36d875c9ba404b74da30cbbeedb68148ff3657168dbd8a5d0030a487e85b089beaa5caf5c6eb5007c626f33d80097c0653f05f93a7c405834f268779337c3448c41d7e7a81acae2822f6c96c324067762099141a138eb664c31a4c1a0cc1135115783b48b8533bdd256abdbeb4500defcc47f92ff2eb71792887b651d2ea60e2660b7a380dc7949dd984fee3750c16e8e2b417b73adaec6bb43ce133c7a79982a35953234754c412cca46cda76b6bee382f9ab5446cfbf26dda3a06b30e0ae6cb607b1575ba2cd2ef7a010865cd2f8271be552dd65fc37f610a26208dc3697bd93f2e0ede55d819e8732325338f19891b978f70acafe1a1119e6a82d60488cfc38bba745c2e33eed70718a6ac11136d4b87c5e3823c41a34edc89a4ca6bfdb92a4f80474295b3c7c7cb6aefb3bc39133fd40d1d27ab069306b2b5defacd1c4932fe2dd0b928b39a4006759f0caa10ca0bf4179ffc3d48e3a0d99b5d733b7afa283b2e3a6c57982096de43fc4c6133490351876c76bea551f41d23f94644287d8b332e1a9705b686e614858c68a329b3fb20d07a1ae53d1f3e42da877f6e1d218127dc64c817c10c3267e6ffa896ef6001d3f5f46be0d315d1314e4ed2d868430b8fe0c679faeb4fd4f08e01364ed2bec1ec4ab97ea615112ba8e66500855c7d7af5816dfd1db24c636c3631c6ac8c7a6b554ca3ca487135e6c498a4f8c6160788c94fba7ac3cbf21a7827aa89613e140140e21176d49bd76189dc21ea689f999468e5e457f3e1c2b4837e114a311759e08a83416a39502ff926d2cd9f0aa71d36adff17ed57c8cf50cc0394f46ccf4c8052db8ded827aea844ec789ba6cd03c98f7b793c8f74f10894efb62276d2e6c8400cb6c4fc62672b93a9d481e8eb94a816a4f54aa8883e3b0ec8c6bf4dbd5ada94b3001a868627265a11923d95aa1571e545d79ba9881cf05131d6ca23aa382bb2a88e74c5d8fe211b99048a8fa25d63a996185884b45f0d384fd109d4a0ee6a369c4cff4c89deeda02b22a592cc25362789422bb0efee20b6233771b296d0d6975dd014be13a47fcd8cf8e4e3e53f8e3f487664981cca222a7182c113205636d315157cab315f4a98fc0e7cfb20bb0bd7bd3603bbd1ea19944e6e1d54132cc4859b52b5209448414403d2bd533a73a95498363b89d2fb81bafdd03f40776025c7c970487077922ab04ba07130fa2da62922c5ec4297ce2e2101fef05ff61de9c08a1da3bc0a053567a3f9f96de57521d9a9f609bc618bd10156e74cc56a37c615d037738610c0ebb029962bdf1ade17caba9bdb3676d09bde246ade7c2586599a32ccffa6773756401401c849fa4890dcbf022ed5da466c2146bc858450dbe8a40375f9278d3eeb7bef5f8d07d7fe14a571f8427d35d0f08a23698a434cf620e6ea6303731e13b2a72c695440acde36dfbb04abffa8f6b6bfdbc194f5e693fca461dbc8740764eec3f12ef0c10257edaf5cfdd21ca3b889f765eac0f1b2e2c4e702de66ff500f5ac0be23ff9b419fb7af6156008832507fd539c76e03304367aab41974ae2d1f0f82a5b351cfe6ff5c872be1f95451caed3cf0cf67360a47be9870639e1ad16670b03d052d227677d4ff05e0e3544f810db1b7cc0e6deb9e242e6e309073cdc985aada6698898f0eafda3e75a031341db81516e64d0fd4a57a432d84eeb3c826e1934e64bccd96d5e0ec32ef6452449e5e7712632f792b57ece00e8f8759cb9c53da7654fa3609bab5267e54e00ad93753eb8a7ad0309da2c9dd37b9a0e3f8fe943625e3209f1145b1504b5550d2827512d90fb8289d4bb19ef12cd8b9fef137d6544e43174d8c8bb0d707309e31810229efaf45400740821db199a799a2d44e67354231dd73cf5f23a87294716f690651b97eaadfbc074632cd91b68893d9ab7dff47a996884cb53742a085adf08df01bcbde515dc14a82ec054d70bf94e5949f3b7022ba555855f7389b90c64087e55ce29024589d8b728504ab6ac02849bf7b258ca12b40468607ab7346e934d1ad810ee831c9dcec250467f5ba940c8101ef330fc75d35674461cecb18c019164d15d9dd22bfff54adffa96738491d2d155f5e88abd89ea72731160ff1c46615781253991c89154901d47c80a8dcb1da2260a03b8df0e1c6d56f862fb2b211b124cc83b1fb145b5de834ea59ac5febc832910e6c8eea628a188dee2962069e712f94d7ea47cadcd3dba07666ed236669c114523e2857549ec67e1453adcdeed62b833e760adf1ba3b575703aac0cc77e3a2c4066af2c9a93172edb99d448e7a669a8301bbb5eb4e03784765c8b1b3bfa2df522475b702dce803886e86f2a5d15a80c7ada802eb4b2b176802f0881502a44bf749d22c7e38bc320e5e62495fc17a8ce3df90277c34b4b837d2855263fa0b4f434f5dc84c1982a612143a5aa118122b8dd2d461c896bcce7f516df5ee2e4a4fadb578d9b5e69af0fa47b3e8db1b9254d51d57a88f5b7ac39f688aef91fdb7c167651e88fa63d517276097d5d6c082745a44e1ac48076c089bbf96df7e904c8e09c8cdbe6a40467bc656eae3a6b66392b3c177f283f381de4faa23181deb503313a2b72b9a25ceed4b0f072d7e1a7670ecbf80372a9c261f14a733b38df38c2cc52eaced185d3b37476d4be81c776a1e638aa864df743bb841eac31617f01b3928fc2d00627d7629f8d9297b6a6ab43cdb6444cab0b94a838bf2503fac87fa00fb4880bce82f41f44444c0e9b1e5c6593f265489594291e2fdef32675531ec6aa50aba8a0b87b2b9c6b5b2435094601a36883bd4868ffa1d6f7547f4f519292eea6e40d310604bf49c8c2011904f9c7e435e98def319b53e5faff19c017942d1849fd70f24b46e3dd0ddaeda302ab1990b1e644556c50106c1a13dfd8502df8ab51b39a82224fa96acc6fa5efdd8a84c933c1fb6195b837b42673efe8f653e2d21168388a49e2179e6329d51fd40697f9c19f21207dec5bf191789042443fca2be9dfab67a4a1be73dc2a8d76f4216ac948c10bf2c464723e4d7f70e9cd61b7737e2fadae7f439ff2077c3a3756455a04918dafa920f1071e00e4b51f11fed96faeed0ded598b79213e430fca81ec713d3dffb08d29d45b49d1bdd5f8b9df65f47cb1ae9cf54047774367f9f1d93ef69ed6f6cdc8f29e8d50da3ddbe369d16bd56665be5094ff6bcd0ed38d4c87b3f57c02a8a33ea0d5221117a7c6b3e9451d6c3097c4001dea5b7f55c31bb758cd75f0963bf795801f7d793aed76819b3a282a0a89b15fa2469b0642338f1e5e8c91f3b58449c2097015513e263a822e1ddd73f215b55245db749f9772dfae81c9e7c6e6e4b1392fbb813c8c642216be45ee42705198ebe996d1cb3f2a8b43469344440222ab45141db53a93badab7052ddf3e111a2b67c991c7f9d61ac73703a1cfbb497500022211e5f0a01dbabb7fa7c2f9392602d4832041be99d07b37c687110177f38c6989f893447f8ba90fbc83c1b0781f2b9eca6f3f5574ad4235bd3762de061c5c1e87db1bb0e3d023a352751bdcebc79a03396dee556971fc2fdbcd86e8480474d4e0b7149d492c8ca6092c64e2f6c56c9c70c25532ac50d8037325ed5aa57e98f29f9ef2f706cedf30ac1ed066dabb1e38c19a0faa1860d8a496db277a7922fbaaaa2ce0c6a62f3348757da53f3ab95370278075f61e3819ab5eb17c96caafc4eb5fdd087fc0d9c3233ca2fad2cdc5cad494c1c0302c73a46efd8f8ddab60f0a65cbd02b509964019e107f23dc0812bea0d2b23c672aad0e3bbcc2f8a049ce8f8e702f07f5301bd91ebd7f1ca128df776232d1a7c18a860225ce6ebb212b7b6b36777ad7f4eef2733e127018b7f7a7adbda8c4ddc653bf496149763c0d69d574a8c47b4e36f08bf4d6313af4ebee9b8857f1efac70c6fe0d050a89e4850ccd102faddca0468484ed1dedd8387b6ef8400a96fcaa7ad1db538baaf61bfb993119a104867d17cc5049bfdbbbf7a5561a5b0d14307b1be07b8075e87e6a5815ed7f3a55db94ed82f0228c078a4ab9053f594ac989be160057eec78f3697a193863a33de166d5cc8418d08db9ef1bb4264c5a04b3159c3f71ad3718ac4f50300438a91875b556e2f31375b5a743d9b79d7e33d52f66d9a176631f1c92b84653646e17e9150c307ceccd67e72d3bd22ec8d1542cff0a775712f537331e5dd49f4d80a4fb6db1cb6977942e12def21a544f1b0e394b93c19b90393b3f30b930e7195dba5d8ab9066a699f78ccd18fa88b4e0cf9b605f1a3ae9b3717ddee812880f8f79f406f671df0c4236e2a070e86c407e3c858e02ecf1de32c3f6313c8da5403fe6c78ebeb4eeb8d2036314543164fb546e041e1122d75c6111c0c6e4061aff40bb0c440acd25d716e16467b8ef2c61eb5b75143fd7fb79fa1f8e57bbc3f762f7c8da4377aecc351dc8d33c1b416590bb1dc456557b30f9ab629dd0b854638af10919646ca7c72014bc655ea909b00627030db7ca6bd1037aa40b55da6a667ddd4b84d27cf29264999385570dce53c4bbd8a2c11dd9dd3684ff5581aea0de1d7bdff0235e143f7111e2c26b875509bd84623c93fa16173c1a6e358683e12ba876b0735034f4136a21bcccfbbacd56691253f84c3cd95330a7a5db060469319e6f188a4d94b03bffecba14a2a3b3215a47d0458dda3c66ce92d9057b06eaca7eb181e1e81f85a1f0bc286c9fc0827818b6fee4229e084c1d2a9e0472523ca7f53706b25890697ce8e4671a649e3e891432f50abecff9ceec2fc7b53706414b6633c89820a44afb0c7c8dd9cba807232e8760d495ea9f496fb1290b72d649858a55dec12e26b33013f45d124b322c85fb848f5a1533685be117c5e3916fee70c4a1c306577434a7c2fee6e35bf9681870a8fa6747167c605052c210aa78f87d0d1135028935aa0fc900fb5c121f17b0b2801b14153f3e554b8cd61d9dbd21f132297035ee0868b48297e54ea26565ac2957d703eb04fca40417f6874f47c1baa7a4a3f7778c2b0e324190b32a6f15be90e6c276657960118dce0fe5ffb76a37b52457668d90f802ae21f66cf1143d00fda9e6a62fbb4295a0c177e6bfdea46eb8d48a996b33bde93197d3c6840b0377e69b73250daa3923a83d6176c62fc47e11bc820047fcf8fe7aaf82383d7096ef3187578e96ef7f13e364858a224c26b5382a4c6bcd0c4a7697bca6fa97282b8871c92ce0344e845c5deb9a73397e6f8549599bc587eec43fcbf7c42efe67f7f0d33792514c111c89602c9f2db399ba4c7f0e15b6292912705254a53956c62ee65f7e799fe591d86a1526c5d85d1e7aea59c2170d31f6864e83a67fe6fbce38e1c35e04b8c679880caafe586f0ca99b4b53299f7439922ecf809a735e01cbab1f7777cf92b2b37c5c7b3bd5913def83e25254844331fcc86c62a0db5d28322cb3b02415115056b3c0af05d9f4592688b70b1fa088f2f92954f8a7da00769e0831e07eb4400d91e959be9187fb1574902ceb56579cbfa614bfe99979dcc5242a0bfd165ebc00d14e2282123035173b5d0c16058e9163368cae371f7c2592ab58c4960492586ae638d3a1b1d20492c1ec1fdf2b227c7cfaccd5134589822ca09db151fcd8ceedb7347d2afe54463c6acff14572ef1305c8445e5a28ec5fb2d365097edfd0ecedba2f45546cb6d1327c2493531b9312a2eaf7c3f1b611ffa13213c34eb5e96c22749c8db110944415858c47a1acb34f376ce58bb04065372f8b2ffaa182ba72b41ab5ab197c56763d7e7c163da93cd2896ed8bb6a662525b5843f5d255613f9b4c69bdc4cc125e9820cb2fe9a22df38ff45170c2d0aee0479473375b84ce66c585b7565d36b4a60c91f929c06bc927e30479a3126a9570400cdee3fba931d73f7ad14fb1de20b4b58b12802b1eb19db1b44f43013b3a9d7fcae4e6fcab5bf5869a77aba5413ea0b622a83aaefc34ce324a62aa2daae3dbbc33810af9c074eafda2c619f4240e973748093c7bccc017c6c9b5c77c6efe385be347059feb2e6646c8a47f60abb10802e222991635908435e1c50ff030a5415102c206f6ab262168f55447e6caef3aeb673322f1664a48d38550261a1fb92df30557d15e93c7fc13f907984e346fb0fe8c5f0ad062c780062aae131031075538af0b2da679fe56eda19c3c26860a97c841e74e5cc01a54781b1d55dbbedc502a91f42a1ab25d7b8e9108dfd09e1ccdd2a153de3b4d9c3973837fe3109f0d04a52fc9e40e37b740caa53991b04c477f5d207a535a90f5ceca02f957d50e5b0ac4a6fa6fc73fd8ad291943a678c8d6a66ace7333d1457acfaee00f6314ef968d117edb3c900b7eefbb31a32fdd167758855dd33a6494834f170973711624325421e58d464866cd88e06eb47b77a388b01324d85aa9933589f3d503d14bb28c36c905b93ea01f3eb205e9079e75d43a84b01a7ddf4da668a62e8789394b412a07a9990a6bca8fc2e259cdc228b091a6caa7a11f2cf95f84c371e2c8f499279d3f93ae070ac6b752988d4055cdea1c682f902e76d2547b82c4704aed9eb78a69258249960321d124058e71005f936459327b2ac38c3f845e45e57324d876029538d198aa90b671c85e3ca9646669b5a35f9b65e77e1d0a8db2e4af883004cf054e790532a10696882b596cc6680db5ccada4964e07a981a2e81adee13dcc95176372eeb42afca082dc987f72868f96e668c76d5ba022b53075bf9782e96cce029d7c434a2f6eb22e9a20ecefe8be0f313237368c5df04dffc74f2fbf4d5b646275bd3c11b1e75f5af0e4af6c75550d71aceda8c897fd963fce489c50155fca1ff9316d6a9ecf879c0a8c6357eaa6edafc4b19dd9e0557cb8fd4b1f7dc8f73795a5118b0ef09144c273b7aa7c046ef580fc02e01bd7513a49c98bded745f8f8f0e280d289d07f67030cba3d29eed4aebac4a0209ed7846d24d8e1867e4dc9dc154bc6cde01097b0abc79f7956c094a413787e59293b1b08d9afada416fe132de4e8fff1b390c601ad4702267560de4b46071617ef2302653b8ecf924af19296af582c48fd39dd18fd949209a8b66af5d0454719eb7fd9925e8052a58f5997b1b4e7682248aff777e149953d360fa915c4752c018b6ea206a436586f7db65c06474a1dd4318819858f92679e245d78d9d3267f1356fe59749a453a23031971e7d54306cdd56115111f774d3e6dc28718eb4fc67242219890032da2a353088f78d0e554f60140d3eff7b0ea449b2e8a00bf08ce44dd1ad8f7bbc910b4b63fbd961ad7f07244e1fa75b82dffc757d99f226e459b0ece32ad72e8117725109dabcf5a2832aa0376b2ad82241a2e2c71d24d9016bad4a7a11b4562ca5de22addeddbffe25f60b27748f107048f4c80c631a1c7f74d93143e8bcafa43ab67b81ab270f9e2f751db7d981d7165c0567ba209129a14e14c33be786f1e408db3576857882d190c4ac103b23aee54d01ff8035d09233dbdab6ef74a8ca6b18001cde12695dc373885d28a7ea081cc355a30f505a5035535e034877f9f9e5e8ffe46b589bc292ba9c624a2018669d30816d7134f14a14a9cbb52e8e28b4bc4141d783a7a9db28cb2f519d36991deffbade2d28f330895d8ba366e8febb03f4656362e857fc66336ac2e52be64e098987e7cb9e618241c3e2a15f0a238ef6e17d0a6543daa21941741e81df1a6b9d26783b5b8a6066844b3e552873b1d3e26b8dd6d8a34f70368b2a1496122c372860985c85d3bde89fcc35a31bc4bff9b839fcbfa5ca05cd9585ec3fb69cec3d3b51d001f59a9409b313252369fe2421ec80b08894e2b860fca062ade13e608cf72acbaa74f2860d6b7311c9891ff51f82099f7388665df1270426f37f5ebd39809bb2611f776ba6f8bdf48ec909fb67372392b38be63c64778e9a1a9f8cfd6d6ab320afaaf8852c67aad2218e6e6031c331e982458ffa6272f739ed8eea4ff64bb64cece827498e28e63bae98d4c29b2fa625c63b0b06215e7ad3ee6f0b088fccc5da125df6fe1c0edaab6554440b5694d0b8ff5f274ccdfe0839afe53ae2e1b011efcc380a6711d41cc0e9fb97e3d5708b17b74792b12cb0f2bf7dc4250434986185b3b8aae9cd10c903a06fc05eaf02d50268dfe2c49e633bd63e10a8971a25900536efb8bb2215b7cc442338118c066d07e9fb763f3a4c4dfc806fd9fc1debd1e2820b79269b16e0fee144e0ba42e7ff9dcc11754c2a50782f3ab80d4a611a6db7ed0fb90b0c74c50724295a3de80677717ee059ee7adf9ff83918fa4870f77443c2079695f7bf0d3acec77e830b4e16b3c7c5e35ededca641e54ddd62f1ca9ecaebe6ea326471627baacc47ed7bff8ec56507ec4bbbb0e599fc27351477babbc7ea82a6dd25393619b2b45e872a57b2e6a79ca2bf9a794ac1a104541e87e9e2f55bbec3f489f9765f9f11ccbafa1edae468b5a507ff9f2a4462fe30c44c1551cdbd92e0f3b9acdf90b5e6815771b4ee67c2223d510f1177222b1100309b33ed05f0ecaa858add62e4fac0013031e5c2e435319ee4e09a425ebc597b34815ff43555149d71c3aa0171898d6b3a3263c96e698aeb44d10b35b1bacf1b929877bc5aa2290b30e86a0d739b2f926daefcaad33e1efd125ced679617d099220f23f98c333cd2f57c42ff592c77876b1320e36722c30e55a850763ffe9a454aef0fe66cdb476964af1003601045e2ebee21008817235a7b9b806ecccbe7dba8c3efa3339ca7e9c9c4f0e9d1c0c495c4068ad165234c04f10ec4a818a8c4ebcbcb7c793a8fa507bdbe1dd93e0bbd5162cd82f8e47e521e62d79055b27bc74b48b16d34f4655491f35af6b952b2f7edce209e90f6cce2ae1099c29b2e4169fa8a29e0760d63b1006c21f47fe080f593ab1adf43a36e5e8841d756ee10948a5f62370f02ff43c6f287c26e33aa1461fb88651fb8a5cd32eb296da1f4def54410816a221c8a8ab9a6b94ed3dc77e5c3bf25905984c9b60902f4a2a179d551f144631836061736ed32dca61844f29060a2cfd248531cfa52e47af73e255cf7b94fbfed7a3b514f3e018042c1be3b6a4852f486e5924a6606855e68420014b00d2c95df93653e047306e581072205dcdf009488b888b39f1fb389637e602e5802f4224afb05f5de6c04d4bfe1034cdf4cfeeb0217702698fe5f5be3301dc8f4f8dc7ce73db8c774ba7882c8cdd3192fcd160c0af7c636d29b347166bbc297ea5be9d1e0f2d4faeb15e67033dddf408f78252cff2c40fdd962a737e5b1c7bf092b85ecc5957867f2b1328b37472922cf01517e3ced3a948a7cf38d1ff6507920dcd1574b5e886742eae04197bc0c0bf7b5c5a0c8b541fe47f817538a433cafef68f5a0809f5a343849c96fa3606e1e9e6eb4298c9716598b2666fd62a8a2e02cffd91c5d638407cc5424558ce99bc5180426bd608c09e62aea8035e2c363ea46bdbd82fa973e01535e36684e3ef30389e48fb3f009aa319a6098e027fac7ad3740fd63df4dfdd2d3608ffca2979fac8a55f1126f5469fce860f791e976e07df08cdcc714c3d8a2ab71ad0064717ac3a964b21bdce846519d4479d670a0862fd4814739a77ad52b34e03021504266d87db87ff1dd182526da83ba230dfc525fec956820031d8e02edcabdf649418c603cb919f79fd98638eb19c8dcea8f2b44a99d85aef7909cce74305d6389dfe0fef63720e9f0b6f4b9d0a004d384bc55cf193bb2be1d962969862614a7d2abd95b1d8ec554639d1f3097ad110b8e7bdf5c8d3a2853f3ef1154bc2aeeed985bd3d2f55c23e2ded7486c1f2e940599ae702b59d39dd9e48d772205b0e0d4ee4036fac4c40108ea5c28d5b9f20ba4cf785295ec025a92254168c683bfcc02bd44fa6af42b4ea8827b79e094c7cfaa6fad6db105dc3e9e796d5de0a49f05c489a776cdd4cbad3984b4d558d45e73e6b7e973ae1b5034787830c55145c87a1bd210c06b4a75e13746f2a5dba4a44c88c0b6878af7a1fbfbccece0719ca5e7e2629084298de508193e2afd57a2285f3ad81d7e7ea1ffbcbea55aacfc994abf8725e574c82172d1951a73d8f4d886b64a6bc282ed9bc71a16d4093eaf38c68082072b44d142256c120c02586074f9eb191cf2f66dbaa76833a53b97e5ae9bddb331bb1a54a5b39a901d5f16ecd4e02a50a7fa78c4862c56f4245bb4a189bb97407c6d530b0ef3be960e7ffaac69af8e3bda1dbb35ebcefbfe45d20219c00943e073cfc8f6dee18384ab190d138187a7f0cf4bcfa41a8515e8926e7c07c85e9de57f8604fde408b9441851fd39c2ad16a3e7afd47cf06fbcbb8b1cd41730dafa9236094791a00c461d1df370498e7de79fda42763d45c941a1fba925ad583bcc509a638ff97661646ba9b05c64d3bea66db0e39d8efab6d062e583f26dc82731cd973cdf2faf34bed9f161cc5a21aa6626f23a8cf633f00f7371e67df65b3137ff9faa427c01190bfd0f5bb430b4421e069b4ccba9eed6f3c084b73c548478fb24c4dfed32ec50810169500de01e49f258a75d7ec89579a5bd7add11df849f50f04cd93c6d6270a9e6600071ccd7f49a6a9a26a704959e2c2f54a505b7dcbcc1a46d14839ff7088abba3650c8b8e94b68c11c9ab9c29c6b3c0088f8d09669aa16222017b5c6fb67b382233aee0b7ed8eca4f6f93b56814c6a8091b6eb2c507322f6bd5bc95d818c495c285ef485f7d643fcc6ceb1162a448576551236bf1b826825cca84195f5206dfee1ca6a9c070d24bd4f6534042309029aa6bcdff8e663270cb474e7f79da6bb55a5be4733e202052625d7d8cd47d573823b7c2d3cd82d0999cc54d1e3bd72f6bee6e94067b28c02aa5ec18b50ee5bcd68895832ae1f991cde7b9bccb37cc2551ef9c18af6f0dd1a0679ad07f6408af3cf33cfe07a210747b90714b8a84294c486bbe1b07492f1e499b8d7071aef05c93286a5d0633ed65a6e3c18274dbab1e280023800ecb8515baa7531367d3658e76c91817b51b121d6289e4a34789bfb8a79aa991a6e02a5ec667f3c0f4fe551afa88e22bc6ccdc778dcf4a79be04abc101dc6d672a48e4380d44dac7758d29ea8eb7ec3c5077219035667a694ef47cd0921bc65def419e9407877351e285f31de4a12e7ee58162d4bebee81f4c553b32c3b5bdd5024055783021045190ae5cec3ed4f60e22941027130334815ffcf0f74c94f27255419da87e3aa659dc0c4951613541a90d2c93121059a096c3e6ef2ee55ac736e136b6a5399963a544f30bec5009208abc588554e79260768187f12a2487f8308e221fdf42281ed9638f4babfa89a2cd0432bd41bd4462aa35c8afc0a1dabbd163caa15b339d552f017ef21e9006eb73726a6032b115791ceabd719967627444cbc23d32e081d07e2268ce77b25a2ea62e513fe667e973b81307d7a99230873bbfb27de9bf29173ac5952e230e87d7b77a2d05e7ae0c014d81d7d420fa4c074a5de625736cbd0c7d3b232614166f53dad156ca4c65bf4e4b774a0dd812e58667eb78b3a58514380e3a308795dea8b56398e83eeb37a10c0a6ed631baa6c1b10c4324f6f1e9a7c04380ce8853befc7747f2c4f9da2402c30820fa2aebbfbd94f14441dae2a3349f85e13912751320c5ed6bbed2e6c6b16b96d2bc498fc4a16af3cc22f0db352d4745f02c789a9346e14e7ce5e927051e671a696eea2cfdbdc7c2575f828cd23b60e9ac4a27ff812db642c28eac9d273d5bf7b1a13cbafe22a7fadc87978977c89cc201035c1dcadaf98f4edb9843528284ce17b6548bd12df614e4b4970d3d60a6866a34446c72aeef39c9483a4a44bcafc4ccf59fc336290a94bcfe6d8116022b296441336a79c5172ac6fc7565f439173d99a201caecdfaab2e3cebaeaaf05f2d9d7a540ba2032373881b0be66fe91ab340a0c6f221b428d3afd31e82bfb6ab920a2b3278ca72522e789ae44bffe55981043a96bd3ae908eaa8ae61497653b8a79f9dfe472a7decd3dff7e3539c003bfb416c4a00e59c9dc9175fbc51acef13df0dc7f4b36c13e69777267632af829d2b07d2afaf38ba429819e00c4b939e68e40f10ecde0bfa87aab2cc6e32d30f9b4359bf8f47bbf1eccbe1a4d744478efbbe98860f9654fed9b608f9ac2393f233a38221ef9d8f7b3ed6628d86e1ba6fa56773f328fa9b6df20ddd146031d4928868f38af4774e1ac89ef7ff03bf2fa3246ebf2c8fd5d2427cef151f3bd5dd271b8ef03ea41b892601dfd60e91fcbdfcc377dcd459ebeef3e05d74b24ecf2b1b5f7c5ebd56eaada90e556ba177e29e0e683cb2254fb8432b3304fa9bbca1a14bd17b8559b6cca6de0a9e3243e666b00d64be0ada1887f31d2181b3dd9da134f70f79288a7361ac88caa11aaf8c0bce9c577dd80f2d5edc46ae964341baa5a83f5c9d59a1d3f5ac727717e0bd0ef55216f0a886d9b00831b443b751babed8ba6756f222864af208ca739903a45fd71f178c6f240bc77abbc4d81ba815b43fcc7d3fbc255f735f4b8219e9a463b55788e4dc11e5d656b1a8994dbf4be0ce71cc0618023ed4569c65c4ac0420803685125031969827aabfc234126be5f0a68636308c91e2b517cd444b71a3d3a3cf2d409d9ff6ce07571365c2737d7903b0116a3bbf46fec8fa279b45250b01efe1fa4eb1d2171c8b2ab95562c23345a227823c43cd0b43d7e9f76842061ed438f485bce71a6497b19448c8b2d05ad3e7b17fe5ed8d3edbf5911849260065462430dac225517110ce07cd6375897fef8bd1c3037c6d9fc12fb3c2d8a2a70140dfa8cb1a1b3dff82087f7dbf5550aec0cf14ba05c484efc40c84aaa4b19b29afa82575e5c07e014ebaa99feac3165bb0feb7644c2d704915ef4b14c4ea43d4480840e499e5bfb4073afd5909f21899cd7008145415e161861c82c292285666d915fd56b9d4126addba722188b109e24ddfe5d693f586523eea058610a356d42e2813bb3d07bf628200dd55dab85bba18aadaabcae2434614fb3dc854f89753cfaf1777ab820519aedd8c3972bb4aa3d5d8a428c3a91164a441556492acddf279a44f13531ada4ef6f6b060efb32f0c9e2987d4a79c0b7ff65c419cf27f9d06de29d2fc1b19df472367a358b3d8a3f18b7fbecc62bf31c02388dc4fe3977c213a668046fd66e61ca8638b1181feed50cf641ade63de5b8822e1cd9b5e4c77d577e7087b0c95f5af6176de08b8339f585b31de5ff9070ccdbfefcd5b33457b4c4e0c0d8634558ede0a8959ae30bdafcca6a1a9ed710cb75df86e15ad15715c1257d67e8c7d8c683daf1ee3c250aa74a57158b38eff934f02e6da7e52fe1c2e566233a9899327acf320a3da853e997f63249d4f717f7985c38a237819050f2f8236af39b74f35c215408d896af42814907f0a2ed7fc8c1b4df29ae6dd9ef353be3b845fabea3a167d2812479e2c2627f9010200285b0c282667b4b4b77013ce0f995aaffcbb7f67467e35aafd7a97c382be272655c92344b119a226329b5c393cbd42fbb42dae9c2f5584bcc8e81ac972fb6fec05857d24d24df02d86764370c6f25d03c488e53471c3a627ad89abed40b89fa7c908ee1e8566090d2c9ef65ba210b0cfba3d0bde056a2b117e241ffdaf9862a74108fd6f06d4b29b5d10ed6f11a6beed86bf536eb7fdb911f0d58dd15cbdc55704e2b0229dd9c70e07c655cfbd53fc0858fe79fca87ac5d42dd27a0940a37cf9451c29f647b3e83a6093d2a78eaeb99f3d4cc6ac58f4578e4b001d4da27fa6d7bae18828ac9807b63e2519851632eb486d641d5b6936a5df33ec4f951f92c0547eeb76317b16e236641c12e5f0f60e0d15c52c4663bf5ed02cacd13d4e45fc099fc02c0f1a2da05207838dd5eeaa370c7a0e3a5363543585976773b045e415b2594bd0fec4ad323fc80248eeb81aab1a26865b0cb90edda8cef087e2dfc8fe873628869aed47c2b4baac348f6630b06b90a0b22e43861431aaff4a8f30a3f04ae188455975212a116da6450da8d217a0e76592a1f608ebc8292b05607217e15b38458aaeac24f80c58a5043d886cc93eb7e768c731b9de9ff6e21e98ede97984411d93ba13304be0838328ccf93b88518ea43b346f1badff8ee63c7a1da219c9b22ca743da9c3b7063b55c19894cdf408b098427c08da313c06e4c305f23887b08410b769b77590609e2cf5d59866c03f27c10373c45e97c511ba24e92992ba084289f50f4962bb3eb6d9af298ac54f61e891d9dd5b9cb968310d514536b4ea33c15e5b4c884db2a7508171b95d5392d8b136fa1aa0d71ab9b8ac2be0bf8c8ee85fec24d679ec354ba3177aafb2617c8aa483c29a8f14a377058e1539e4f31207728dd844ce54c0efb83e9b63c35db75861b4845ad299c933672da00dbb2606d78ef76b8eebaeb2089ce3fcba660b05e778d7adb3946e16c50982734e0e0c1f874d53bfea1ae82fe9e6d948d1c1bcf0f942a51dd9fbc9aa54b191f0a21ad2edac5f59dc6282785f67a35a307cbe27255a11cb8f4991865d55eda2bb6256510db231cc9c42cdc63d960245eee56510bd2d6bda0bc70563240ff8900bf61a4baad99ca53d3a3f81a2e1d07ffb05264a6303c6d94037b551ec6084771bbd8afc9227742550ad7383e1c3aa8dda5c44f21b6e34c2c1d0ae4fd4159c188cbfd9052fd365c0f47039d00941955a31582255d77bcfe896f1caae7843658266d292006fd6e75dc5c7b1f5f2c0749fb1ec13f97c652dfa47509f1370b3010eb7abfe0c1849c715b390ff3af7b36b6cb8b8054d4f93a9bec41fb0b1a05af917c5e5be16b77e73cbb9a758f1a1333b13dbdc8951c5bb122a6269860803454b1c49bcb206de0c51ba1ac13b424214d8991249192d006d63fd0caaee93ee6bd52b2de3d8d0d9bdce25774323690ec5a8b68927317f1402b8abf7b5367e2e69eab4f505e1891b23834efe9243576173551381b81e96faa0a00c8beb42d2f3096f36e8cbdce079579e8abe64e7798b9a78e1384458d2b560e6bd50fb54e3ee8dc6586ab5a4d24b94d95a9a5bd92fc0bdf4ba938e39b667a02be824e016e265ab6aa6bcd9462b809f5acc17b3966461ae704cb33f51a13d8924893a6826ef97ec7423fdd0e728e1e5534f719ec2e60906b07438f9720ae23a0c2db1d041e04c88e16c4ab84c099178b9549b8906e62873732a2fce2bcabab3098884505805be81f0455354bf4fafd68bbb37ff90bc55eda51a75bbf1a9ee481b73ebd147747e4369862f35bcbfe0ecfab6ab0d9d6ed4766c72c51b28b08bbff76419f8c5b3b2b4187993d6425e2eabe00a8f8cee02e2e44c0bdcba756701a0121b6977197625d58b895d516bb41c7e604752bb7b506e2ad09aa67690d7409e8510265ef7f2b3cf510240b01f3efa7fdefae931af5b6b3c14b4de44062505d11fee1aa486bf714fdb45469e9d01aeae25f21bd65e79902253bc7dcc40ea1e1f03e8ba797476ff5fbf29c3a13c807e3425e3711342795ddc2ed6b443d2dfa99d0971e1ebee0ea4ce4ef0887bfdabf43156b3bc733d2f05f016e9059ad49e6c9682f8db3b8892a0c227f20d567e72afc21e63c633dae5a8fab9016f79b9d4997e090f3d01253cd041b6bb1dd8b2b696d44fd9a49da3ab476768f6c7ab6180d00917ff3adcecf87ef1fb503d819b1d016e2fd037b179e43b6d84c3250f6498d23977d1daa7ae3f0da9571feea1d50cf84b6c15e4fe2df901779996a8a0ae96ff7e4a30f2c5ad6a8dbe4b6bca54ab6d12ce4c44d585badf8e0c5519bfaf53792f1c0604dc5ec3aa37a71ef55c9128775abc3073910e276a68946180a8734ee1d4eada92f4de8fef45c3b5bd0197257cfbf82cc9854b0bd9cc81e384f168de7f81b33c0fca0cb8de4b212d03b03bb8f4d5af892dba8e6893723959bc87d7ec26d8b027940e51050f6df42ac4bb0f9b925f0e150256e7a3cd79e44a49cadfb75fd41f9b28084619fbfd9808128442fea0f36d1356c2abb0e925608f1c0faaf37421c1a9a76be5b94e3de8c375064ebe99f7b4323cbcc70a5aec7afe5b4ccd581de47f0000200f87f6ffec9903fc10b2b4199716d3f84c80ae524a69307a6cb589b9decf02c2c67ed13fc36484953a576300891f0db316962665228bb1dde9890cf6737ee7362d33fe7c21c104a644a4d8c960300c5ee2ab3c40bfab1ef30abb66e13b4cd419f37a2dc91b9993627f8f1f7540e008e3466ca6c06ce4d7877d2a2877e50d2631ac9887f160977dcffbe51c04625e7a5bbe35223e2ec3a5fc8f9274472f1f86d0bbce9a6ee925bb045cf492be7f90774786d97eab881e1fb6d6e810de8d3d341fd6fc9dccbb969b30665729c90b2a1b21cea88642a5af4336f5e17f8eadd721ea2a8f6aead264a93ffc1a1af272ed8645e7e03e4f965f6b1cf2ab9fa57fee9727bba0ab4b20183f2a3f4f5561414787488ba867bf29a0bb90379db6d07209512703e97046eae031f419d5e629bf16d914ccfe4f1a913c7d7b6024f9257b79f443e5561eedeebe577a12ce35ccb3b5b8477469900cdddfffe5b50cb74525ecf59261dd33b7c31f32f9bca39f0c4cd40cb850928fcc8d435b0b6ce9c5e21ea395e909c66bd7e9ddd89286efd0b049451ef663ddbd873697910be11a23f89d6564f22374748a9c4cb771240d9856e17d0b036c5e2c0272de4645ae4d5faaad100ea45269f69d0527c36b1e04a586cf78ecb12a884bb620a6809e7cf566f032018b8e78be236e2576066712db32b40960505201b1e9b2eafd7423052518e09413465112ab000866128179a6b8b9fe4780a69be9b5baec9ffb486d9edd237eeeac249207d9a8255f2090bd0700e8d27dbf620a3669a2ca25ae25c67e25283390ec1fdb573cfd398c8b245d5e8adc55ca3eae05a1e3148459a2b21f9ae9b532b1ff9da2fa4c248216d419beb7bc7a35c94e4a70c0e1151a170c17988998829ab1c8c5d1b9e71fc9e78b2cb061f0571de01eaa5d7d3205b929c0495757f7bc0cd3e6b24feae6573f51667bfbffe3e674d6f0b7a369f2180f8652f1572b1d89d9237803fb074c5d2a18a445c24ce4b1dd787ccb68a386d815b0ea6ec367ed546306bd555cc3a0b4988241205b248c0d6f17bb582b5d35568e265d792c5101425f9ba0a870456faa102e4083258c2adafd1e936dd43dd5ae284a3ae7b5d661cf65c52414a0d49190e343476297c5ad89f60e51171acc70eda0845fb8ad2cb8dee3ae8666e43c051237068dd8dbd521019f2971753f16961cebf206ef674382e73ba7d8b06c202963cef7abff4158f35813e13dc1bd50a7b579e1b8f381235a426b5cdba0f6fb6c139a81bbe8645b8f7f3a58bf4952ea5062d1a447d8c89afc7c989ebdee27ecadeb6080c036339eef4ffe5b8a5f89260038e854242494e94592f12b9cfc58441e8e8769a7a32ab9a85276fc480c03d1576cacd7d26f306631fab5ca517350a287955d93cbf32f74654f9fa8e8481b526a35caa0e25a9a625aa5e37d077105bdd74ff1651fe602ebbfda119c0da19f7028e05a75102e05f7332238db7c23e4c86e654c12fe56869b1561411f93d470f828e5311dcfff383a8b5094f0fc3d9e72ccec0358527ae1d4e61225045f83865f148b3ec174d794f956da9476b5eb75106b30db96ba178b0b5387dd4f17506b2c06c77b9400261f7d7b8b3b4ad3e3ae1e8c89125b07b54cd7dc439f4824bb1e6ea2c03e3af5aa463d9e707bcf6057ed391518c8f8ac108732a88ec0298dd39ec51c5b133387e942d5db32cec0a04d98fb174f6e9ef0761a87986747b30da50ca72981c0d63bcd1e52ef152f9a69d8c749c159824441b900d0107b6673a4ab66b4a898e6c04d7115785313d56d3873549b17eae1eb36e6f2cc58a3cd7127dd6b1d7085abe106ae9a8c512a3bfb79d7702d7cdc1c9e168bf645dae6dea9475dc2301c88b9713e94f25ea0d06da0227f5d8b624c6da7d94c322bf908e9b4648f1124f6e365adf2fa8ea029be62ee1009a82d94de35ab1d53e51bf6882cc116949dd8052bf871db075cc4fd0369fc7255f375f7cd9c9b5aa48177e4783da8456836bdc226f9d05ea1ecd9cd56a6d29f97b43c0fa054ccec72d07919c60db9c7daceba63798c897150e0e4d8843908fef82737e8a170cc5841779828fbbc2d66c1834d4046da0f01c3f2848b7c6c3aa4d47d1a8b8d327cac87d9db8843f14efb0d3f63e6c633b367063481cc309750b601e62b3666bb9e6a8bcd94743cd4f659c0fdf39e68fef93feacc61f5b38445f9a9192754bab9b13ba4037f58b250810dd9ddb7533507b8a1d5717a04d03d7ab3ba1fb4d04fe5fc32aa3c5cc9c5cbc68b3f56f65da113a7bff5dc6883d67f23afa875ca8b697a801fca06f9e1e1968a126a9c3d0ecb490646cab65e3d9b4225cad2cc1d0483b0c37127cb328a1641e8541dac911ad1eb3a90c30af2decce9d5e26c04adb1a178352561478ef83763c1612996bb77efa72b72a358d2c4b1fbcbca23412249e98bf8c63ebe0386b81721f61c688cd0c377eb6bfd149af7487ceb6a84428ef63d51d34a8c01cc0cb5bc0d62935997534196401536be43885a3c64ed65f62815961ad20bd1847110bfd575467bc9f1862abed6fb41d1f348380d89eb9a44925095ff19d101a4d014ce4c4d6769d39e7521ad119fe06130a1a3f89fb1890300a19f165a54e391f9194dfcda80facce2636a285cf7fc8e6f2afba4277dd75a4ffd3ca0831c5cfb2aec974f558f683ef4338d2979faad2c7af69c90b509fde726d5854c381d9056be1e6a257ef0c07270f1f890ede0465edc48abf4d9e3c0dbcbb19762d1f1693c1bea3b343eb4051f01a20ed2bf7385542dde66263da8ceee9f9217f69f253e8875ef27c38f8a7f7ffedd272b13edd091f5a6720cb534977758512e52a5cb23b0b4073fac566ee9907bdb4ab7b002df5eb395818c7693b3db615cd3b1f9a20770839a0025854bce02ff59f6dc15332b16a253e2482d39d8dd8e520310346f961bc7743fc00cbd70ccb734e756cdbec25269e5b79c758629102c1c0f35c6f46f1544580d35dc4a9101ef9aec8e65432a3f39b758d064fcc663a346189472b192038a3b68539a475e8b2731ce19b4b84805ee491c4d0147c475b38155e264edb549f6bddde3eafc13e78922b37a8c867b1852c42bbf3d8586db86027a45f37c4cd48fc5e85c2bbf2356e788b2a430372858788bf2ee1ac619d5e5eab740b8a3dedb70a7cdd7dac775e2f2e2b73b7a02fe3c8a3fdd301372b536a6d92a8908705ca06cd90a6728bb94a32fd06fc84675b14771c52dcb38b4a069db48914102c7030caa40fdbfa5b125026504d7c874cee7cf9a77965d2b036f3bb9a04b9605dd75bf2c68cd7e84ea4e564755fc4f27c02a3d92940481a734396ab6c4f311a15406fb711e170cf0b98632a1f6945ffc3a2283b5e3a05b3077ec47376b1eec6e1a6c83ece8cdb35de9e806585f1b77386f7c12426665c63aca27722ac86a9a01d7abf57703b0867c1093982710c77c8d7af62fcb17e548d689d060d8e173a6a21e9a035cf09bfc3b193545d66f5ae8f0ead90a0fff2776215cccc58b2fa10b6ac87bdd2fcca869c6faf6af74b14b861e9b4f823bb19c3ea3c2f0624ff92baa50aba0ac779e8a2c32cc353d8ae5d857fbd646afe59db04276bcb07ee9b43c522bbd1b0c5de81d43a115b08c65af7b43467f3088c649eadb9b61d2f93d10e5933034cf5b6ee24c2d07d64886ffd0470277fabbc1e80f363c1eda705715140336de738ee6beaf590e92340a57b7e112713d38d3800a7c522094c049032912e39ebbc57c6793faf4d2be17a6339bf40a2cd97d7701b79c899a39ec8ec32f0816ccb6f74a56af2d9c8a79f3517bd8471c6e1c6ad6ced92ec8eb704772980b5568b237d755555016417e7ec6899b4132513e08ac8fc6d0dd09e13cd0f4d462c9fbd4fae37bad4ac257dbdcdef9ec20fc44298c734ecb56348445b3d0f2659eafed3c0922eae71998e062df1c615fc45eaf9c6f325b5632d99ca47feba19b5f17a6766be60f4e571f5ec49d2d0a9830db16834e6cdd2d39e5f8882bfd910da10737aa19f1168daf759b738543ed8c1d4cd2aae84acd9275da9f804cf41d67b5dd108e71134955083cb727185c4538170770d94a927f0e0784a7087f0ae5557e238c20a0791c377a9b8b3c0f9234e580e511d633075ee20fd02d1d6362ffe0ac7569f2699bebe0ba745c7a668ec7d0eada48042bbb8d21c2280e8296d7643a507058bcce3983ace0ef31f5dd9f450c82492c2ce5331f1a2abaff735e899964fcf1eaa979bae99812fc12c6740d9bb67fc24d2c7ec5b87b08203922692f69680fed9eb8ceac562400e704b0b2319a67ad236054a44e7bff1ecc235cbc56abb582d8ce5304d56da6b7b80673268f219f00238a70ecc15903eaecd521e4908972455744e8eb50af809e07ab88a765747b44a44c4f69fb0ed067598eb676a94ccdb8b55b58bf68f5a9bbe99fb2d828ce353fccae220636d4d3059cba4deeaba9db75744a9b7a400ea0767229dcf5c474855475fe1a3beeb581050bcdd285a9e147a76d0a948e98792c7668c11a6f116bf98c1b32c118f8450bda45e28bf7e36a3d50e2857773163bd60bdabb01f61d794e4180fc9ad7efbad3c30f321b9dcd8a010fa782b2f2c342afb71815b47c58291b4312b28a7ec92fc5535debdeb9e88ecdecc8ec7b26b0c7a194deda48b031667be711ad7756e61303b30929446432f7197c08bb9d476b3e002cb0273fed36a35eb0a116838b9e25ce7c52461eb25ea71569561387fee733fe6927db5e794986837a61f0b2bdbcbfb62be711c8bd368932a722b9e3967ce43cf6448a1fa1054180948695bc5f2a9b38b19ff166dac7a6fd27cd92738cc8d3b23adb9996f58fe842d371d524bbf7ee9e36cb77c9e5ff74f19c32eea3f885b29a1e5f2b923f29ffe5df55b22bf301f6aa3b92c88de02094d7b5c315a62649da2dfdc2b736099d3ca80d5a451ba10a1e52bc5b1f7166f4cc251e2af7e7c2b029897a8f0e33b19d5c44c2148bc4830cfac20eaf265c594ce172700f3797988df54ca809c105670de33a96a069b9aed359ea918e85600950599f785002e8cc798665fe71c861c0b8890ec6e2e90525ed85776a8b0ce70716654f2756e846da03fa1472e7ef299cb1cc9d93f140379726f0957af8fd4e0877f1779e10b01eb3b3f51b12cd3893082d90e93229f354f7bd9988b3b95d92257d2631a4d9519b682f82b2db944d6c2ff0e7bdc184092e119b1a5c1515e2b588eadba0ba25b5dc8373546983babbb50aecb1c91e4623e5250038ae5ae66e2e1c183781e786c0071ba2ad6b5e428a18fc2e26def338946a60540c98728ef45efab213bc4b88a8e0f62ae88bfbc7c7b60fe3e0e6bc49ec079b3af2d4dd328c1a049eee5a4a7e8a87bbd17f216c8614bb0b13432474191a456d51a7049c26d594b78ff35bbf8188ccf93d7c00c74f885971710bf25526518af60f99f5803e25f7c0d3178e4362f82c4bfbb7acb6e70f3d08dbe61b6708360bee6f714ad6ce23fcc559665af56d5be1423bbd4bec667720742c4180e3c305f3fe5b988d17bcd612175a1074bf3e04f50a4a77c291dac973564f7846c31f8d0ecc9d1c6f9703df9a18e9c88fbe7a2d986eb960f2ca147050245f839a90788c8bc22a2d0abc954e22b46d86657acfce170d7ce3b61fd5f12ba4a7cc573e1984326d9bb33230982bb8c8cfc38b8691cf792dc509f17fa70ec8ea1063562791554352ad7b5f644173b1314a094f054f4c8a17d42a9aa76dc03b6082864b87aa5ca0a0b635e5a0cd1d61884fb377c77c6d5d16adc3af33a5d952074643dd4fff0b115acd520099b8efac9371e8d8e5bc433bd8cc8fb1ac280dac9bff517de56161b946fca85b064a48209853f7d16e702514cd8e9dffad4f3f8793ef9980ce78edef7afb970dc9d0e285b78350e41343aa0723119742ae0d3a31c6bdc3f622f65802bdc4248f0f100ccd8710a3e4b69bc69ea108d8ec64cb55b9ca47d73898e75a54f70bc3fa7199f6eb9b1b5776b4c9b6361cc6e9f5545f8fead4bad8d4037c9612aa5f17beff9a80243752a8b643f398680073153bf88a3cd3ebaca89047f8ea6932dd68a046a1dd61677dbfa4ec3dd81b16dde624bfcd6fb0147cc0743c562b1668375515d5fa03a0513588ad60e86a446ea5b49922835497a481d33ecbc94f129914cd2cac1295bddec1cf9f101f1402196367d7a8d94a6b7b211c89c06dbf5897dd18143f80f463b37f52417670ed5f6ac942015fb303276a48ac1a190f2a97ac07584f9621635bad3a830f89a9c31a9aa484c1a5e8e72893435323f299aeb45c53c5c613f99132578f100cfe3473127194854b9829e0a0fffb0f42a2c6922465eead8245aa41c0e07de58a0f27cdaa884d9af1d1c40640e9e3833d76c1ada3f664eb8cef13d94ec42980b2fbba50cb6b88143d5065fdd0c5619962617327880a9c164c8d177fb506450cbcc6970c924129162b27eb1c9326d522cad2e5e6873c4927228ab1bb771cd509599546f3a6332da196983e129580ab1d0daccad3656ee5b606413cd572fce601a2a247117aaa8e85e1e5221d27edeb1476190f2aab34368f54ed0c96c45a37c1bc207cce710cfb7b7487aba060d8ba4b3d009419c5844deedbb082cafc1e6bd9c3037ec7a4d1360fc2cd2ddebd7e9825c40cbcefa0e14ee2d4d039a9722b5d5a96bab0af6cd8a4a5ee5e9306e4083de99f13ad1a96c616e467c386623ecae4ef1592954018aa07ecb5098f7563faa4c40a8adffce9e4b3a40aa1b9a4ef58c9cf9a26d674ec6206807cf1fac6ad5e05175fbfa2abedd67846864fbc733842818791d3c201ce8dc946ce2b7e9dfe3c5f95eddf2279fcf0c4ecb38fc3ad25938f9a8a41ac02b5754db65253b15b028c5a784f8b78c2fe2455075c4d4172240fea39ebe5d0353378f8cd7b2a7d80434e1435462b5efe254d0fc738065ebb0526e293bc97d19a7cefba2420e3db9f90a546380128597bda5d4901e25dce8f0a49dc2da50b5a7ea3333ca9e8330afd3581d1456762b8053cd3cdb6a81cd505419225796c5191a1f972313bf5e5d58156d4c2407c64d0e0509e25b2a8fc237e8095758c85c4abdeaa1bb175b97e675761f7ac974be9c5ffe7285764eca01cacc4bab3e6319e79d0fd3d0690d2db898c188aeedaccf63359d775e82fdf9152fc90a5873efda1f9cb78c4c45fa01686441be2aa52c550330db848980be9b700849fb015cd7be2776d4dfbe72a2f00872acb794c69871cd0233ee0aad3416f73cbb8291244bcc09e8a60ca0e765c05c8f7c9cd0d9a77117be8dafeb4c7d3be6b05e79e51b1a9a239a27340ad6cb610e0e7574e9a19a4fb3cfc3cae470f7954fb2c4f4ddbf4121ae380f2994dbf30e7b529ca21247d9b428e48a57cbd062c774d2c569e22ece512f0f717cfb1f5df367e3649e4ebe172d5c063eab2c8cd34ac5c5f1fa192d75e0ebf69106fd0791063f5b35c36195bbc9e16f481ad7c9b4b92a40312db01929e44c84efb6c04ea40059b720513955a0e691eecb37b1adc7053e7751815aaf40bab188c98ff5c87966ead11bc91a5219c6f27e8ab3fd31733dc27bec345bdb61a7e0a20ffff836417fee49e0c8d6cca1a9e8c24e78404b0068f5d09e47599135cd8675844cb27396a61c58bccb7e406158aa54e1596ee2b47d0f0c6a1c57ce677c0209bc99f75a8a6f53789958be800b46a4bb92ea080f67235f8f464099af2f2a88f7837d5ee4d29a85ef7f83d3cd53e1b45d8a7b9b051f2034a994a30a6ef64ead161710ea7c2476e820f5066f49f676b2b597ac40e7c8ee4c7be72a47f8cb3bebf754afd85f36949175f61613b8f3abbf1564f65713d69a848dd954992c705a89624906b828c25928d943a82ee56f61a7192cfa311c8e9363c77dd6755d3450e9ce671f9543678fa5855714a3dedc11ba6f0cc3816b45cc2b45594f577b09eabd9e00779349d879b970a06ea6e526af606f0c2d6ceaf668c5a13614118eaf0c64d13b1ab42c28e03b4408a8dce6f90e9c8a719258bb9dbc731b8665a361a6f73d40bb463ce603fe750a98aa7485c3b5f07618b4976e9a8f2fd1dc55d1465418b3b8228d099d7d59c919af101ffc7eced8a9e14b97113a6b0a8e49870b4c33afdffa17a946de24e2612f96d06c092aacc51cb906d93f476aba2c569d3c234bb578b91d32ba6e9bd6b41f7c1b48bfcfdc8bcfeeac83c2167cf423d4f675f6f3fc9ebd5a794acd042fde1c908d51958ba5fc9ddb02ffe74a90c9614ef5e90735ece253639893b5e3356426c684f58cfe2316f1724e776850f39a4a4b39ed3046e898eff86684491eb4cc56219412face9380c1ea96051814d94b969fa212e69f46278fd853a2709cc7fe50ab5b4b1449d9cd30786c7a2fd7ed7ed40916f735e5f9bcb51902d7e8e16fa60ff488ebecb93fdf434326312a4a4f77aec881d346971a912e718a9dda19907413cc943e50d4fa58753806291ce3f839cafc56b865e1e50aab3ad6e8caff0e3b8c9ba3eb16dded8c7f496ce5251bbb37bdfbb1392f99b6c968d470a6f374b967d29e649abf3ac4642096d06ff12d524c2a48e3af86d17b571876e44ec0d885e57972de4acc5e68a1f016e890bd59b3c7d7b17f5348b280853b7bd43f47af7c302085ae7b991e70a8e847b1cea1484013e1c427d443f399c6ec2c736b9f1dab226826859a424427782206f674212a170bd4078063dbd802425d1a7291be609f94bdeddc489b64f732a06f06e2234f1fdd5d4595fe3c13c659975ec025f59eef6b97f40d29ab54f89cbe63163b21a582972fe66ceba10bdfbd345e47cf77dd2397dcba2c4cbb2364ee7360eae1fce91d7a59612226927a8de97a3dd477a53e80b3a9fae88f90ce7f16804a02f65235d72a68e0bf3fad77fe8656819261e827504773566f2650bdd6519bc345b18000345ab940fdffe7ecabd0035325f87aaae957a7d0d0c55de08d46fc5adeb4078e3c386c363e6038625ca8f5dd550cb0ee18f2d2ff392abee283439a6977477d6a575f91c4a870358c6270d0211378db64fcc653ad983204317f0156bc91dc4d1d8e844cfa610656f468fb2a7a7a470d298a94cf56c1081153f2c3f02e3c552a86ebeca5133661ecbd75ddfcca3702ce1d08ba1efb74946bebcfbd6ea0e5ae793e6cae804f685335952f5ee7c42d3a50a7da12c4667639ff20049af5c4df45129c8f8d8cead10868134ef942ef15873975111d79c9936fc654e4f473fb5f5e59182885950df8c9226b03c5ae20fc56c06043391fc10a6da6a51ab9966021488def4a5ecaa8bec539b8092f0bf889394667d9d20f0028467f54ed7cc4a5f405e11f09f37995e1ad21930fe2dec1072313ac1cad2e20feaf5514b8ee54fc025806d8229a135f443bbcc71016c54105b01183d3ed5b5dc38fa141758032eda61fd15efcb721f0a78772c858cc1f257e63abcc83866315462eac2c64007df4a84202f8d6603c104150e54329d4c72aa59f19786aa4f1b6346b4dc4aacec9ad911dd20b3aedbc2673c211a5fd7598e37501eab82aab6177063fefd88ca90fd4c7066005960e456628bcf63fe8115a7286b177c3771b73cd9dad2e99bdbde8e174d8c04667a5e724d5d8bea9109d393872d12b92faf6213f599f7d28c5d393905de1e4a0cd1f2317f20d51d40a5c28685867d3b675d28ee2b0155d742d2bd38d3124d9a36f0660fd6542a1b6c8b8b3b06613b9ffaa7ea909cf8c5cc1d90553c66782b5f135ef61a3a7c97ee1f5ed15afef9d685743271970b65e330831c4676bb0988ca846e7bd6a33c44eff88d9fcb8aa9ecc71def07d0cb9a1551c8ce9408e612090ba8e95b3a210193619851b7dc9ce1a0599ca0ea480c339fec61f9d671a7d096b09f2078ac15c6adb53546e8f9fc43810f9053133461f881b53200e2fc04c0ac7f2b0136539fa7b303e6cdaa5063f095f2df20133e47f240567a81f6a627b8045c473c1d1dbbab7db888a43467ad867f44c2eacb5370d0602cc582ef8062c7d37100f6f94da0dd445a34d8e5b34e82d6b27e684cffe9fd69fb3fcf7df6376d0d72dddef98d8f0f6cb34727cfd0cf802c456506f48cc2baad9c3837479a1ebf44c87c1c4202e64d5e2551541e1fcdbeedf9b7b1666f49d9f7e9d45a7d4731e2bba6f833e7e6225a03a5ecbc815d2ed5617d783245530e5a3d3f6f4348015ae7553d8c338a020bba769a68ca73c306feefa47f1de942e273f07dd092d716b286bd260ff1cadf881c1d2d4c351b823b35b0f094c3b3aa60af92cc5fe3f8d7668c329c80f3f93ba09ce124ddbc8dafa53b4c0e6624eff72ca0ce3114cf37d2c069b8efe7e8b7fe4c77eff084a849eea4a3a4c812b061c3ce62975a3c7ecba134ab2f73a13af60b079557448625561df2a5a4f39aad2807c2f28fd47bac3a32247d94eca190bbf2795e517b633333917cad7cdbc4a336cf3357f56306dd874928edbde00c51f34eba45743d30f39e04c1798f8b62e4751a742c193ef605aa3b5cee04baeb7acb893bba1eac7e80df835258e727ad3db7e544bec1cb825cd53380d12cacd3148c8c1cc9a74d55f8bdcca334d652b7207cf5230c5f33a3236e323dc1a26d861139641e1093b604df24ed8016b0539f4a7db0aaf34972f08864cd978305b536c60eb55da44ea7d0ac19d2f7adad5c72000696afcd5dab2d2365d90b9259341558a7c0ada2ede7aef64d26010ec7210190390e8109f1a0be4592ae65fcb51d1d2786a988216ba65d58a8606979ffddc04a4bd20bef19c9a521ac9248fafeb3826811897e970096fdb65225aa0018fa0dd9c3cf8e38380987dd95358ec8cc44b17898c1f77eccc415404970ada072a0452afe54ec3b31cf5e68ea2843f59a19f7b836e18e80bd358d240fa1d0a709db6ed435a0a007efea7914eca8762dacae33225088441d416255caca28c994f1b5988fdc742ea7d273f0a97a91254724a4acf84b62be4cbc1202f34e333c27a006a6fe0c2f86f67999806e838c36aedd63163e315bb202a235e0300d9c3cce0ed32f4322818a3deec2c2c60e7b60cbea076901380972d17c84c6f43f83f8a852133a87362dee5c1b6ccee7ca67cce9d89ec15391b1fa0f74f78afe07994f4c18300dd7d04415712b8c72974429dffdcf63fd1d5cd2a629a5fe3fb6a0fb27b2e114d7f52708d7d6804763e190c932a11c1b2c876397f22307da7b0a0a6f443d09c7349527f2d57100fae8e713aff49630e05d927cdaec26337cd87917bdbbbf08bb849b9d6defb295a6ada9c273a1e6ccf6c9aeb26c6b3bd834ce75984e360d9271cb9f538843c3b5229f3e5452cfb9283b21a991199ed32fc24ab6e1fea225c02571c9822d38d94166e352110b8c5ef04ba6997f92fd06a5fa7b457ae16a11fe3a093c40eb2ad73a1333e80c35f26845c09e7e4dbc1f494f44e941cbc24b1e5b884140a2230024834dfbde99e10fbf0014c91301b5e07c9915190fb0ca283606c35c71cde9f91c5a803c9c15affab902f0e708bbbeeacd925d67085d82cb941ad6ab170867e48565bdc72091de889145b781efd714edd25bd31bec0a6b41faca181fd467c1fe2e85543216d6d62e1e4aee27ffed179ab7ad8063a5921e43f23b4ca8c2418de66c8dc59388a9cf5e21344e33ff4f497825464f0eede232c00b35f428dc6f2817fb3491594663695b7da1b0372db1c07aa03c40fc5af4f2702b5963b66621c65dc5a082db3ed7e895ce5afbdfdb9052e5d88952afc44048170ad8b00e4f0519e753357bc84b9b0f99d4a78a081a029e9a24e44071c61c3cf85849e52f4539ddf5a238c4caf74f6a2f0dd75c6cb706f42f8b9aae1db6623de321431b2d5192fd577c68d131948e924dbe45548bae2e886cb31f525ef2c2f90c661a1c3b13c72e1f4557af441e304d8cabc6e37d42823990f76e3e860b9b5526e58c485e7213a75367d446ad5bc0d87722027008d637f6479323f9a333784aa8b2222ee857cb67225f2f362ab9b3990ca40c9eb38b970796289339e691939b875642693438719dff5571f0e54c1ea49853c7ea87961e1e8e2b3bfbc4c61d841cd1e4567ad578c974d9da486945ae81d34ec5403f6aabac477f611950066de807e37be49897a22dccf908f5e74d7dfccdb7cf9b2938fc7b58f86f80d91de78d1f21661ba74f240d3bbdfa24247c19826b27917026ecc4020024f7cfa0cd484a216c4d9148de2db23b547523fadea53e7540c9812d937b5d3139fdcd4f1aa0349329cf14136cb3902a6c616ffb9452fe887c0163dd9cfe6426fc5b25ad6c25837e90ac9f6ae7909f0bffeab21886660aed361d3544300b345489df4404a997cb7bd79cdedfe3804b9058f025798d69d613de74a631ddf6f89d411ef0a97dadfe7bf980b988c6f1311f121b4535b658630d6e92a17a98825d2813b048b359eff8800675778191b1e9b192f171eec815d327b5aef80b6e56d4fdfb676adc7965c81c3a738b1e1dba9cc000796570901d238f57717912a37714f4c2d7a5d48050cd845f37077fff6ccc6c2c4efc5e905a09d226e3af36d261bcc3139805cb74856a0064983e230afdca89be366ff078c22d410f33576d9a9640bc412869209bf5fbe5803895e54df3d9187c167b6fa4b53c7bc630875b6ea314189efefce3d52335f851fb1cdf45b4fae0dff3598aea780929b538894f090e0c203843a493cae32bddce85be2ab70a03e1ab00e610a0fab716a4e6c712d549def034442603f6da601783a48308d20c19780315d58303920f641b6434d5fd4c09f7d04786cb4568bc844e81d2a44798176fb1a0e7e8fcb0731f5d666f2d28734da91382c2856dc45ee141dcee92c3b16b39bccb422b8f459b92180372a734b1e662dd538705c79f78a0ba751bcc7b3a197b1fda4e12864887db31dc7875c8b8e5ec3455ede0a1da26317a57280f273db0f5587f01b20d7bcf2a32ac37b458b9872caf19dac39959a2908fa675cd2417892fc7e3d9e13aa7a57408dc5af53f4a23be62420f317592a837fcaaf2211c7fa1c62e95160e6cb37297448ae182a51b3bd0e004bc80cc6f3ba5e67bb0a7ae860f3944ceaa07b98d3f9955a29d7123f355013066c52f4933ad7dc39973b610a38d478c554fad9cea34988e27604854462693ba2ae321b743a3f5c2d1411dc0d85f593dd7805c6303f3db7c751e41f49f240bb6c3f1afbf7007cf87aa52c606790491ed588a425ebc8fc91b7bc3d7ad76dcb2973c5527a58c04b33c104b55e0a04d55d91fb35477ab22759d2515498fab3b9c4602ddda291d83c7e86e9a41c28c547eec3a82c3275cf007db0e8ac82054f0493d4921b80c5e65f52da5f20157b22fb06d11998c460c34c5752d2b44d95f5ac2112a19c387693cf649f6b2252f6b0c8f543bdde249a08f14876d4607bf4479a34e1e6e3d42f8aacef99df0dec4fa7cabc0672384a43eecd4b2dc8683cbfec24d824c39c57ab1bb0b0bdf3b0c8ffd62096589d53917a11842f64386dd542d1a5524d1b1e24f44057b6b68bdb732286d8ad940ca484bc2b5cc0b88354be4d02ac375d6f1b25831c1acf085b28ee24f3a438e858ee5117044db7a8d4bea2d588740f28f77dfe7c186df135b72be6b8daeb60b68b1ee98ea478f496aebe4ea3cc8acf89839dcbc9d08f4ecfce5dc66bece3688108ddefe264ed364fe2f108b465b456c18712b3687c702ec6f172b02a37ced168891d9450aac22bd65efffe44c2811e8e33cca61d9404015fffe029881c429fc1c5fa464f7a9ddda07ed23db7b9736b2237bb56103f62bfbff9092837325938476656bf18b8b5346a0d4d37b495f85aee6f079f6e1cfde66eb2b673f9ec9288a79315edc98db0f098e39fe1e28ab73fdee2a66e2366203a0888328a6ceda006888ebf57d9afda3bd63ff8e1ef433240906f1c40c42af0c2822c70bc4a41743731c4c1d5e9b6025fec7096cfefaaf5980be1428c7faa7f9c3681c77a53f59e81bad6e23c18187e8c3acabfdc66d660b7010a0d2eb67d07dc9da5a5a7f7aea38d2453cac4c458064d6052d01f03e103a6f06308a40d313f078115ce363394dc3dd6f55f1f0949cc98b70863680924a5c9f30556c2f22753b406a0acaf3a391fd36069cf9b0fde24a603f036009150899064721125b74e90292482bbe0d471173bd3d5b97bc2000c1b97e594371f2fb767c8b121f821d21c1470af477c7a7293a713650b88cb3efd2849af56d21ec15d38430e0c89e3335d20e6aeb41ee1901bc8d157a7411b2e5c6b0dc5b305e5759ab3603bb83787ec8e6ca0cf86f4868d5b92db678262feee7eac8fe6cd4da2f32e22422d4d1bf012c13ff21466b597997570745fd3e82e406c6e6b2408fc3973699288b39e5b317e15a29cb43e30decb717f8def6ba6a74d7b7c689aefcbd631aba6a71c2b4b505f3c256afeaf4ebc1cac2ab980fd55d1bb6c61558935738a43df6933ebc97324a2011f4b7dedbf0a740905583067a04675262cac0789f3ebfc32e971dadbb84c0b36b5501f3a8bde579f3ad8307f7a35fe1a0259d4ef7ce3991475cc33456b5d5d01d86a1f866630a599d037213222a5ba25b989d6776af5157334d871100d610a2378f20f28221a193ba39af63dfc368fc7fd1f8ad58d914db6a1f296fd3aab912a595ed3d5810b30d31e77c7a8bf37c3b333cddcbba0558e5160ed7b7bc9ee7ac2cc98adabd1fbf9d86fa102cd5082372f016a0cd285133d883655d93dc256570c9ecf1a559b030198ee58ff953c9c0e76a70d603371f5a4237ddd632923033cc652e959b2557a62a1631a17939db7884de6406c93931ea52a4b742536e21917f75b39eb6a46df7b34ced693f26fc90185016ea1f3de265b5a8fd8be87312f5ee27f7c1fca2cc9b8b1c68830493025fb02de2f0446472ee5a4cf8934669c090ff35b426d3e5be2895925a93e87835de0f27e08f021f2fd25093c522cc793ba3d3d29966af7f5e353c30d6dbdabb10ad2a2d855818fcf5fbbc04f2fc684c83d8c66cb322c04ad65c3d79896015227bca0f2ee1a333c1fe45413ce3f8af2a5a264c7c1da08d096a9f486468ba0a22642aae54e71a048ce6203b756a694aa46ab8be966828579eff2998dfb510045bf2f25ea8868ac56cb17f6f0949fb71ae8705845b60ca3d8c122663b3d0781795b05fe980b4093aa14a966c460f3a8c4229ebc8618036d05379635917e344588acc7f2abbb328e2ede389c48d40d4c08c0055311de5a282693a413f83490433c246c49c7466ce107045349e1bc90a731fb12ce44f488a355e69ff2303f03bb2ccfe16014ded0e1642b88c62499fd6cd93d95039f30d3bf880080bd392f404e1e0cd54192075859ce43a70048bdfca9f25f50dfea3527fdbbab8256ea95702674bcee20a04cc96f727b0063195c13611a883fa29692b43fe89a1f9b14237e3473fbf78075a6ae0284dc6a7ea076e7db0bc51aec4c4032034bc89efd1df5732cb7079f5dea57eb3c05156587ee1f9e9046928f8273a4c9bdc53d9c3d13f03d9e9994138aea8dc68b1e2c39fb1213d999fed2cad828ab2cf0e72f7a6a5639963467bc9411ca262c19a157ef7cc47eb52237247ec68a7cc2d9d8c0d63c75e685678077ce023af341c974dae1f80fce50eff7fda6d20d7740cef70106c463d39616aed41e03144db8f3dc620a8a8f03f51915d64423fab4ed461ab582a1508eb176f7f6beff0313f42263ef6efd670ca971eab66a472d259a5c0f4ec8dbd2d54454437a3de59827e4db71e45209128ee3d1e7158fc69f18bebb5263c8e5094c47b51010de45d934925514f5c9a84e09ee5d2c62451509d77f60182047a90ebf5675913b1aac36708a63b353010c360ae5cf1df0ba1a0c4f87afe02da636819900abb9409e92cb13d42b2665a246cdab4cc6b11e08e353ceb1ceeea657092a8e18228324c93bedec882abaf346cc093cf9a94c3be93b8b00a636722b5ee5ea2f03d47d1681498b1b9004477a0a531e22b51500f33915e44c2c629e9495cb70969f461b389e149a5bb4a18673faac2a55100baacfd6de082aa463183f6d94fa37f7154956e4aa18c06b851011372a19444a749f06996073f7a0d539dd96e4e878542d394b9eb0219550904c7c5ea580cf486933fb7f8a6ac9f6ca6632ec77913975340582b28179ae5050e517d21c6aca20d4da93c62ce66725d678ab62f788e19b4b4ef231d909bd03b08815e7b2c9ff8d85d6c75e5aa6333296abad62621bb7a7b4533028e5beb3a2050ce4202746c63d40664dd996176c50ede03bc869faa11f7098663942e348622403e0b833c66147f70fc6cacec85514d26908bfd50c7e1199b6fc6e116e26f6a3b93a21a69b55535f0e63a7148d28e00717b5fc28e06f7cf3902e907b5b1a81895a6e95a5c4717aaf39d9f98021769148366f3ce526fba1456dcf62078af451b944f76ec2921c0a1486aba6684a003df151e16015494fe8ecfe7287aa04d5db3412885a8fd2af3ee1f2cb901c4a34ae0421163ec03b93c9f03f3eeb2665b7eeeb7729cfa51678e794c7a596dfa657707ecfbd590d0bebe42179b4d1edd6dd060c68665b90a46aa16df6bb9e8225ccaed11e285328b78c224814a694b43abbcbcc4de7ffb30e02e23dacd134c09ff4361d8b9655bdf0e3c8296009030ccb01bb92ed8889cb908db7f88a882e0284219926e30021916e35e2bd643b5a20d0abef2fa216d735a235ec1304c14d3a990330b46d77b2f1bb107cbb4d6ea297c581f762d31417be3516f6726c51e7839d958f1ee0bf1f0b073806226e702f92315033945d6bdd01b7d7e853a44dcfcf52be6e9396d17745f6265f4e1560f75d4d2248e480ffc119d6b5568fbbba2e9a3992d08d1068f6784b990915b70ef878431de865e33456bef1fe7fade0e2058cf96255f5f84a5777c2582791bff32f4482d5fd28cc4d81594912c7fe1fd9214b6fd5a6ad66b8e6e260efd6aa72d7316e987a0f3688d0f5c7a0b956defa649f796686cbcafb7b6616a97a7f5490da105cfb9d7b7f952d095e418ccfcfc2f146af48478ecbbf3d5879f0b6a668d8a3a28747f1030473c332001998ae0d9a6617686f990f4632918dbaf6b26545b0f3fc247e015411bbb36407757fd98bd0b226f0012c031a9e5a3d35a846630b345f0e01fe7b0a99daef8b78ba79b432296684f2f15d568c1b763c138feef1f5ca1c6427b6b2c4b6feb607b877e842f3dd561e6054973362fbf61717b803b9a6dea8e31fee33af468cd14794673e72d3b81e4c083c9e795af5bc1dce222776bcf358f06e957af1e7a1e19b635fbe0bdf81f96addbde2b640e4212e50f7fe06444f98310f0792bfd83539db1b1bbcb73c7d1ac73878e3ffa2e7b31b91010fceba1a87877ac8aa83a9f91cb1ce1b42ea74ea71c228f070cce4530683c78fec812c70a872f62720e3bda02b7aeee3317ba3092679999e6e8806e06be92997507acdbe7607e4b04d8eb6031aaa163174f1cdc16ead03b30d8c64429d2ad95f2ab682ca625254d0ac48da22079ec4355d8daa0aee5dc19975ae98ad42faf3bcbb8bc941deb60e6d1f21a567fd751c638c1248a202c0173ddee94fc854890def04f0487bb3c39728403903cba81b0655d0b1e4c3f3c122be57a6dcb26fcf5353d9440fdec16ad0ceb3cb4d48a2918f1ea21e3f845f20431f1055ca70570638fde9dea8b6b11cb19325d695ca5ef3991942666d9e5506346a87dc233287c9682394cd0ac45e64fb6f180b7d35c2d686a5c41ccff5452cbd1f1633f7b0383c16d26ed76b280e20ec7ddf81ea3fdd0bec8d927d775adb83963e7828e3d5f7b0b24e9775ab1d47e76ae4a2fbebb3c29fd64de2c777756465d1a2fff129adc0486bac8535d7e27d6056149a1c0a8d914c7267a0a3b7cf65c07568fa690d550dfeaa57b9ba4cb64fd72ab1fc42974a80237f613d181f0b9156d2787d78e5e51d9dbb8496c09596310293d6c968a1e444dc239e83601717f1393d945250609834d52debba60c6059dde264e67a6932d508fc23f7e69694d7d2c3154392146acb85f1dd4067ef5253ee7631fffa5cf54a3dc3d04b3f8708ee700d078801ddbca423b62b89393337b1c47b6658691e8fe2f280ffcc884443abaa9e226299c78fdeea6ee3d0c5f8f93241e3006b58b338e1c6726016709b81d73b4647f43aa9724602f6c9779fde67d11ddc5c1e27df043385d6588c91e60922ba9d42a4c0d799c82b791cc0f511776be211160c7d88060d4bd2a2374d613b6ab32b21f204512d2f8e18ef9c537712b37e84b4ede7b5b486fe5d9ebe448f8c110fcd0ac7a7d2f3ce15008382fc8e7740dc8766706fe4095180d5427f59f2fa745a51fbba521bcf8b57b947e24c0c673380f720e5c0dc9258fb38cb422baed524975bd6d87a27f2bf61ab7afaa1b05ba609068cbea784ed0be6fa7c269b8d7ee8c93f2961eaa69bce91bfd2a2c3a16856154ef1cadbd10d8cdc401a4526c3601f996fd282451f3bba3a5c12b0f0aa0b573f1d21a9b5ae8482ff60cd56296d7d7f2d8e625dc47aaac11af4473077155bc8dd91b10c23a8809235f8e81a2715b83112202d790bb500a4920f86d61ffd230153410ca7c4facf902ce433b215fb77e08e9e3d7119c0f7d21a85286188081b08037d324ac7d862bd51e50f7fce0e721721a88b2dd20d930819dbc71c67a57d4ab37e4cb90f0037d38db4246cfc0fbda280645798af145f92edc659aad147c2dbc6ef8009fa9578f6ac58eaa959a609ce5a62fb8a56288193f88f6d058f369ef95ead932b53b398bbc9fd822ea0c025c3e9bcd8bb08c5f5f602bf14739b1e65204ff946614ea57af094f8d5d8e373f849c641bac90868553fabef61e6e459e5470a59e27c89fe05731d89d3371eb79ed6cc13536f90e7fb3efd0dee488bd3611e85e99eedac8d94715bbcad1ee11a8130002ef1ece7dd8b1123fd0015fc0c37223cf959d3de4a0d2a8971a5f78af14bfecc747fff200df1a0d24f1320498bfd799d368630c183ab3bdc7938287a0b6ae569d79ad68aecc9f70d75b06ef8bbb88cb879feb8fcdce95312165173b2b62348694f4aeb547a728ce4eb9e27fac9d98e4e1f34c9e0f8ba8b29de27849f94f8d3cd769d5f793ca9f9222478736b09b2ff369d78c021d4a4908384a0d58145d6c93b737ac081789ca5448ffa1e7621115ce150063c1f67857f88a6774bda88a3e6c35c5a6f6f8c748782c8b39f4984e93e70b8cd798176b53c0d04ce07fb542658aa1c0dcda3fee9dbd5b254aa98d5f65dc5016f32118e93eee28b71662d89b6a9920a57307d9fa2e87c05f4a126164126585f2d1908492ba55ac3b3f37955955aa6b04fac5dc56e0fa0f2aca8af9487fbccd0b83370d63d09431fad25fa3eb7b4d40b09d4b46cb4d6fa0710b7e7512d20160045995a09c44e14fb6aae19a99d598d44f39f7b4cf422bc9dcf0d4fd623a9f6eafb0c2fdbfaf9679e844a96c6e81672684406055ddd9e103301224d34fef6264572ebd60fa946554b7d609219e67b4ba43a748509d191c5ca689f13347145aaf556136c543e0506f4e445e6ae86a9b6d7eb8410f47ff17e8258f3b3f2421b8146f6b8457a759f076dfa8517f088ff6276d46cb29947d00df0ac76dc11979adad3027f84da3c47b4d9ffc6763c6441ad1135076e7b06d3ab7aa24b6636749cffd3f559ffe48c1570f0cd55f50a3cbbdcc855b93392fbd3e5b6243e8c44deca93f49fb2cab62fe9a0218e20858bd2f4228063f4832e695404b2d65b3c7fdf02829d850d1a441e2f97ca7f3f7f9d1cca26211e756755df9a6c9de2d5aaa56d722c5044dbad569182da3b4ce6ecf4cc4e0cde094ac90a81c3cc6aef5565d42bfc8c63ae1f9a865f971389b5afa0cb1cea9858a439f55db211e934085009dfdaf0bef320880348c62edb140fc79477f6c8e12e658962ca54e47797c2f1b0bc588e4a60d466043d6c6fad73d642532ff235c1774940e0d7afff3500f80209053b1ceda186016f35e4f4ce4e3e232e0bb4db6e060ded5a5963a086083ef652a00497542dded583c88897b0bd918a982cb35f2912a8ca2757151b77e8f35f653fc8bbe05e71b9619dbfee9bbfa4e40eb60d2469d75355bce1517faa67f5cb132666d2e16907a54c20efdd6efd47ab46ec04dbf1786abd796a97ada6b408a5e5419a66d27ba72c74b78ef532ead69ea9722487ad4c106027c0d5858551c66d56e71768b28b8f1ca23ccd57a886990193906e3c63700b40f14def673e56c97f3126b48accc991805d6f8b461b2e20129e9870927f697755d8d0aa25912842a2ed1a5c7b187f485d3fff67a657235328a0f813726dc2b40ed2dd575f24eafacf8c656f19fd97712c21c3a18207fab98a3f1d5259cc473bc69b906704b0ddff9c50632dcc81f3829e81fb5831372bbadc80e74e7b5471643a0b336043d76c458084b01ea67734866e31e721e82caad6500146c9bb2b9130f581501d246b65c5a8d0d0cef0f7b24ac066e4a22fb1601313fab7c9381673c9a99f4c83304883a77c3106fb15106ee03a9b1877d8979f420d299d18dddcaff9ca9e74d5cb1c709c9c2293d0d261ebe4fe9e057d6300f1a05e852ccc3bb5f3f70090c163bdbac6f5e4512189541149975c8ce19a2bd8e1a2f018f35a5f1ffe085ef26e993134455a4a7cd9f75b9f210ccdfbbb493ac91d0d041209c416d8db98dde01b2ba18007560fae30ca55d132bfe2e28a4ec875fdaafa3d4b7b7cae9d0475a4ebf717994b436b175dc79f48e95e871746d21f9696c449d167715626bd4933fcdace936d883d195d87115add42e2f47bc88c14b4b9712187a57f563e809361aa9596dcb34e80117b47e8116f7b896ca2f50e99e2fa86d776b86d159b37debc8a9719904af7ca22606397970d74f85afb3b5b6a1707962e72615654278fd2cea22f04a3f5c86d3a841f6e8d08238d03569a2fc7ce58decbdf01bc7fac52c73af892c02810419bf49e70f19488fe2899df828a726e7bd0f71de4eeec0a0683b197b18dcd042be5dd6fb067b291bd224d7e7bbd2ee26c70ee76bd027931d9d6c14bafb893c0d4bb065ab50c1cfd9e273cedf88427b764241cace5a99b58f2b640f39d8f4fa82d3f48ca74502aff7b16e026e844ecebf6f60ef8649259278b77d4d2b3c4249d5375c7c740ce7d4ade158713b359c84a69f4936f6af3cee07cd6eace185406b6a9756624254bef133e60f6e907f398fd914939275222bf150c6a85ab4ff11f0ed20ebfefa6efac90edd3fa71fc5db7594d725c1abeaf4bb7901684cee4b490135538a2788b0a488d87a87bf6661a55599904e9c5f757c1d9fc623a4dd5a275ca0f5d1d3ac426296c93c7f217b457b678c41f7b8dc20fa243bd7f0fb5c0dc61541c99cbfd4cf4d016f3e6f765814b379c18bb31c6df8ef6dc47d58da0e561f46077d13915a42b12b99ad8aef855e0a1918a3580b7d0d02f9623392a96c1c66e7eb3d4140264cbe70934dc8e16de29619421b9694de3c17b43886f4a68d68b33eecccf5c482e60b9316dc8709db26254238360205dae359055579e1b477ef6db77737cc7d79f3969266a3e1f88505ad99d1c9cf5155c230def0cf17fbdc189a0c4e14f0ab6aa669a8f6924a43e79eb420df03f5706d5bfb50817cb71acf794b4a2d49a6ebfbdd38ff5e8e20571cd46290b1193b8888e66ac25854046c2881f277e3daa08a84067477d974bdff3e7b2d58afe313e5ffe24521004a4b1b577ed2babb7101bd329b543495cce08fe800319487b956e00b9face4ba977a1a4d097a8c465b6db2d715d30831454a12f636435e72533b6959086d1e977e9a0500129483d09c0528a804eeab2bcf005df8ce09a44222ab7118bf9af818cde25b867f44f224af5d0a9e1723888a623104c72385e020466d9bd3c20fb0df4ac955a1074f8a3baf0093e15e996bc0b043e404f57063e3eac20976b1d3132b74dff662f2a91955b5e9da0611dc56e9de98877d9ecd9aeda92265c57b84224b695c5728a86c31bf0ce86f81085565298a36660087f3b76aa472393b9f98d604e02e7a4a2bb6480115dbebe14eb283d61e6295fe9764d166861afd53ddcc6fe8495514ee0f71c48000ac57b321dce3132e1d867df7771adb10ef298e3ec7e3c304b5b2dcfbf7f6e0d28c4b6bceeff7313c5199ae895089f3041c2f1bc0397b6e720d0563344f986dbaaaf78299b354afdc99946389fd3b6d25840251e4500145610a5c16b3a1f0bf06beed415bc1192eff9f6ace0b07dda788649b743351b55d11c716b9c32975b6063c64ecdcc7bd794246d39838a290af842ab264b6a5cb8cd05270df29fb3c29a5a105601f89632203e932a00afcf6e6f7431a080fe59a50c0ed4a7a2b197e3aad614278ccabda830e3ab8a8eadb88820c899984a5d38d46819ae3afaf4848b4baaa8251bef888faa9d291641d0da75409685af55d37ea36a794c11983733eb9d7caeddb86498e8df91df3d28a8a870891886d2f56d1881707aa93ef1fc89c4479ddb84d82e3213fa77b3e92da49bb5b23e65b7a2240f8d7abed04ee1445b3040082f30568a7a77d840750eeb6905fb5103c568a950f89e16d9d043f23aa5f58f168d6cef560b1ffd14ea3141d0e5e67aaa7db8a039fa51328da035eb00af52779d987dca5666834dffcae094f02a717bb62655e66eb6259d4691ee9a48a3edb3aa4948b50807bb2f9222ffdaa058a3620e5c3432eae602fb8f6e6d9cee343963ff2742660ce002de24281f2b74e201e5d75268a32658b8bf973a70fef7485a75c541e9f9486049d7723c6304bbf9e996f42a2e45b44069b2d5c200c1a3e35ca5dfc923cefe458717ada47fbff6033e5484717f41703fb4ffd9fd584d0af62e86c3a78e76cf277fc683dfcf30ce1b2e72d6b7f1e39615d53f6fc21bfee7d2ca949bc618c089364ec477f39de54c68c7a5bad1f69093d987f4f6476b2722217aa25c160dfdd44ef2a119c16d79b8a6b625ff4f388240651411bbfb9dc1001239400f037a0b5600beaab2ac3a0a4ebb241fcf6218d62b9a61f6af8294895e3e7e6c8eb99d96955ce9d97260271fa8db1bddac6f191ac2271dabba49fe236a7d19ebb4bfb5c1be72e62c6d039e8640cbc93657e4aebf2e47afec65e17e252661e5ff753ddeae7f655f0b58040f92ca86f82ad942bc2e427625605a9243f74a52b6fcd42c46dd3f3ea3c66fd3747be3279e9fd60b9150a5ad0ab1d70cb56ec8c85ab3dc0c4be3b2db8b3a767bdaaccc479a5d2b6f64e107a7d9a2f3ae6583cebcf4c74b0d353f98f82e9a9e95b4620786b5eef6aa3966b407320be5369e04c6bbe143cf7a24352e60c9d95ddebb4b89ce51fe6713df0e777d93d116d20608229093c6274062b24ca705f9634389775c0cc140cbc7c6278a82ca33c39bd218611b0436bb71d3df4e90e22cda41f861d563bce46557996b9224bf7095cbfcb9a6a1bd1f2b358209627dc86f13e1814744a8964395da00fad585868a6a46fe39b51b72d41e4ff6b5fa8955931ea11a4fd72c0224303542f03f030229a0d03fff44f5b3ea187b6f91e31a9283b81d77d938149f933527bbc41d88743bfdd17b3ab3af9008ed8f95951eec1bea6b06ddde58fc71869ee7d2fd7b95ab70d23771d117c70fab7659fd15701cf3bc33f83ba8f2de364bfce0ee146c87f510ff910ec288f3dfb3c8a446a36d60ffa6437dbb8407207d6528e690489ca2509137a3f884b4b550ee6983bd0bed17d65aa092cd3c29436e193ad27e7f2dbcc4cda1bf075ab82a766ce7fc7d15462322543ea41724266ae5e0b9eeb2aae4aa332fad18a2756676b32e8dce838a392d15be73420b579d73d9a9e20fabf51bb8d33468242b8d5cd38490d114cd99d54c352eb90bfffbf79a9186f7e3da4481379abc31bdf7e740458a5bb82baf94028f1f470c6263651019229318f1129fcd346fdff2fb30a51c9226265a411a6ed8730bfdba91b04b0372df59e0954e34765f7ca9227a62cb860c087a09f5d298dbcffc62574d1cab277cefbe73ed4a3c4dbed5024debcab02a5827fe2a7da53f5e9b8746db1f370230f9811d70454bbafb15532bd6e59be0f359bc1036cc4747c45e627c42615fb69af5295d155c38da9b092dc9ac5ab0b43fc9d34676f0753e21ba063aaf541a15dc06da366b2411152f4a6347bab4825b13834a391467978184d02b3917fa0e9f79ed200209b3911af4432e4d438b3f28c6cb24734c031788c05c9f5e112c28257a2489b090338ba54e6ffb5d03ca65455c0a0fb3528d709f822a9e4730436721ffe0dccf5e51cfc8a971f35fc6f3918128551b16a3397b6912cf376812250c383e5d1c37da6153da0d5307893bda1597dc6cf0bc942d22ff68a2dc9b11829a8808721b6a915576ea7c3a11b66f2b9523d415c97eaa34077bbb308e575ced55d8ff67f5f96f9958aa729760e16722a1d5f22bf70d95884deef518e7695f0fe653443d492ffd36add3bdcd74d870238fa442bf944635794bac188e20f8e5496d6b8f1d1dabd484bff08ae6fe34ec3db02cfb8ebe9a6d1c1b7e8bf42e768cec26241f82517c910b88ac46ce0681cbf8d5514f462a4329aaee7fdb61a95ac3b55346c55f1b96d597f0ca0df0075c19b36ad73b90baffb4a423b1f15858ab3ed02ac0263454a4582e987056959884d496b283f90ca185b2abc751ebd6ed081029d0b4121eb7829dabc4db29f0f9c46020c315388fe96b0cd79f7cdc54d5a9079c06c42ed6541e1be6dda5bc76e46e1414ead009be3af56aff96f2e6e05d3b74fe9e3b799a7c99c367137f95c83ff87cc716f0b941d3329502be60086d3f8f2999453d51efef28665f0c6eb4a2e347ed7cac7ec59bf1f2240af1492532b529f85923b013cb6abfba96769de34d5315eebf397a2e70bfeb6f4c4f4d6688bf18a0942b8bf294c227bd3b037ef9c0052120f0aadbd1be0f02e7238ff90f1202bd22c083ed54dbeda675942e17d87ac9e9aeaaf3debaa7f838aeb194e7cef301fde01d76c208ac5317519f49248d951f79c8bdaa500af9e19d25cdf8f59fab7b870378e682bc9bb6a24bc4bb936f9ffdcfb4b1c746c6e621502913de1043d32a7e060717b391f3f21535cf34d7d52a0a32c7c02c47c0d3450718dadcb35eccacaabe61f89a0b719b01b71593232243b0503223699e69e098a00da2735c7ba2b859b7d448f1e68b061cdffa74c407677567e76b440f2aac748a071329ff3d729ecd16937a8f322fb924300c57be0197503097b2d77710ee56b1447d2638e58a87e059d944dbc376c36fd57fd7b452ab10064baea4257b7f14f36da39c89231541381cea16602a974556a066eaf54ef5312ddb1e096765bd7f5f75dd218f0518ac8f6824f47afe1e3b640a84ee9ef5bc515eb0584d999e7b840b908be671ab98dd46f65196f6fb14bfa522250245487b2d89125dd1ab73916b0fee6f34dd0b9d15e784b68eb225f1a12a2732b7a831c7bd7aa358571c5d9f32653025d2792243ccecf882b530df67100c9af3a181ecc657d01e01458ae52f008a56907cc83d2cf4c99d222b1088ed75282e429e005399b14b1cc0372d50cc33a7518ec22e9fb6015fbcf459bbc48754fbb9c7a8dcd108362538678619bca8e51013244d26bdb2c814fcd039d3f3cd553fab5cdce7ee197ebea3a20d57b483460c34f1efd8e555f205b9b0dca16e117adb4221b822015d63ce2c06e939998c6ddf49e566b37baa983e9eee3101eb41f0c7c0e03f5936624834715e86389c050bbd18a792f1240da2d6590755312a60602502d822d8919f60e916307b337cc0181826b8dc46a175cf63565bbdaaf76dd56ebd1156acc2ab7c6094cc4aabd3eb6b37d4d568fdb7ad51032eab7ac7afbcafbc51f6cb08f5de9ae0b03fa4d717b7475cdc8d1b4a958d81debe72cade19736e5f7f5920f8d676a5adf3eeb900a1c7b6a3779fedb314efe8250856a7eb6b28695095ca833ac963f6d041535f0ce6a200f6700fc4130aae88c71c9c8898af0ca7198c3fbf3e72cdfdffb91a1864cca8646a4dc8ddee08a174297dff34f9a3393e1cea09912892a1b617969006a27c05f483854b3588094a18c9704a4a3db3a065ae7e6877303a1bfdba2bb2b0bc49a421566a2bc45a61eec8ba2f2d97a25ddd70a2ad59459ef39735c40641cdcd0e74db868b9de48b05d5c57cd8055711fe5415e894d828233a1c954bda54b3c6866dd8b5f3f7e2af7ce92dfa1c8b9133c15f743880d09cc98f09d2589e976dece30c50d73c7cc9301f4d547c7dfaf194995ddf6b7530e76b057abfe8166de17a99847af7ca7fab81d7f9a1575d68746066e9d0b6efed5c77940a262a1ac6acb3a510eb98ab1ee9d7a061e8872e7f38f42ecd97d8d51a0266da1e46659760708bcec2e5f69d53a4b7605123cf02b6f9381d3ce69ac5628386c89f52321b7d9b17a028efa89097ef517ac7446c816dd4e8ec2b1f2ed028a74b15e3f32e05aaac84668ca79b114524a87a2d2f0d0a0df828902985212af892c0022055efd42fa21a64fe4aab31534447aed675b83cdb8d7115fac783c0c4cf022c5f29355ae25acfa3680a49dcd32fd4c8c43ee4c5464caf74482779384cac595eef1047ab6b3c6f2e8aedb0c9f5f8fea88dc68ac8aff94c99b3aee489d77b1321f41739cbd3259fded525643b35e300e0e7bde4144ff12e0ad7b4722e930d64deae3d67124c862d74a6554c5152e0baa6439b977cf002dd412b9a3a9c69a9bb683fa821733f7856ebf90b049737a5564f51288e4b00b3f929ae3db39a354a6f1de333260a4149d78b14b8fa74f411888653ea09bc42e7013294bc2b88598cfcadd6ac2644048f6ada257b9a0e9835e16f6bcde4e8aa2c52ce9cf7345be1b94888efd95c70a3727ff7d79de2cfe7cc8c1418c3ef252ca942b674d0ca2913d3864ebcd2b5dc1fb5cb624856edb895fc3e8a03894669cb83a599738576564f856d95ade5ed0125eb28279fa8a7d120e87667be84138c9df6a62d0964633fe00e4222a7614c913e10a6a1a527f423230ac6735ea94f9f96dde8c29ed3b710821dd02e5cfc63662aa03d1853422d6bb9e8a805e47d4f92673718373d3c6fb408910adb49cec021d5007e9b0337d21538c3489d6d439e8e13f7f79dd7ad09e515423f46baa9ee5909b06eb3c92e80c4e594a589d1e175b67838ea6c16849edd94b83ae0c25d59480843994e1076fa8d4ddbd99840e025be6eb7f28f3489163cbfba357e37c55e9bc14ee8d273eb94fde6c9129a103b9ba23c35775a1333fd244d9da416dd9fe87a41c962f74e01ce8ab7167206aaaad9c76afd22117c3e5b75704c5eead81f4d988779e323662f820bc858eeaae10999bbeb66089485d3fb07b3e758562a0d10314e1ab8bdcad3e8115bd12c88b1075417f76f24c4b0bd23c3b9f430c8c1ce612c9ad973fff119e7a8f98d304e11f1db995873f16dd3898bf0fdc0de6ce2f6f2ac647c988fd354a410aa0e147792f53bb58f04afa04b0b772a9fd9dafb57f31bbd25f31153eb20fffb1b4d6173cf353d6bafeb7024584d64ad641a312a72014f12f2d2157038d8d19edd6b33f03938390b959e63584c29417d22dd88aea59a4f17a6dcb5d4cb1414b7684cd6ab3083304667e8d238a69d07682dcffd4e1e44fd6c270e960f6c95b3575e447ccce1ac694a3df714969a7f86daee15bd37359e3b6791837884c93b0a0ab6e9d10111f7ce2616ec58654eeb23ad65db5cf61d3ef7480e6bf62342a191602ebb321c85ad1cd1a71327655178b55df5d06d07d92e81c96a22758c32adb5989e5b003c99aa67423ddc48e925eb59240d960c4e24dc269c22e0b9272c39021c7c641e84519c2f315ed553e33a4709850c5ae7fd18dd5c9fcf8ead12cd9fa4e350f880bf1addd372ccbacada656a3a6a22057101cfe9416fb79619cabfed39864a328f6a4fa9b89f60de7a84439093b0c814da9f638bc3c0c2157edcd7bd3f5d2d1b113600bc21fa4b7811fff751b73e1dd6f781b439187a7b3e5d77f460950d447188f4a06f08518990dbbc0e71f873809db9bc7577d75994a22004376756d0df81ade0c8de74fa4f1a07a6286ecc97622208f5ceb2a1a5e77753e83f33aaec09297f4f8a0344477c6e03c478628902995ffadcff0a0dd81caade1af56afd79ecaf4476752909c18f721c0936e804f760357f160a9e1a6e3df1bacc3a432e315f3b49b6eaed96f2e912451c0632a137fdcc3b72da61bda2850dac03923638bef6a78ba716a4a1316be2711d3ca75d6d612fc8ad14d388121c967cbf459510362cb056c6ad6337827bef8724a2d48a61d8100a778c1ddb7585d6ef3a796d4a611fe177927fc12f145313c6a0d00b1dbdc4613dcc0d78946c28b7683f2c27f8fe8d90b88d819f298ef0dc55847354e3a64063be8a7d9a5b4388bea931e5f17c84ec8bcc08517a3662023c378b0b2d07f74d06677fc2e7cb90c5a697ffd87131d6ca0bbf04619dfcbb76f052e68c682cd924e3c86ec1cb0e863854d8597f2614fcaa8a4114d0a2f08f29e92e0a808699632ab5f39466cbe099c21f4c250ec1be6d6c2782f29fc08119c5eee8c068171e2fcf8dff294f769933ceb98371bf9648550f3dd37d70aec3288f16602eaf61fdc2dab5814c8e087e96242697cab92ec75dc2a739655a6ef279988da2d265ec452abfb59c44b91d0e68355ea81c2ceed891038c1c33156e00217f42dc51fb938d5b9d4d092a205376f0ffa2b04569321ca95cb1be7d79fa6d2164b257a469b0d10cf80e3a6af57160de8c10c74c73bdacce363d87c6a59e8ac5894d62f80c34823df632953863a20d02b3d75f3fed1c318c557762149122dfe85d8878564ba92c224be912d52f4f0ecadff234583238c03789d3c787046fdf5ae090b677a38cf658c729495f91905d24727608a6d8cfc94bf5d8d6bb214218d5cbae0cae49b7808c1009d129445332b514920b73f3c6e97698e772a6b05172ca207459aa29251d44cf5cad9ae3609311fa07fbf22de898d050b937503f6debb8cf8647ca454ad45423f8e538735fc5627c5f694edf922469121758229c3527d08eaaaa30a99cf7581caff44ae038c6b8e977d72f5dc967f3b1d9750a770c29edbc81760bdf9695ac9a05cdbebb644636a5654ed09b7b145f87515dac793bb2f184308a91248057538155f13d9ca4d8aff36fb7e4dca3e41bbfa1614071689bb9d4e3e8d91cf1055008dd862f7a44b418edba32005332fa16be09e2ae5aba54d05c33d2e16b07a005365517a14dc800626d8f9cc0aa566cb3682c509ac6032aa2876737716ad395cb2fe61a9b1437b64c2d77afd899303de60daa49dc0ba6811b605d8e3b32827c723d49ae02d95ff12d85317367596044309891eba4f05c236e5580b3cfe2fd8bc81bbb85793af3820000d5ed68bbd491c9087c461322918a23e5a04bb91b8b2d801a47ab4112c00b7a1c38af066bd9a5281282f2c4205520975f4cfb61d27ce8c3e4452ee79c9fe46c1c08b07383add58403f66ad9b8bcf4b32854c38566843acf4269eaf3df0f300d947abdbb40a1ec880127876c2acdfe65428c23028cdb0b959cf4735230f0ba9b97e539168f74b883ccf6306344876ebfc2866f158b1a8dfc1e9013f086430fa113de1041ef232b80663747df0712e2b04fb5e1abc87d0665c728c1f6bfab6caa6234795c23fdcaea862d8bad584a8f0ac4af6c2f5c2ecdc7e21d0d1e7ce234977ad09ee54e64693b81645b467f76cc11e3201fc2310d768e914344891d3c8ee76ee920b7f2444f05aad94756a703a04587fc869f357c46164c8a9abece09564a60969caab83745bb27896c72a7f4784a7dff5d465867144f9e20058a0b8c26a9c58dc7d6c2d110f752a4af4aceb9f5f184cbee1049e1360812aca8d08364028d28873f44ed61af8c3c9a3356ba031f4b3fce2450460e68a01fd436fba91de6a3ac1e56518462d3bdf577814c2db192794fccb279572c720274603d208c109e2a7ecf570e9d1d9eda682514b261e537c5b5c7ab369dff2068baabf1dbe6f57eb9f06b73bde3b7078ffc60b452de92821f57411d57f79c158315e4eacc147796c811c5022a3caee91a8dcd1bbe7ac69eb4a5fd93642e76b095913638cf50dacfdaaf15d126562af7d829f6eabf6ec27883cff581a2ab80e9eedcff54bab8e2e958db4f3573a1ed71d90ff21a3470c47aefb7d8d5b0edbd2803f26326cab6679ad540a850e4bf8dc1e6922114e509d67a1afb34e854ba0cbfd6619897d60ae83da91f78e09aa5a1e564cb4bb79843ab91475976dc45592aa60250f8ed77ed01339a4c6b9ca0eebf16a3a11fb8362096e61cbfceaaad0c41649d377052af996c93e0aa6f816315d4357cd9570160eb011072c315d105c90adefc1587b149970bd422c4144c8b2434cb123818f4e92fe8c66fa0e60cfee7f2ef47b43544a507042bd85d13a70111279f4718a85810b86730561368ffe3afa12a11f6d85b436e16c6cc3630c95521e9f0aa0bdeb5df9b6bd8a852ebfb1ea6a9bbc718773bea47f596353d2de6edab75bd27764283549c59d01df8f125518cb68026d048badcba3a3e06716ecd2990b60643f78f9dd1a8ee0a0483a2e19b5b0950cf5dab9e8de5759dde91342964205dde0fc773a94756d85864640e38b7d561d758b1f6de8dc5de3f87b53ae866968e9d8bb4fc9f66dadc401bfede5bf6fd7aaa52e766f31573dab8572fc23aee0d7c5650f78b14f441af106344499d0f409a1709443884876fd3050c75f30ba35069f61f4f044ef4571687f1b58969e7b39e2468ba08131e7036668850ebcc104d761bffd308e774aac0e208df8221c614c8a26873a40698a0006a87099762342b887e136a9d80ee6e13ff2cb727327ea98470bd67700f7c32592110899a5834e0f1dc050e991964c72e5662b69e72dc2a1f7f5e56a5106735e6a790b78b90fdee6464fbe7453054f09c84ae952df8ff089f301f9b35e30dde5bb213c1165cc3b8b30ccca9bfc46a073e5f60c25e866485ef268daa35862e8af9511346a444c47cafe9f810c9ef8d4c13ddd4f258a7f2d35139ac4ca84e7b98aa0eb21256622ff9ac2fd3ae3aca01a65b321049fc9a80ec0bb193e68d1f53d104fb7732422873557f6ee7be5a736c02ff9923cbc6754fdfc7366123a3b206b91a38e0f7a378ca3ea2eab261007d5c9b6504dcc58df3b6a7c9478f3da0830027f6794808132a1827bf8c139ff10cde50439bb4fba0a142ed60b607353a3da27c88b229bcfaec6fa1d615ef0b9a181eecc5e882f7d3ef13ca2bb003b887d953a54f54c07b6142961205253473e5bd865fa61cea01a1e77f6d884c2ef78963fbca4b10ddad2e4f863461a83f713ec386d2ec6a2ba5fb996c69e8f404cd9af300b1c4f53cbc445be39414f8aa8fba2a3563c4e9dbccc0dc496a8962f3d6a35ae1f34b9cc5e79ccef7f6d2879dd0979dcb44a8a096662022b23e0537a93fd5dddfcda3c55f409f1b7db25b76d4be4664884a5f3e54fa417a919d37f2bffeb645f7c7f224f6348c45daf18dd2c95e5134c58a50958897ac6f181ece476c463bf0f35976c6bbf7742507b23c003893ccab6fd2dd4e78afebd66ff6d55beed28e13b41722542736197e8b05762e5bf49532e3f1b67886e3811de9415672a9dc55281c63b1288996878391380f23e1096082e0770cfac10f7d59201d32561e83ce4ef49b353b438031713f498bfc2b86575de41bae450fb14172a662f107f888e499b4b4418105fd463b874c2c418e42e98b2e42160faa1a233e6d1bceb00fdec9cf17e635c66b5d82620f944463dcce799e516b9116fcbfce53982783aa87198b5c316f079dec7a34171de9132a535b0001ca56a875411e9c7fb93f8e50e9ce751cc748248634bf8b69e15b3f55384b4d66480377e35a4e6ba0b9d7a073edacd1e8123a72d521bd98edc2ee49fd4dd557ab5a6c55c41265ec7b5e57681ec19cd1addad6916eff3940bc289e1a0e5a6b41f4eebf6beb9056ba873aa0ce52c9a77613011e6c2f9ed7da758627cd4cdaa7a67ad32af4f98818bc605c030c684e42bad6dd11638db9a45ef7ed8f5ab49f08552da626499dd65da92310e09d169b7a4db36dc4d7b30f8dc97d3b3dd94cbe109160a243786cbb2cd7964b83dcfc9ddd5fdd2669704ca6bdd6a4fa2db8863c735e915d0e79f1dfa0e5473481de8d170794da45908d42afd4974772f9ae19bc6f7498d6661843c345be7c3ca4fcbc267f140135e410462ff8e8901f3a569e366fb21583eceb1c3d63b097312d53e06d5d9b80fd9a343b9576d760256649e8289c38ed3f44a45476cc508c3122fffcf0054a158cbb0c83cc8147b69ccf0e405439ea76325d1d93e12244379e376f1655cb4551facf7458863c7b3295f18fbe77c66f847bb459d1f9bc80a036c2ff92bfd783a3d7ec207cedfc018db6860ae611611af1a6b1c24dff33d201122c5671446212e69cf7bc0e2aa2f9b7339175e009af155eca72ae3395a922f318a8d35f3bc58b120fcbe59121bd7dd8c3ddeff83e9f33f2a1390688b6ff37597d1f8e60d11c74a4277eaf9795f3f2cf96138866f8fd4a7cd6b77eeead77e828cc0dfb9d18c1ae697136c7e7deee88b15b0a89e494f2ccb1d8aa32962b531ab52473892458ba465ebc7842608d64fcc1b53e2dbfa7d3f2d0343db41271905e5bcd49e64125a81ed98f4d9d6da21a74622550b25d32f24be409f0f437bb3c7ba788e2c716d4cf413e05cbc47cf92f40c28bbea3fd9bd8c5621cb444b460cc5baefe3371e3a92a1f6a2ff999df836ebf91203621bc440ce5494e86199df3b4577211116e342b50f14b41cad40086f33d177487df1f13428345c65e712e63d7dd2091064abb6cdf8ca5da76a2d683713c1f4f721704eb0a07986f3fb73e120b9c44932e8de82a23fbe94434650a5a7aa763cc78367500bdc026e157f4b775ce51d1d412d7ec555bf4720f0a5d13265341135b99eb956e6ed0e9817e36c5f4ad0097b19f290995f40fee9e59adbcaacc2aaae16bb55e349292dd28877da2695655b7522377b021f7c429ba93ef57e9a5536d8cfd757bc131abe855f4acea0f20725a9c799eb8e8e0446aa7ccce91614e86cde331b06e04d489d6c18e63df48a5490a2b7cdca2765a89b79869cb885b9b949ca98b5d3b5c04058a93938427b32430d3c3fdb3621fb3c4ab8d5e83a5289423de64c05bae74a0eb5e9c4450209d810f1dc135102f93c68086eeb8073031312edc76a4fe3be24aed3bfd641c70faa0d8c7e7d3fe29141658aeba4c3541fbb97b6b114aae96c14a3692b61ac501b6bde57d4a211fd35b9a47569cf08d6a16c001412eca8fa40af437cc3ae1bd8d99ec904e2e4ba4e13c46235d850c5cff485c7f5c29066646f74788af25b0376621fdd1dbb719343c0a22651c68df70f57d2b572255c2bd755d59e2d1c64abaf35f38aa02cb73d35dd88dea21ea9a7b025d999d6d591535146a3ac723786c6d7207c9f2a65f27d5d60356f4fea78156ffce4bdf0eca0e67cc26e3f38eb2c9f86adfac3606bebbd5c34c345442ceaa96c657c17c854bbe9d14643b2d54f24903a7a7bfbdaaee5b6d61389d38d876aec60b8f63ccd533b44e7d17e7cda138b77c713b28510d3fd2aa491653bbded805445e4b423a9ca44dcec4e6180b5a47fb6605c16f6e05721c0de0d6c33df13ac9caba31cb58c41419fbafcdceb9adc35ac5950ff3f6663d20b79274197dc31f9dfa4cd0d5735c78cfc5330cf7749c8a0d9254142f47e69434e1b1859854700ec119be6b8d73c4a5849e74209805b4a893cf5240e68749df8c4b0c65d7c3a1d54f5d7d05d753b90d6c6ed5c5a920312ec789d82afd5948cab45059e83f59f9076af0bf217f5ce683e5337f7b08ee56b99112634c053996a65d62fa6844705c84ebd615601a5eeae09999aac091d55afcc64a7700070dd1ce1d01367032c5e6195ec5673e2c436220769214eeaf23e557e9baa5d2e213c1ee36c4b81d371c2f4a16c1e08c7e0f2f68d3562f54b76718956ac5e2fbe756c7bb130c8d4e062debbff3d48ad3190928b51342a59e8d06a146aeb1d354d7fa458b16fdf60ccc09287cf640de9ca122cc9f492db1ea94431073f78a76454f74335f79707e3cdde7a0d17fe07ade368f954b33488c22591854db55b9f52e42a22cbe33171aec9ded3d80881eec6677cac5a909915d735b62d6e477007f943a0234003a2b6a8688539865fba3ae270b60b19595dd272cd8f94d513b56ef46645bd4189bb04c5c8613281b5876fa3fd08e11bb7b0b24ac98b89f5e29435a2f9112842f5d59938217a5fed2dbe64024c3b3ab442ea220b75ab342bfae3d32b7f82c4038b0125b173db045e39c97f1fbbbfa30be2f2c569b0ef1881e119c02ce7db00ba6ab98b0a0ecf4da5e353e5aae3aa94433907028534fadaed02e68be161be3d993f15aa43f31d2660b3e789b94b4976156c24718a51a5f801ba81ecb7a0111bf3607742f41a050ca7ed6a2e925b3bb78cb8aa033c8d378f8d6a5e8a68e0927c8fa67b82405ea50e75e45e713045a6a4740282ffd91b5b06d21952253be9252fbf3132a033354eea1d2b1265469d8fb1963a9740f8883ef50224ee781639db62263d3e699d36b8f7dcdad7dd46be8513a396ff5cda33ea3316feed0233525bd544e5d6f82fb896e8f872a94be1012ab59991dd17503b4e8ec5fc627eee9287a2b06512a32b1efbad278dd43fc9b91830319b882edbd13313fd308ab75bee3707f3fb4f249187a9b6f69805439bad7b6c584ea1766b2ed162e14106364fcc849a38fc3dc44f397bbb2b4f3f3117e8d9fe8da17b74d809b849054c9f40deaa0f309464333665e5b80c33f113a1708543c104965963c62aa452f6313b878a8e60688a23a84ce91ab55143b8396c7f97d220d1bb73c78c7f31d8ecc9414ac78549ba3ab56459ba575cc75ef884cc2ef75ef6f792b2535e4487247cf99bc7dabbfe440e6bb7241787b4df31a8c2f2c0ae87ff13ce894af9ed88646b97aef898b3a8e8da6ce0d8c859f29f9f8a478e509139637e359f04ca915a0e88d692c927444487e8533087305527f22e823a394fd3972f615115534b346600de17f2cb944e5b3298a7eb1e4905e6758507b42ec23587785dd0d2385742ca91d40ad25ebbd580cf508b9ec59b9efbc60fc692323afbfacc025ce8a91bf0fcfc40e6d3418159d8375b3c29851b5bf3eb63eab32d7a4dd947f9766cac7c91c7a6f0eebb77e3301a82a0531eb5ff804af967a4e9dd94795695c3a5d633efe0ff89bc29278b50168371e9f629daa1da353618d25cbf5627cbead777464c6ea1e4a2ab36b92f7b33f7302b3dc82be32fcc7cd927f988777751ed8d1a54845dcac4efb1f14797b4dd83c67eb17ee3955de9432be2a3fffb32d6a6bff34a6ce06f134be894afcb9e55cb34baa8a8ce26a25a92a9922525beb4fba21d1bd8bdc06d5573e7837b17e94b5fbc6c79d6e69e50dea837cb1eec38987a5fc56daafa9a2bf791aa20ca77b84ebcae8290510a753c7f21175f9b633a29deb2d83adc908b09480459ab97cf5ed50b572634c51da9f11047b6041c62c874053a2bfbab127a19df69babfa40936e32f33a360b6b26e722d378a2576808fcda6ada814005383dd529b43a67f625da72c93d1df244126a467bc564a32ed46c17600871506da0a796a4f4d5031655242890423b9a1d5b43346c1899dc9a694fb458f48cb26b4e99483bf80e7011d4d2c9d5f31da02e2717f924af8d5cfc98b04187b6e851588d5d8269a2b741527bea17ec8ed5ace81faaa9e698679789f23ef252f72f2aed05f8e03e68f6c165beb1da704a669aa8261cc13f341854cea353a8f7510da0eb9d772d7752916e3530af2ca06955c1f52a0e290ea7020d846edad1a6aee7e5c9bc5ea4789fd8bf7692a37859410c1b91b0b9524de6b73573ad2a0621ec855e1f1c7d271da9e651e9717ab456ab790ac7ad66a7c226611644e157940dc28147d3e034086febc3f5e5eef80ad6afa290ca93912e21585b7763b9ba4e512ac4f4e4be852edba86da06a0e5f8df4f8e356741f8ee8bc9f9eff7d914e77f360ae32db63b0ed1a841da8129f5b1d463da0db83dbaee8fe2a2c0f47e56cb4d52f45084abe88b4ed66f10b471081615bed413104d530f20e3be33dd6dd95ef83ffecc652805689c2fdfe2c20294aebc5e4931daa78a9dd1b254af1b2536aed0de71530e039ce844b958bd635b7cc3c69e806940d10ee193c7abef650495d03dceded19ec65de848e6dee973effc5a41e1b64fc9d8c51913221c92655dcd027d19140e846973ad9b8aef4785481ca4b6e3c9214700ad3feae16a78ec4490de6ced267ca1f81562b4968e7e755fe3c9ed028d09fa9c832b34daaf0f76b37c3a7db8c4c2ce2feb1748a9f0f8e9e2ed90af322fa54feae61c63007a222b3b629e7ffa4cb14b5994191e6d08af409f8502ca54eb1321abe86458b8303c51b01cb144067089312cdffc1705378698f1602e4916b250e5308598938e6859c5da911b69cd0ef28ab536cae84b0d6b0457186025bf8ef9fc2a54c4feda9b6e280191463e4d3226535310ffb3964e5fb50dc8a8d8565bdca10829c62b44064874c6ec932c02d985938b6aa8963e736138f22da3d66a1d487589687d7d4d19040360aed6edbd08d56a08a21dc5deafa93b8dc5ccfa88c385279d5efd6e7f53a868d6d3e4d1f893f789363f1ea32c7edce65c9aedfe354d7b4230a92c6433a23c047229a608db65c3bd5d3dc446e64164cb2b2d038805d98a39a6694c03854b7476e7bbf8852ae1eb12a168af09fd48553861b031f97aa0d692c886f58c548b6f054862f8c96c2599e3577d54c6ca41a53f418c73a070407b2fe52dac665d04d047e5dfe9e5571aad591f83fec9c1b7dfc2a20d9fa718d9f9a9eadafc4d0f8b16e05ce695faa44c4e0e0a5c03bcd654a03bde911fe52d1d405b1108c6b7de3d618d1b5952bb3cf772857bfc9c8e69a6866731c31c4049d80f36e39175b9d71bd2ecd33e265d5f0b38087ffce3422d63ff27f3058d163c7755324d036cadf7511c8f427c2c506d82284a0c68b7b0d83c8dac0449a7efb4d5accc53ca673a4d6b29e65adda4eb5a1cc0e627385fd4ab62fbae0bcebda7fe2f9d9acbfb9330b9c563d732c0347c208340d30e4b15848f2023f2a2dca08028669483094cf9a99d6d71b2db3120c204c9454866df2f54033daa29d13bb6527b6562e3aa7c71b7818c9042fceec65c6aa8ed028f4f56e55bf082ae5d645cea2ba894679c7d4cc3b2e3ae03e421ee924c5eb2f74411ec44c35647332379aa317a826aea8b3e05b267321489c4e1c3c33240bed21f552982208ee41f1777b48ac38d7ef907a7c67be5f9680736ea2a57104cad90da67ad97edcbad8eda6e3fb19dbe36dbae009b1ec96a566afeb9c1eaba6273f231eaba8c01186c82ea2994d6126b02435fcc266f8d4fef32467d909cf97630f4fe489b9fca3fe0e9ad500ba62890a593beacfa3cf2e7ac897de542762e1702575d4739ab74cfd4c1497b434d8182651dd32fc3bd120d02be021136bd17326cd8cbd8b3f1b17ac256c7ae2fa1d2c6a62e113e356c47b99ff9f71ab80da32f3056430990777bdad5d623b8fd8b109c06b83373acd7fe3813271f13d06e53fcc751b0c4e5885c072ef814482a96830089b6290f95f0d3a8ef3caf6f1ed64532e54727f62a3c6f24d2541976a2ffe62ce112c9b276b57dc343ac0cf84c4d17a0bc21f633dcc843baa91952a575667ce5f434bc03efaae9674d62c6f266c4e736597b510aec80fc675b48c913a38287b538ca89cee146cb9027f9d5a6bb20c9f9e2d8d34da23aff6bcce9b0452c16ae54cc9c1321c5407f210797f98bf51311229859b484bc03bea332dae5e5e34bbd36e7a8475e9b1cf461e3a77a786608b17d09b197914d56886189e3ca408b49d262f2c520b3ac8bd949071ce85a5e8f467848389c883e4bb60e89799114e11ee7fc651f11c4aab8187be42d69dfe9f9b413b7a0dad4905f2aadd5e1273e58cb6b57e365ab28747db417b943d30e89e062cf29000952d56fd7720758a8b6ed52f76aceda0077f1a68d64657119732eeb4217bd5ec6eeb9d1d63d1d68b99bedce3378b1db6a53515ebdbff326e5ea7e46489caae2d2da931bd486654feec9352c50aa75a5325d89e2c4c463a9cb6f5cfbb7eed819986586c148e5107df469c97dd0565c9ed3f8763e0af391334b5289fe192a29c72522e35ea134d8d673897e8ced5908bafbf4bbbe2668690fcea9c6bbd06a838cbab05d295aa2e2ffb04ac1d81cb80b4547e7d7f47ffb34ade8f0cf98410a6b8ad8b6b73c80b438a50f4c6f7e1b69b9114957ddd39c392c666ca33326e0759d883e6e7660183aa501ba4c287f248cae290c4fb409d646fd463e3682f19d1258857ff708a5ae2252197f3edd7879356818cb5e141812f780c133896ea8413c69aaf906626fca5bf82698243f2db63b5df572bc082ed8594b702ed2f619a94d9ae537d38f468ae2dffe7ba7852fc57e2bca49860c658ac3a1806b364696a6329bc679ca6502e4ff6cf48a873555d5bada2d4e3257aed79293169dae0b0f128455dce9544e94a38daf785afaa7a5783c6823523b9f28035b1916d8ec5fb380d35f989497116e26f2d6ef8a412a60673efa3c5f5a81834afaafe48c2e6d6aa334db30f8efd59196bc84851c6d36936188241887eaf3f010cd11f6e0af5cd2efb516c7b9e3e0be2887542a62a1b2778456efb8842a001675146e7836daef3ec80af4ae83b998933334f47d5b44f7909713d23b41bd525d46d26d72f86e1f734ae7c027b716996574b8f16765865b66171686965f9e7d4d91e2d1c31380e19dbc42c1aeb5725413ab27aec8b45c7e8bc99216db84fee366dea9da98842eb04518755bb7a01d1203a85153191575aa2fcdd1bc8154715cea3a4f3992df7edebfcb1565d6672b6c83dae7d77b6a2a311a2be120768382484e3962ece159bd2bbc7a6ca44e69f1aad3a70c73e8379255551421851221dc41d49fc31abc58cf41c43409e024284b2c86b1405e1bd8817c58cd770a31987a28a7a830d5d6fca0f94a1475e77724a8e5cee8bc9a93b328cbde166dc2dea0a94b9f9c439eb731fde854dc5d53f97623c10a951f1bd8f849c1d29c972eaf96d36a5e1d1c2da1feb00929028db2952f204645d491c22bbd3aff8cb29dca9f69e1b4b40a0c606d12f664e27f22b240f19a693849c4d9838eebd6bd4ec12f185ed5991c195df2b1616b0895c4ab3a7f5c45cc5737c5f6a70342eb0d11811a244c0afd743d4c21a134d6182c36fd4b19cccf5a28c1bcb19326677862d9fec2f5406cc4b33a7aad77ce70cadd4aa74c82889e20f24461a73aa4f09f7b617e39a7b4c6210d910e62177f8a3a879dfcacf91e0dcfde087ea477b96d9438b1707c20400808191fb349105ba4f22428cab850de36bd3de7676cd46d988c0f2abcf85723f1bd7d51cbe587e3955bb183d13cae8ba860660ef124d91ea006f5b84f319f1d2de1cec41ff084c8916b46c7aae26bde58b3a7e50842d9bb0d851248b113bd9ece1643a0ca95c0d2586647fb2f886879f89952a139f887ed060bf56a81d8e20e2a17a6f505dc95c7691b08e9ccd6c04246c3f2671762036354b947045ea03500d6ff87cca6c1245159dd07ce2f370a05f5ee2d4b7e3d4110e50bd15e3f3087fdf0e002d7b89510cec4f86b289a28b4a35340825ecbdb81de29f05a0b223ff5d6399a1ccbdd634326dcde0785a8f0d8b88a3db92db4ab1bfe8bfbc8c298dd91e5624104841c8551674fcd2540d41b6efcfa51e1de928f9ec55e1d06239fdefd782a8b8d28fd36186d85d6657c5dd9cbf1167523c8876b9342e44f73ded3d1f307d5b2b06164f3b3a50cc215856b5597f3704e10b3c9da279ddd60a6863c07c7a81a740267e9d279659d2d0a8a9beeff08eec2e3a3e9e4dad3221fd65af7aa15af629442b4cd89b5f1956b066217ed71b1a0c9e23fa6e4d2ff7e7e2bdf5ec078743179f55a76cffc4b9fb010ced0a1f4ca17e3ccf001df357882f9b6bcba5e0725b07d4d23d82149b0ff5e2db83333b8dee015bae5040b5dd570f92cfda28b0ca1821d898824a91da0cbb034e47f4beb1b152fb295d6c0927a493bbd8a3472ef727f97b832cf09edb2df8bd0f1335dffd32d7075e50d29e552af08c7deb045e54bce805deb17a63b735cc77a5aad4d2dd47b1a66111f8b4bcdd5cdb798e400d5ec8972cf6e0cfe5d13155c01f135a7c1b091371f2adfc02deeca937c2f9b6476d3a07c345f7f97984ae0e1061c76d629438df92b1719269e3cb5a5e346a158a7b02ba1bf6c078a1aaa493e0dfc75f260132eefca270e8bb421691b9538c798652236b787fc39433136f8264540f32e2dffcf3d0610e6fe5ce1e17b6c66564f5622605412d90163fa6703b9004173a06ce1bf1e8fe824b7eb65fd8c242fb2948e921eb78fe758c8e3e1f847edf02b63722fa68d6431081f74ace0c6a60dfae909bff48b25b6f878afdfeed0f7d6b62d7f69518bb852fc407c1e90d70f705d1e9f1e964c5d413d1763a93d612f1a61276c57ade2788c61b0d5646b34746fbce6331d979a23869cd5039d74ffdd52ff3fc055625ac0d3f1726152fa303216bd8a85cfa478a79f83425afe8ea1c6594afbef92d3686771f5947e1d28e11d330bcf552ae8bfbe80156d10b1a61bf21dce11729a54b645fd0b5f74f6cbdc42914e5116781af44f2d10e784e8841bccae00d697ef2216b9417ddf321668577cf8e86430b4e47964ecaffd358b683afa6d6e67f22282cb7ee97ef5467c214fe067e4d6bc77acd328accce7ab2f406b323b74f6d569a941dda18c64eeea8d9ec522a3221081b688e8d4706a58b7998ab7576477a16bf173aff61dafe7d1529a8143a13a911714872107a9e6605499fb51589f9ac72183da217094569c27b41e015c7626efe4c33e5eb66f18e7a6c501d7980f383781e1f6c1558eceafa186b93b9103f2ae2bb04da37cab2e9eb869d94f9d09580a0a0e1ad8137a956f2c046ba51fc3a20ce9efe8d88cb988244a58e7829cfa205fde909a2b1406233a51228d00a76517f04870181b1d25551a706d977eda8175dad5fe17a0c445329f597778a5f6778e19af9d178f6d3836a35e74d10ef6f7c16f0e5d3c0eed65abf5a901d30449f2b25fa73053a8234a127a128ee5d3fdf884f09c7b9160c4f0b385fde49e23222e71bf12eda2cac730297da7b564f864c27eec9d2638ed220cecb997ae2d4e5f05ec4e5b7b2cce7242d78aef59888513961f0a0338c89431b797903b71e4c77fa00ef067fbe64eaefa1a64823d13fa0a57d367c72f1f0d3eaaf885e0363c901f5e5be28b68b2e7ff9e24c06b12cf56bab385364bece7136576034a1eb96df5aaa259ffa827d9e068d14050f2f6d9dab436c6bc8426b820b0e6229333f4c9501d9b9c0485a1f866ca75bc898b89e174d25b61d45a89f50d8ed72c01d6406ff5054d6bbc71180a31fffee9684f5d84ea982ac7b3bbade2ed988cb8be81ecc92f13aeb599ffabc901790fc5d554fef09ec58e8fd15b5fa5ba335e80baa24ac2a41fe600969e8870454c2b0b0c9560c7f494b229cee5495ae5b205b2a2fac88155c19b9a77f3c3a20dc08db76e876cea8d11ada7ba375a2aebafc17e1046717e25ba2617d665ff03a51203b9c3571ecf0fdcc54f307005c0c1c7bb7b7373eebd3a60ddaf7a76d329f28f53d74dcd0b16bbab305ccb102a34ee005c20fba1ac7da77831f33f07f12becd0c5764b67ce07574e2c7eb53f45729d467ca3aab74e213cd2b65a4b175e9a275b10d1aaccc636b787386173c381bf4b761b5b01327f5070879c05a50841c5f37e767c7e3038b590b2d23a8837db5568c1979e9ac7ffcf71da46088549af1c9d59a177011ce339d60efed7cc0413e1b0f139d55db44b83adfc5e2a74e5dbe0ccd3f156f631352da966a8fbace58af4ad5e3751189c668ac3df203d98c57df6913b8c971a86e044b86af1549bdf806aaf936b9cdb0e383b4e8ae9aae1e0c31c4e916aea27d4f1a17595a54e456793220a0c3f26da89f4155be09ebf233de3c9e58200a0288642b9214a6e6e02ac70bdc240427d206824fabce5c7719d1bca07524227a0ab70b1f7a44765f8fcb5bf783da8ffebf043ac5a5efdc362fa9546aca64517b062a7787128b380fd28d2ca06999abd9deb4b18cf3726cfe50be72578b00aacfb883ebccc77ca66d3d398eb7d86f8d8ca5bfd6a322929f393557ea1d019f9ad65f1047c300539df364f29724d33413fc49d85c1cdb957f26f36cffb9c33c54108569335e1cfd6139009d89bd995e87daf9c414c97f036a3b31a9f24fbf0730bfa184b9c2c94386d2b577afab3179ff13a0ac84ebf20b2ea640a198fb13b8f3f50b4c98f881a5cc28d83add3f1c5d41c03f6abd66692e9d00ad76d85d8685726d61026a39a18d99b84cd767a28e004e32d2b160116244fdff9da96c73e068e113f10c1e2144ce47fbe4435173561c109b847306e416514b543597c6544ad141acebda9e19a4e5de105fa846d8c741c18f98e012fa7597e0e2f43c789b4d9c562d6b54ca991855b9bb2728be17385724f295204c734815186912150baf916de3d1a02f95fad742cde302174be353abdfb0fca24a0e88eafb1e1cc8b17afb41333c4ffd4e50911109169e0470712c211448ed315c632be73c1a367bd8026ef538f7d6afef0af57a722a4ace05056db82202a2b8bc1cd89f4663519644d0ece7febc55728f34685f2a35de04c11cb8d9dffbac57befe9810450a5b504bf8f3f4642f8399bf8c19535fac4b22eef23fb1fbe969973a09a2faff937c35a55cd99e21201fe4530f7bcb1a088a2d449230e8c00fb5f23643a2ebc372abb969c82b00058097849222a8162a4fb03687df9fcb59d1058a4b9b87dd5d8667830ce4831cf43d3b40746f7a61064c3d932e2f083526ce8b5ec800bf929353145ac2b1c2c28332394b9e34e1db0cdd154471758cf30e69c4abc0ef190ee0748d6ce173974714a9a6d9eebb8efd5da0a148b6f724aab7154a93b675fdd9e42c0bd66ecba78b29609749c91d527c6435a3fedf5e3b7f6b3d0c62f4e97a2786d275fba90cc7c7ba41dd562c5cab6494ecdb2b0db4f39c402c21d14b269cf4ee49e8af102cb703b6b3a15ea7c75bcbe6b98176285eea8469a265552b98853608e7bd5b5aa4bc2cd93a9665ac11e3f1d30e19d319baafa70fbcff24efd0ff23dcf64333d86c2eeb3c9254551d92e2cc32aca082b7fe31a02ce8c70dfad0b148baa26f4be20484840fcd29752d56ff78598da7f2360f46b357829db9d46fbb7bb798d7b5de26d5a131f1629b4ce48f136f570857b811e75984af491fe08bbd15743e8b41f3f8eeac89417cc449008bb9c35837da9773060987e385cd1b9d4efa908fbbea753b0fdeef8e89628216278658ec19975b68c0846fdeb1471558ad7b00bcbe00b62dd50f3231fda78b4e829654671ae1625fcc227984a5e335bcaf34b5cffb60fa40713012297dc5611b6364cc7944ea5d5be617cfdbf19af035384316f4b191827423d0136fcedbd8532f93412b841d7761d301f9193387d1e0678050af170d53ff83c779836625a3509fe8396b9a5513945f09b9a6857ec172db106fe5a7daa71a97ed64f15cfa7dcdf04fcee7046ba0cb89690e16b74f385ac5fce991ee5c97a581fd148ed09425d660975e2699448126fdc64112654e0f95f1a4e78b491245ac0a9a1799c09ffd914d0f06408b36ea9fbbc9c1e165f622d204f234a4bfe3b13688252e3210c52668695c1daffce970e45b377b28612ab9a7ad651a76cba2b17e0b615e9593bc7619b83e4a21b4dbd15ddea7b62a722ef4d54e269b72d392d8c2e622dd33b3afeda8f85ec79e9bc5950b2c2ebd46ee96339904775f52d05e726ea77504b330194382b12a0732dcc7fdd7d0009162ef975ade320987170b79fec1c8ca4ab547322e013bfd9d307b23e75993a197eaaa6d9d14be4b08ce14b7741b060f1c94675cef0ce6db9bc1d04dd7c4ed51536ef3736ed60d9c337d6f71cfc4337efc152561be57d25c217bf8d8462bb8d64169b377f6d476a300d6b5aa8f41153bcf622091219d372b6f0d9978af2e8417e0fed9f5d7abccc01b5978229a09e3386543fd5bb7f702db846a807a6d7d3bb97ef9f03fb5f8d76cf8b49aa361c54d121c808a98f27b66462157edc13324576bf60f081c88c220cf0b1e2a25edac35dddca5244ff94268cdfa22b0622f6a8db9cbbd25da28a2568c7b60afc159412c71fb8b21f595922ffbe5032b4f9dbcfddc3dd0ea7e9855b390c89016783b37709fe2d735d046a7e10d0d8cab707af6f211a273b915da84b2522fb4653acdd4bfab0cd5dd2894ccc1a4a0dbbd1e9b8d8f6f9d9680436790db096f87e270712090be728c096995ca534b8c0d27ec6c904e08807b45145c5f02804c724511290c332a74dd29b95653fd806c4c5a68b40e20b8806dff07f07c36d5fd7e36e970a6755a750843a2cd8072d906b36c20436de92ea061e5dedc0ca761bdffefa0efcef01b7a8ae500da58db721c0c85f840c8a5b3e78445696aa09118226db0f45506683f3bb0c4c334c03a8ee1c3b34b789616168b6ec28bbdbd3c6ac7c2e11e90d0670dbda612d49e02e5c338f0f06d9e47e9643c96edf2090e4a445d49274d0a1ef6c09e401b4a3add7bcecb1d887f12bf3ed35ebc5cc7e4da35af28b8254e34044f004cdf722f07f0d31e95d71be91558d1268fc5a748619669a42fe0c13d1661c31788ed33c3c5d816add9265ed832b82a9346dd766316fe137432b0b228a71a3676beebcff679b476495d2a7749a5add4526a2d833bcc3cdc4a4d850d078640470c0c1370167c89790c0f7eb3e486231acc424572168385cbf73947e3a1ecf5ed60b82800df37051dfc7cb3883d589bf4c427a13222e4632347c3a92af07b4f42e420b39bef7f88f967b18bd5c98de182be0dc60d1ded0afeb7e779104177c93fb2a4932c685fc00635240015e6a57ad692f2db5364cf9326f6b2c285961b31972194bc1e191b8fdcddee47207d90e42634322b10ceb963f90599f28cc9e6260a95a74d11f924b02b4faf483d82755ba454667143cc4559fa4633854f69105f5b6f1511c1390ebfa741037c9eb0395e1c10aec4c4ef8d61cb251ebb311683f27ed4b442194ba0ce9ac9abf0be53f82bd946d8bb528c1409003fb8f48e6f291c1b0e4cf6d4f5af246703876aa768a0edd157af5d35c9337090806c15d9593c6196f40d602e760356e5533f6100eacbdea5bcdc01e7d9c3547bb20fa24240192265cca48a0c2bb9d064e1f97a649b545da6a8f0468714b2dd23de73696669bd60715c029115968937bd79bf3ebc5a38e649a6482477331bdb85828c454f60302535ec633d5243845f7ff4b724a94057ff02b2648fb82107ca1ab31a51d1891fb25069468446a93d95cdfdef61f5607dac7022b52cbca3d305de8d6de8cc6609a25a68cb33b38a7b2cba849334dc669cdfcc65f302f156580c2ec60f77c20971a82a8a95cdd5d6f01b6e9b9708e4ab0a9f8caa0794a14231e9437e8befa35277e11fa5f9fcbb6f69376af841724fa037748e200fc7f73ad39e3ffe0db7cfdf3645cc5c817ffd2d5ac6ecf9caaf9082d558e6553f3f8eca28875249bb92636d1a5790a11924dd45424e2dab13fb3aa5424e434f15a6c0e26e5b56d5faa0bbad5a49709ab6711497e785f2797bdcfc1f041d20f0ab09a86070e19c936479a3b1cee4de56d15f3365b580082f8427c10bb0d03321295264d37bd949137651eaa69d60ac46fddc03c4dd1fa33850a996aca77df1cdf1e2686002f07e39ed8d234d52250e5b9ae602f7a5b0e028afdf4721f4051372de1503215a93691dd984e9569c99d403311424d1c7c61bcfcdc11334b933f3e132fed4acf37667a077f09257559857fcc3d12b92e06e2ba65964c24473d23e4edb4f662052248656c6be5c63493273c77803f242d0ce4c669a3b2dedfc13f38a1a78653ee8c095bd3d36b6f30e7c719149e7ff79489e1c9bae4b623152b9fb6a6a0fc44036004abfbf6a9d8b6b306f4ffdd899b8a50ba88c9c4bf61ee58824df0bf193145b828d2db5d0450f309c543b51be9a1c474d77e85c33f5eed65a7ee8f95c87c6c38ae5557dde00685619762d30574715d0bb5d71822cffcab2e133b58353fdc83f678c09b5fb8166c3d82d1eb1da809d0929fb7a93de8cd15c74359ee73bfc2375b2d790499ea8ca5ca9523a1a404c67ed70812c23a1943c4ad3888ab36a87f1b6ed9a5e97312495598117766d84bcc430debccd1dfa50987904a50145f9b00b1131b086af33e598778030c54520f6a9be19b81ec838586d5108ab3fc108f132cabea6b5399a41a77c8973399848b5d38ba01c7b24aa374585ce9cb23b292266a41d4156310520d276e4650d34d3be5e66cdc5ffa5dccdcf12619a972e0d6986c2896311831b7650639e48954f2670dcc92e0368ed9b597c98c42943a01d252df72ff74a36bcfd928e37f6125da8d63cbfca1683e01ff8076a544092e80f84c7834b996554f04f638f953b4d9e6596e702748e54ec1197b92cc4f196a78a2869a53757f714f9ad595f6da99cc5f87ada3e51ecd0a3fc88ed068a8764dff1bff85e116b7d17095eac668216c8b18315e5e99590f65ba1714689ea5660f7dbf078bc29407c9fb0a4025a773c03c22f68ff234bb58490e3e2b17a6baed5bf40c3be22cafbc5220e95485cdaf266a34b5824e727c810daee26d950e7acf02a9bdd779d0e5352cc93f554f9c4c5c592d0cbfb6abb4e62ce00a0cb27bc91c9da4bd1ee83cb2d29408ebf144b8a08bc5b7245dfb21aaa9faafd20339846d3acd266b531a6488a26157e8f82e812b340bf297e6b46ceb05a7bb124c706819a530c0b7c177f90c5172c53100ebe5d30eb491c7db8c928396e7109fe36fe550abbc4994382f38f22b8cd3e48eb306b53777d623e86cb7b52456b09f842c721cb95625b8e335db2b10e222dee65a9e17bc7582c1af072b33eebe8bb85ee265f3bc441d0168e1f7231766e57f6a6d1c92a0d781f57289739353cf654413c63a295daf5ee45c5c05210a37ab443be03294d63227d67749886ab0499207de344e7c1551bdfa4eb9454a9ed2bbaccc95e8cff22923cb00b8297eb5b8dbcfb1570fd8b8c21c5f2b0ecba367a241b55d2745fbbab134f7a49616c27ba73f49aeb09a1aa469cdf9c31a4971020d1cf41cfa0589067e750d822965ac4c131b628ec6b71a0cdc2a4c49bdc7bead7fd04c9a0a829bd9ed6667a1ae77ca7eff79bb6e1290ae326ea47c83fbf919736c495145ee1024a1ff535330f7321242f9a6b4281ccd2c6001eaee68f3602e34942cc26da1b76fa9b88b56be68bad8e6cd656e899aa6d513b56c2d2106210f3c2f6baf3215eac578a20af3db2d3b9d2579f238f3efb37d51308f69affb6d949c3ba44a63e12945095badfc6212be0d9996ce0a27e13adc14dd83f4d9ab64820b2ef961a75450bc76390980cd8430ff54a3adf01744307979942894f8d36f2450962bed61982145ebb0e24aa23d3b1594a1da0e1508f19bec3aee440f22a4d3267d8e0b17c7298cd030c2a83d132e8c18c0da96ab5540563788b0adfb2da254ebfea75435c33285f130f2cf134b5b78db06e2bf6daaa349932ca235764e620017e0bfdff2353d1951422cd433cc3e64d9e55b506e1e89881231c3b7e246cd41e3de80d82e21e89f103d9aa887e7f48ce5bfa91cd634a2938e34321575fb9721a9f7a7722b82b0fe28103b76edca769add49603d20af38512555d0197291b60743a60df0dada5aabdb16ef54150fb48e8d6ab48d8e5f9507a0494dd2e96d2dd47446fc9dd630cb46e016cdd277a9cf40e029d7ceaf98014905530eaa14bc183915ef3c917dc313b4795b75bd271506e60b0a8ca755336a657af0dd0d3b785f76bf08e0eee70cb51e3a1109e8bfdfab2ca69f1130dae1174e1895d136b5676fe6f21cd52b9268e3d3c0354257f717d3f6d4980aab4fd2fba15bd29370b85f76fb115ceaa09caed08090f074c77220bee2a6623ce54e433c8864bbb02b7140ba34bca1f0a075c05b49d983e0efa7185f07e185d75f4614cbcce8821432e35592cb7fd47fb24837361739c9fc7b3e2526769b194b493efabb3ecfa8fcae58de8f9ca3a2829555a9fb1253276fab030802364ab9bc98c1c29db6454810f8146d6ccc35d9f6bd5c7c8a449bac5793bbc1df6691b456eba54cd7c9e212a0e78010d9241995dacdb39c07b7f2bef3b5f3ce7eca24339abc9e83b51ad8984f37ba6f1d216b39d416354762b041d92ad6cad54c231a4beeea34740bdae1407d3a0a9a33bccdaf66ae62815d99b7cf9f63262816ed123ec12d5508d3206cf8632979cda3b7315eccc5084d4f07e4a240c28afd9842c8e3f820764d89d2023e22c16e79e96747681ab1df949c1dace01ca0d06e5f4143eb307eab19b39939d77012a406de288b4182be475bddbab5500e04609a3596b0835b2c414c863796618e58bba97c22160b2279c97ffce787bb45ba7bf55ea1d5f41a78f2455e5bb24efa8a0ec3c255f19dec08bd2ffc6bfb84c3166cc91dad026bfbe933ad20ac1d17007ae7dae41e922804fdf96690286691ce35ee8e69369494837e7bb6d8db01aea5d5f968294831fdbe4be5d50694432773cfb10c934453f25ae2bfacdeefddde54309f41d461d3aa646d30dbf79afdd00767fd4b5f299ff4a0a414962c80f6318a637c6f151ad7569415e6ed5616214146e0a33f296c9b9a7552d578864c703652b4d15ca07e30c999e8287bf553a6813adfdf1e4f0aaf543b31a460b6fa2cdf7c7f756fcd7403482215447bdfa2ade7762178625c8a3f9d1e2ace173f9094e64b7844a503e5c52ba13f7dc58e0942f6ae7cd4b019633a934229842dd1ce2be85b9c86b6fd98df1a60781cb64f8ead73995d4ae10fde01a2da44462644e2c1f9d6a13547fc270324cf39fa8fbcb0252eb471556e2da90db5469693e03c02a6e151d28e9c9317527c5ec46b6501a4d473f7e51e16b38c68e25d36361e45215704a328de9c0655bc651ce63088f1fe0d45f39e10a255976572305815176b0e9d67bd745c76d8b664ebcc6e63c799901ad6b5981eba909c7536a42d3b034dc17e0259d103513607e6fbd3409e81248204068c8381968b566cb4ec6ef35111687f3ea0951c76aee08ec20b91356841faa7d8a3235a6439600059d12111fba7de209d5a47af596df76d982b5b0fe4694e5d4528d08d1b11dca3799590db9eff1201408ab197c516fb3e28d4bf29cb1d9135bfe8a993517e947239ff022fb891c7b4baffa87ed2e4e3f22797b7d001bf610e064d4aa096cb9cfb0a04fe473c2b88945b0820475507198b4538587f6d851a1faca539bea37c163ed3b69135e05ec4814b0cedc89ed898babf6498ac70cf0cd0204f23ebd2a74e05ca81d870adae37457804565ce15561ff6d59672b37a98888a9019410f3709cbb74b9ca2a4773e2133728b78af54f9dbf34082924232267f2b9a3d0c19d52e7de6e442946a2b541b2cc555f33717f70d0b8fc47b1f8f241aee4c05de87d18c612b12c862ecc510f6c0f51ece9c2c7f328f468ff55b1cc52dd7e1cea4c592e97304011c7aa8c5aad1748b33010ccac8a2518947e43b5bda284bb4b9e577401b5bf82658e5e32dbc8dd3ac418a1c49941bed7caa32893ebf272d91fe09cc105dc0e2389e68264c8e20e3b6a9e31fc49f44e268deac0f6b23c265a6a0982e22c9c27cf1749a3030e472e8cbe1dd06f7e12de2be3134ae56eb6b05218d44c96bc12d55bc19db8e9f59975784cf13ab0aa075af63eb777206d671573ac568f56962d881f2d58e10bd004ddaf591bc4fcce8fc65ce8de27aca24f686547e97e58edf488e14f989f882027559aebf144933994e5c346dfc7fbd66b58dfd06c3f366920508b95e1f15e29f06bb0c0fec4bbd198316fe008f97638bb1411acca16c05611b9d62ab6ba0772ff5ca6f07f167a5729a68dc2cc04d8e6e1f5a98230c19a6fb17e1133766551e7ba158180f52ccb82c7c6713c69a79a46706e705980cd6a2813388ea814d0bf1badda00a2accbb73ea6e21529b2bf27d85da6130ef86ef82d61127c9137c9095314c2de202635d30e0524c50b37504770601322a1d43a9bb53fa446f9e703a9fcd246dbf49d3eff0d43f18ecec5d8e892d16390a048fc091f1d24f55a53ecc38ba514b780d1f20c28997edb0161edc73d19dccb0aaed1c1f5d9ece10c095a868234266c56ae08ef39d100f1680a53b6a07ecc8f3b54e48701cba991e567afa41565d29f72d4d9347ac35702ccaf3ece575604e040e5948e5213c7a2c04ac8301078e2066724c82ddb4ed261b8ee1b1c0da1af2be3b8703492cd5318e97cb90f78584e1fe895f5c41a0100cf7caa6e96267fb5f07f5dea1006295cc76935455d90571fc60f5d716b97b33cd1b57fb9f9df5252a485891751c267e4da6116f16a6128c6717d822c9ea28b1c824050405788c316a81acb8c69bf40f1d54260645c791c1be3a7679cb6bb9ca3a11be6d36298529b2e261085b06cc72bdcdfbc14029c6f811eaf5e7e884fcf9431d47afd826214484729e38468b46ead8e7d71f2fee410e7f3aee11e246c4b04f1b64fb929dca09920bbd37b767ae44835df6e573ea921975c4a57165f22457b119d7d652f7a3d6fa72e28d143272b61977c3a844a9d040515ac11e86c2681c05e924c79a1a85e3dbe59e7590b7d4a88c355aee44900e7eeaa7c772a717cedc8f6336c541fc0bacad3ee5344be1ca7e90033e70ceec21547f14396cf40421d22aca7e44e702810ef1b6ab2f74aafb8dd2926c7a5de823f2c99d4bfb3f217fd90730c0d37fd5c23ab84f93fcce6b0c2fa92dac3e2e72f82bb6da83082eac780dfee380476dc5cff603adfe4181e2bb874f228ee211fe8834cd06577ca17da7e7f0363fcc7de5b0383413113dd092bbc8750e89c52ae454a002f968078fdcc190d3b684b15857af7b0f1238f570ce7883eb40ea5ac4edaee508f17226e349961928f011122ec2a69d878b2fee4ae653703363fd05c88a37844deb0b5523f741cac3cc28d37c01ed2ef63a39b2fb9106d7723e7f7cb4c1e4b855e3063f752178dc46a55e29548a9a3dc752e33e29f12f7578ea30fe83066c6112e5044758729593b7a7897cf7ec19f16b2f3aadc7baf120bca945958645d59b03cae8a0343d9b6cdeb20ffabd90a345e8f72ad57282042e9ac9a3e0114c8051e5c164e57aba2d2dda4518a2692d952925bc7196e9f5380f857c0fb2dddb405145def9ad618f56c6aec42051f1b8b0208225a539801021cb1397f0a2c7914780e008b8a93b9d4588e2fcfbd213832abc78e233c463a455e0ec0bd88002dc7674ee39212c4077aa163a9270fd8f9b4b3ef9cee861d67a855febdeaa1f1595e4e04834a6164c4e011e1b723d3b1b83ac2da13fad7f2ecb9688740bb540b413c06926b5d5524dee9ac2ccd9e510c777e7355e58219fd1482f495f7113d3734151fcc604ea3a16a8d106938925ad5acd03b5fc646903726239f76beb7799334e76e08bdc36612ca8d9281ed8d3a35c995e7ee86352f859cc27df2378c5d22bfb2e95bce8c44062d79ffea454006d971b65aea9f7300d177dc768bdf26dab4cdd07607e8ac639caeab5b894b2d9019d98dc1adf130a539ed8a4d0bb190acb38947cd4c142f7fba9997dfd9ba7b8e0c06868bec4ac41134584cdeb21e446665915c28f41739115b1f3f8432b3abc560a3b648a0b7266778ed151a6431379533334289bbbdfed69a6c31e5998f17b5105b13fe23f76c3cbed48fa415496b6d6028b5fc26d882352726ba95252893cb9d58642140f94bf92b8a2eb5d59ea6bcd3149a5a005c99512a248540314e837ca646e8c5e76f2ec08c33f63e5689bfa16e8064ff9831383ae311acc5534dfb4d2d76c4611586cf854914c9cace1a421f16938314b701d8588eb30f3a0a161a591810fc924da6f43e532d3c3635a5fd2383e050083051fc73e65dd2ef7beee0ed49d5024ca928e2c4cecbe30299325a54685d8e47a3baec8a6addbfe4b160422c161ad7d405b7009dbf449faa9f8a352facd628261ce8293459465ed5dacb88f63679e78864601049a0301a38f755fcaa4c77d91408edb57421fb61527e1fa00e41de08e570d588d46a965c464408ecd91eda3c4d6923acfe446bce4637bdd366c51ead250f3e2b29598811ccddec1e96588f67ea99bfe5f634ae1217b383976249ed22abbe03da941ac3e5ba852b04e538cb7e0870d87ba473d3b87ba0c4a8ae1b9178206f5d41dd29fbccb6f45b391b23f6b7f4df88b68cafc442309e7a90e601292b043d502bd416668acfdb312ce270c81bf081a2606bc78171ee7383b332dab786aed195d507ff15f730f1fa386cd22767ab4cf7c845db54cdfd47264e32c9b0eebe474ee4d4407b3190ea0e08f7e99b673d7b556453401d0f1e4b46045a70bc5109a1e06a3850f81813976ee8651282fcc0d8e5b194c4638c9e0890d0753282f3e2c7fe6408f6989c61739cd91951466df0e8d85c6f582f21ce29821a1d7aca54c1af1fc87189745ef8e7cfb12386754709f32af012e723afa5cfbadc96bb49301a673c2a89d64d7f1de95a1339aa6814e6a1e490f2c5c285c173a0a477c7b3775a02541af5b96a743de29d30da918400d6e8fe28470ba75a8ee79f2b8cbc77326e11e33a69cf1d4c5ba4edf300fd134a67c6ea2bc09f7738db6afdf21a6b65787c2f57a7976a065f1258d2def8cfbe072fa3929dcab2d86fed9488924e5b623db53cb9a37bb60a1a0107d3d96b5835db25b492cda511396add354efabed04e1429df5c280c34030e4e873ce423914901660e1d9db5f9a001b4fcd33d3e70200179eee6410759dd6cf7c7b54edff22541bc1d583e50aabb579b181ec9ce4d7bf234547671d3811f9af5bf285948f031c9484233f6a2cec66e81e5731a51354c5cf96a013b26663037b5f397f08383127bdf01a011cfde0ebac85ee81142aafa5315ba5cfd2ec4f18d0d046f8c3be3b957656b4a632efa6dc08658d993754dcc69f14a6718106f8ff227fa09a5dd69613cd4595cfc1819486d61acfcb1acbc8f5f9fa6ec3db957a867f37982baae9bc57f76e08df81c34f7b0051d5886c2b39c4d88352c94eeac47437bb68da3d7c58161517aa32cef58368a284289347e97195da8febb4c107f842874926d09b0c88baed7646d08d3413810368385a53acc797f1b33858db6c2a677b5fe4e6365f1bb62b34776a1cb5afce3b61d36eea40fd94d90a671a9b144bee38b79102a4f50e367a3727c4a168be3e449f7c55d84253dcfd0185368c4dedbdb342d585bb1339edc29d7e195b576faf29ad52a92f9437df5432d31753f65b10d5f6ec848bc4902e467aa52b110b10a2ebf56f66b1a360015b4a903bfffc514868904ac2c0a36bfcac71b4f5d1c3133ae0b897c62d6a58d86a268232019887123ef9876adf7d1b4a094dcd609ccc311b6907b083ab60ac25942f8fbae8d7d39e2793da0dfa34403f61ce15a2b3094439106464dead857398b8b42483964208a273e4c759b209a03423dd4813897abb7ad2cd93c81637586d234ba702dce23bc25df32dbe0ddf6b5ecde87945317a04214634ed75726d43371b2bdcec4ba2777c3422cdde6487ee69a00dd8524136fb38f562faa7ec096f1e4d78dd0c98ffb0b9256136affb22c710d8c0aa303c32e4dba4c63e239c51f32864cee7ec6df338a68372fdf01fc6cb70f165eb58ab2a65e5d0d304eaeb106ae26fb8e17b8e349497fbf5eea1f577cf79a2a3251579b0b507ae2de62392d62098717da35648b14874b04212269880093ae1b518bba7a4e31a36317ef85896fbdf705f354dc065b378510483f9d0a3cc64a215a95ff253bc1a73df871bd7c5019995df903b99e697512146b0dd0a83e4a930656d3c833bbff303491ae31c9d274a2ca08ffcf591ab350fff400f5bc7bbe51cb6cf48ce17ba8b0fb55c0b026834a980112acf4b97fabdddb6e9134e3aad064ec5e310ab17b4406e3db889e6f62993e35ed0c0436435b3523ab7b64de0af84fad06a2544ed9e4cec694027c9db1942d97c0109262d19deba8beb19e69305465430fb2d25ac0dbf32c2ca0b3935eb2fb782c0837a9b4a9e0c227738755e4cc1b04f2fc7a007a711176ecc7c0da0c73adeae4755c5c6fb4f4d03b96ae33025ae8f98e99ea93dca5e966c1b257a90850c47fc842c42dd8e14c68e16bc7168ecc7c3b307c1e1ce5d78e807f04c013fcb89c9e5f0a1522e0eb8e93f62b7d000b60b66813ccf80969c7b9bc82317c30c240482d26e4210ef90239fd2397dceba3d14d234100c630f9bcd78d63c6d4aadef7ddbfb73ef79dce77b12c54fe1fdc9273947a4d04dbf683d6d95e08dd4749c3b3edc4751008bdc6407e2c531ac36d37143b41b2aebb510884189601fa9565b46944c3d49cbc0907223a2fdf3a1667bf4ebb1405d66e294bcb16a3aa1ab198a1b4abe9b1975447a42327e617c6b602ff2bc8b6fbeb8e955eebebba1756af480763d80f15b694840d7f63e5268c5a11f1f7c4f9f61e8f8f7182cbf0c0204a750081b1663935d771457d41ef79d370e0eb054fef065ef5fa24797a38d3695bd2f778abe4adcca5ff53696ae6eb4321dd0375e72b04dca84fdf9694f7a30c119c57e213c25cb487c7b7055f4ad83b96da988845f6cc44cae1d197f7d4cf5dc89498cf0be3c9db0dfab54cce8afb6a0f279666dc4152e803fcd4142ce39e33c66ce33c821a4ff17804807f60dce88dc65bb0986a323e48a05e9c7f69f75306b7d786a39969e67b708fe645c7d88d4c39e0f12957009773c757fbdebc8c93be8a9f81e047866d3f40f3357e95e4f44e75ae2ad2bc31bfbed1e8a91e9f403683333acc0e5d83218dde69be97b9a8eaae315b569fb62581964e83d5abccecf620ade7e431674f7e0ff23f018c6ed6bd4386cd3fc6b08d63db1104d2a6bde2e2fb993e377b971492bf8438dbdb620f54a5bacf64883d03bf53dfa6625ae4aad19c8c58500a600e1510e844e511e33bca0ed4e61a08c4fb704b94a7c165b86ea669c72855a68c9dd5f2b7c7e4a7edf1c6191c974cb36ad2a036ff70a16595c2cbd680227e65075799db43059c7ccf14155d9a454ebf4da34411f6dfc5f753a559bf733662c8a8cd41a87edb7ab493af050e9c10c123e99d60794958d004dd3bb4370b833c4cd6830032808da6c40541bcfffb08cc500ec51e4d029f5b767fb254f3c206bf7617f7e994160c246aacd446533c7213bfc43379ae8294ff74456cd41e1cf90cedf05441a36e824e7e79677a172cca74c4338911e9c2b0c8100f4fd5df004e47052501ae5d760cd58158c7a5e4ada3b41d8e113c002a9e6ac86ad227cbc78c4d37f8d99b351e4e656763750ec49e83a43f0ce663c8acfbc2f772347eb51d7d4d79291ff6785fc75fba75c04930abd0deed07dd2dc70c93ae5090ab2d5b7e880532170515f780ca4f51a32d71bb2795bcf14ab94f2e491fc17943a1d14cec544bbe147bf2c89079870255d941b52d5d3fc0d219a38b07a8c5be0caecf59a4ff444b8a7960cd84a2021440618cce93457a1de11fc5d993a008e7e54f2ad5768aeae663ba125f0b67f11e57029cca14a55074c5d89046609c7fd992979991516621e5af9c0ac448fa01ec58dcc643af405b6c3990dd7b95c6728cdd700fbc3ccd85c7fa43f93b7954b2189397dbce42a6555e6acb9df207018c1caa86800ae701e4e1ef455a50234c189a64a9e6f8bdf49ae52c8d099984cdcefeddb983c751ad3b920de658da82b57007d81b8ca0d93bcdc11f2155b0e88d9086127e7b62aa018aea3f11fdd1a10bfa66b8653df133522c3a446262508f27f67220d5631bce1414c17f6b5d4d6f8f41af924a3fe18e18ba6c1b9f2935418dd4800e8d8153123476accf4300d00c9aa4044500d1cde9747ea1515cf19f0bbbbc0aa4b1d2b2adc2ae867e9570dac408f89c7f99e4681ce5de027b616981a146d1556fb8e22105f9dfd5f36fe40a4da8b71d0cf40219c793ce18c6743679f9f509afebcb993c4aa49c7d01a5183a2936df09a4b3e07b7f9ea174da3e39aaf96fdd86e90fad13592ae6de0ece67573cb75f5e53755265c86eee8400c275625e8bbc99a782473fdc0494c8b07a3c9c5d85d130569a8c026edc39df51bca48fe270367088f18f0429a71f87b81661a5940331f351cf653e754cd90990eea3ad16d66a0e99c5d0c7e3499fca661011f8ecf8dd330837da414bad54b3b928d2050eddb8f85127659db93ab67543cfe1edbab3100b57c20f4ec338957e0b6d6b2f658454d0c9e25f1dc37185de18fff6279231dce5061f823e4e7bbfdc4de85b4d781385eb03a7ce3170e54e753873bf521445eacd8f18a625bd3b198b2afaccd90addf61c1f112858cf9f604fae620e2b2a94bddcd18eb3eaa8735b957025f92fd71eb74115b56a1010367848e00ff48a6e970e8c26f098ca5290693276473b7f013a0334bc46512b7ab6f4ee3230edac9f39ca4a87e3692f62bc99d14304a7dd6470b62ca33b07a9c5fb6ceb49f095992c0b43f531ee5df5c651d54b32f6d2d217e9eef8a15b03fd509519d728bb326229d2d10b2a3ed17e45e43fa6f2a4a1d5c9150b30fc3169ff8421c3cda92c3a9b3c05f6739bc99051dfa15903879edc394708aa473f018194595f90e31ce776c501c4dc75298c4663bf3557fe026bb61b6adc4e236a2bdb60b22c6d46b374d1a00da9d2dc5adc344a6058c066893e79dc7b6da1b9b45712bd8f0ebde3a0440e65185e8ab4dedcb6d0ddf33f66dc164cb2dc1865ca8060dba9cefa78a8731216a4a69bfa7d19e3ffa8c898f93559aeba7b33e583535f2fb0e433032c23f989fc5a77f0892bee73cd332e2bfa3116dd8bb1535c6808ee99a24aa4ad68a2a2566405e6dcca1b22b229a3be81c93a90a9ab2c8e10e01e2f50118790522175877b51e0f8aa35b2ea80fe7ab832f563147a0b115d2b76a739d716abd0ef065eb1f9e058bebb7b3763254604fb339c9f5703f3231695885886d0bff616250e87c1f2c90dd710f1e770b4d8145ec7ceca4151673af23c9e9ad2aa36c7368a322944ad9600040cd597fca94804d8fdf979794a7eccf55d6eb1e62d90393a69701c4e9855e148cd439e6d1132d3c1156d71eaf35c7c81cf2841cdf95fdfddc3721410bfa3ce4ddf2af92f6a721a18dd1559fe051435769bf8d75bd39e8c9a05f0ba4f5f99caee416acd9f2ef33b014ac7ecd213e4c9a6d60deee3263e451a6e4e0ec42b6ee2f0448b84f69754e5ef14abfd8ae5903d2994d80bc963e0f84239c798177768a814a2b569ac96169d85e74495cd5ff5f8cf00da5399a851929dd5324963283bc1c1b78682ea3816a906450059ce4a461118d02761e5a2b6775991ac532ca2a2a6ee4bf78f217c7f6f5994f2ee3100c723f20dbed7fe12659494fcf6a9e3881404bf642f270c42923205855b42e229987fbfbd7a01cf2dabdb37c21ebd222acad1ac24c2328c6f5150f73a9ea47850fb25eb473eb21d6eb259862591df0d39d1bca7bfa29d0d401911f75d9a7a479f8a68ac2a4f9d8b28216791fd816de5a3c258d06498b107d915fbc713b74e12d8b465f3ed5c6ef17631f4c586d7122ba02895a9717c8c301a0c81263b827b366e1120b55c88cb34ccd9660e0aa65e39e564d4e69dc04dc5928e2829ccbd71223817ce6caeb3a84aa061a5298ad516d51a7fec4393fa40c96be93a2515c741d9af4d43b48f6d195c013d28bd7232e70a1d8efe900d730bffdad3f67a3cb62cec4d757b9267285bf91c8330a9596367873df6606802fedecad63809240ddfc663c369326af48992b665d6d58b661203e60c884c358ca8fd30e756365a48f46ba39d93c58f9fd4b706dbdcf193a96c52b23ce9ff54a0aa881fe1f31bc3a3a6ba5224e94526ec1817aa5e5b533627d8726e2a93a47494049bacc8e4750e91feaac1dfaa24d4731eace96769fd0190540f2d4837304d58eedf2afac845d3507c684886a15268df66a5cd63fbc1724ede49fc1b73c049a03aa987eeb830773daf2cf723d5f02f0804830fb50d4b722d18ef4ced70220c4c4bd0c9da5ae98ebedae64fcbd9b29d32fdea382f6774d422cea4668db8245caa9bf81be7e772ebd431b4980c7a4a12e3f7a1ff89971e49624a68231d450eeb17eb9ac2f5e048cf4e79aa164505b6b35d050084d894a40f2dcf305dc682017d25d35059982b3d7a125b7d4b4e38bab4f17ae1002664fd2cb386380155e8ac2afc25d9ef5754c2cab95e29414fb223a0600a752035e143d1f9ede53f5b3e43b6359451e5bde292b5df0da521e5928eeb61bfe3cbc84ef51569755b1b15c006ea0ef05678c41422fb9bf6e225142891ef31f336877d52e84af792a5beefaee11d4f15f2f93166d91df232eb1a6973e5e5c74e8e626d784c0f6c45444251f21b075814670d2f60f4e1500db81b1555d5709096da9232f84c413b0b3d564ff2b568a0f400dfea239e209923f035041303bc1445e2f9160df7ba9de203bec618b3f3aa6ba6dc8c10da80ab9a452f2fb223f42a7f12e15115d63efe2ef863564f8c83d1c1178f8a4325cfb945dc1e1baf4cd965e1fda66b3bf2f31969567ab74d89803e3f56f5a39c6ab83ac47eb8d3d9f050e61ac1f65b53fe1610a4658d72b35816e4bc73117d63ffdb704c2377aecf21b14a9d0617f5233c443eb8355bf1a199e56eeef4a77711f2bbf7ddb194c55b21971e1555c80a712058e2d598aa5624c0bc44b22d2b42e2d7313ad5141d969f4d79650e0b32cf2dcfca808af54c3868da57afd70a5754ad3ba35269938cb0438d2d29d636c59d6c145ce181f03146769ac52fdd32f247af81218b98da18f3c8fbaeec51e1edf0184180d79f3f09c98ce4bf3a46f3573768e09300ad491d33223975ecc1278ace7bcbfd60812f661386391967b0ab55681b42a44a039138a29727c590f39f8f3c45764adcac648655b2c63a3560bb00cf216b8218cada89994728bd2e93dbdf20f91d25d9b31dee7222d8d0e6ba4109635de1a8ba193f740ef205ba324b9044a95f24c203eaa215070b62123c1abe19f853bc5342ff6c284b599f2abe55821fdc4ffbb248d6aea189a37bbda63449bfd7af886f014ce6fb56b135d7cafbde8614a7279b4cc2fdf23804cd3b8d8dc1077714b5dd24e8942cb33adae825bbe5b2d173263b999ebc4bb997bcdb075e188978e1ab086e4167f97c2dfe811348c50483a8b2f46eeaaf1da41600e506597cfc83831ee3e9f09e3257e98b038b8d6215ed5a29984c3db9dffdb879fe09b1b572777dc664b619029ecf16e1fe5450f83cdf7f3dea62572e74352f58c2b6c812829e6954afbaa60dfed72daf4a63f14cb67e5977706c12d7ef83e5356006ce4f2e62d974ec1153db90eb402f1ac8d9af94b07d969d26a4d51cb4636bcf9747f26c996895bc79050c22ff7dda76776849091ecc92536e8ed0f6a6529742d7d5255224fb7d156d13cda48849a3b27853bd21489ea1388cdc7a5a5fe75072a05150b744dbb4a1834c9901aede84dbee6c851ed87075fdfe84ef36fa38aa5ed7fefc1a8977f4c733e451b814545ae77e7ca39357c8b85d76192d6c660a35f553b2e61ab543737164c4a6654e8e3a0f492ad0d7ce28ef9d3d875cdb8509b0fea7044f39d2dc900dbaf009c2287c1d78477d80ed1e2b225ce4f0dbef918625a085c5073711520cf8524de0755211c74673bccb1982217faa02dd5955531d49907da2d8a5f27a0c2a9b4daae53ae3ffb0b883be89947a3460fa419cf79db0227b41a4dde179aab6594c9a280b8b435de6ce2188c50f511ce537f2d29fa0f58826d14e39a9502bc14cf373250adc6123c2c9dc9279bf7c2945d7267fa356f5576b25b1bebe5fa2b8ab791b316f01c626bcaf2b178fe40d0c55cdf94fb0835ec834d574805f03963d820a3be3924cec708cb6b3bb628cc7c050959e8b9bf4149e23f5d205e1096e572386d160c4bf924081ba1f791bd648bc4e6a7e89ab1469277683b9769622f06a98d35eb354f158d2f1428aa961ae964f3d980b0f9177057d88ca9c4518fd54d2f8045d092301de409a3a0ee0deb3ccd9f065c8b90524c7d94fb969f2672a9cbd83a0c15329fd5497cac9e4eb90746de8ea2d3aab55afdd73fb133e8f846d8e634b462bf3278f7bfa139a2ecc2ce6056aed49565c0d9f17e7cfee95af812f8f2da2b3351aab44b09a08f0b1b7b25d58b70cc9562e9b1a347fdadd2f5f9143fdc098c927e12f7d5e564d1d0ed14072549634093ba8fda334399b255949e57045e01f255d1f162380602adb3b0b2a4ea0ba3e84fcdcd3584024b331a6898f35acc06847fcc16f5fc7e29e08649b01e0c92e5e939629e4fd1f541e6735db4ee3805cac6135f19758aee5702a4647a68f6d12e15eeb8c1532728faacc7bcb02e3003e17034c40b969c00d0460baa9be886bfe879ad16686a88750c65767e61fd4e85b43b6466fb04754a4c14c496d8077e160b1236b624ad7aa65f580c1399391c78924d32dade3b9b02854f1a9aa767080536faae4a77178d12d1e8c92c2ea6a000014dca34539ed3f9cb76755d59a66b6b7a4bdf2bf475b7000a8578838fc3cfb655b5e90e97374348ddb9754258f4730f16394070cf2fad8b3597d274b78e403f894349e46abe6ebd0ed626e825d7a4d818c8e7f7bec010558601a4f8a307b00743d97b5b6013cc5a0a302bfe5e561299046f106077508a3cadd60067b4f729a9cdf5b8f40dffdc17212895dea89f13084c8d67592f68517be85b51e171b1b3b69857e9686cdbf214fcd0c3063631064b3b2bd88e3f530a2b63c12408702542bcde88029e6b3b5cd6bc595245a38595db0752f6a7248f90a39b435ffa5cf3485b29ceb5320b66775486a154e9cfe70fc20949311cc100a7c4718172eba6bdb8b0c4dcbae25b6a9132d2a0c1e599e5bbd99e3e6dae81e79111f89d64032256382d2df0f2a9751b6bd4b3cd067372dd341e81ae320b75a839cfab88dd8e0795dbd04ae9ab383b1be12ddf09ae5f9917f5ee45688cfb2bab8e290ec84a0cfdf40780568ecfccaedc5a21322f781e2e69d132f1bcc278b1c4dac9cf2313afc386989fb5fc62dd2bb56a7b64f8f62eb16fbd480b79c286afc305f6e88df4008aaa33843edd35a63b68eb03a1b2dc128e63c36d24f1b2c1dd7eb1f27238e5ed160ab7c83326fba7c04cc68ffc9aa9d268d96984f7e447fadc5e00b94618cb25ca247f88ca3628e8e7858592e7a0110ca3132734ddd21123e8562ce61e6305bbc1506662461361302d131d8caa84c71fcbf99912547f570fc5d05936c64c83a970e4c2f5570556812205da9db80c2babe48bb4703007489000033a61edcbe80cef2f7e7775885ac7409dbfd4e4bfba01583aad37a2c8b70efd5e65ac2f75f1acd7c35e17fadc6daf69785c5cd9b91ed4af8990c544d344ae551ec66fb24c5e0a277eb4e28a14e27107aa8a9004ef8cb5adf39550d0a2cd5336d977d3ce4ed66b259930774010b92cba01eb8654e75a1c4b8e73d0a81772b533113ab6ff9ac610cd77ffc71d0a49cf1fe3106db0f60f4a6db4443a6a211ecdce0bfe08e55dcfeaa9b88382042a8f05726a02ff67f8dfa1433bedf111613e4f0dc8bf7f1c07285af4e41587fc7681b50b40dd8276d54a58d5696345c02301fc784d497106064175d820515f7cdfd6406a1dbc33acaf5fa22fc85e6b8baa9c678fb3a78f00569751191bca56906f19d32664fd1bd43361f3888468b5ffbbea7e331d98b83f2cc7fcc5a7c92f0b0b42b7b12574b5fd87be75a13790a94e9e929cc7b31ef53df4199c2aac9b30fcda895091ea762de6b085ba6c9478f6035e405a382be84fddc6e8e10c4af2a4fba929887ef75307e309bf48fa66e049d1f134d8ebf0270593b5f0d1695aae24c66b24443a3c62990904aff12a65028b1eb85b3b2cc9efdc1a4b0bb3aa8ab122e0800adbc54d30e19f92ff2f23fcdd5740892eb8b975e71285777ed043374a86f40ec3ddf7ca6c49bbc0b26f93dc11c2185f441b46f8b047cc3b84404c2c38049798866acc90c047524b1db4d053e75d18f83cd7872c97296ea67123c0d2ed3fae8991122e1784b0359abe35fdc20d7503118db6d25a4f88145f7e68fbfb137e9bc526555795054882daa314bfebd8c2272c4f57e5918d63323f6a063931685efcd626028599b995f75f3743e41e0983cbc81550210b34753087821e5ebeb2a10687e8c7b41bd4e0b94236503a78281bdff3eaf138fb74be8a1e55f73c3e2d35e598bb4de7d81fdb9f5ae59f612f774db152e2609c16a65b99d97bef45b40f5b246a4a16c6cc2655c21e6f8204bd329c80cf8780091b9424494aef6dd69267588866069007ea43c3ae6b08b4d1135240884d4fff64a4ed32918ef67fa52e24d68a6ad84aa0cc3242dfd2f6d5cf457cc45fc2cca5c6aab0c9cea7efed4f4dbf100822ba6203b215eac84afec6969571fe22497b9f75c24565c8f6bbd641ff19b9ff6cf922e81e82d9cb6014d8a99ebcaa998dd473276661cddc45363d96c6e032261f1e7b4cdc5b65a6c80991302248288ab0b13e7795c5a01342dd7f35d639ff9aaf7e43d122d692817533e3048c9b550ba2b105a223da2e0c20809d13a0429a9acb9a087d570c3012641f59a2d823de43d91de201266539bf72135ad72bf0f88d5b6b27999ba25d28199889b2b6600c8994056e8dee603eb18e6b55a1d88acad50b7f6149590a4f308052d838296d538be9143a7dd71f27f4ed588297aee5724c34d27b15edea6d543c48963b688b2c09f4297b1912bb0616598bcd24b07a02ecc3c369e818456485ca6e4322909b33d40b75782d151188687ff8439fa925f8a96c1cc986d384cc170517c6d120497af19da399ad002a811ef4b3c9bdb710d8a43393069a1b5a269cf8fc6570100130c18e5af751ab7ce560851e5235625d1bab525ea2f380f37e7a75c7e10043861c1cbefad33777d98c3dffcf7640b345d8145860f90302a90e9f03a0141b616261bbb44f2c6db99a041e42ccb7f341d5a29ed4ee5d8e6dfec77923e5e6ccd27181246db10ac62a42c51ecff5eaabbd4bd03389ca1e23ff10e755ea658b02d3a7d862d1d4fda846a0cffc020ca2244d4dd0e4d0abb07179b837b44da2c0bb4a8edb34bf3c30e0c74c054c67dbfd42aa60fbea772cfabceceaa1db4cbe22a2e7f7664763d6d8e79520df905599803e2839feee0a1bd8d64bc434ac334deb8c84e3c0c8e5aa000e90f0158c8e09918a9ea247748627cda99724cf9898d46b0221294add3677cce1c29827833fa6dc37bd40b56a3efcfcdb0f930131904ffe746188de4bed7cc8f177f5b1039fe6b321d311f168eeacda36db96386bef04888453e2b1cf92a3abe405283565a5243efb77cc76fbae329830de4e009b2d6213c58eb201c7eaa07f43dd0d825cb9a697d7bc7d065682b82a36c218c7819a1b652aa56429a01f77f677d992d856aacf736b74a486ab4a325204671903ba15fa59910b92b7d379570d5e60e2dc094ffe1cefccb1eb44246414e01a006ac16d84ec2ff1c1f94e86565cdf5c178fd64613e00311fb508382703314bfccdefb95c7b1ad068a243ed11ab10499e03ec5ce7d883d3b7537c2d101c7809f55468255241320c3ef64f8590557e945497ded2cc76ee453e5329797e1a71ee843406837b18ba60ca7beef7536f463e4eaad9dc7dd0befc8db0f99e133173f0b7f39c77ab54fdd033bbb186ad56ccf124172c65599004efbce351c3480e7e9eee25516d62006a038d27c8df365eab969575e3651850dddb1763dce5a82a4d2ca99efa87798759d141e3b8821e67cffdf47cb3fe3634303d49103fd678636f34f9009ea326b1df598223e5893b3ec071343336ee5c8c50a2c0835129c215c8f8f4bd1fb563456cee8ccc1d3cdb0d7a95b150a7ebd2fb703bdf1f17cca27bd677089dce81dfa10f98a6a5d24fefe2628607897b0a8a066ff453c30ff8ba4a5768bde81342667cdb4837aeba409d03d608c4f1b20b0f94d135a887bbda39dfa25963fbb76db043c2856d9e7d16ef2fc17e9089e68ede7bdced2826b16445375ca06df75126bc4f46aed6327be1a39a144c135f58781f83cf7b092327fa664c0c30be6e845f86f5b49858be1c7c002f44778f9189aae8b4dd2a5150896861acf7c6dbea02b980b6c5e70e5382481c359039a3e7ba78d0eed8e51b752edefa9f6932dad238fc309fa35767dc17a22974779f19b4c5f1656d779f2053ecb3cce400953d7aeb8db58f4d373cc95cc1b14ac5244b92e6198dea0f2b75a9c3e7721a822ebc4189f0a1871aab4fb09a5cc3208aa478e77a3bd4ccfcef234a78c0808848d252eea520b09e77acbf5667d8727d1ab8901f646c4483a11dc503b15c490119b420027f0359ad53d6e1365f5f63884ed2745fa1942db0a4aca10d0b3f7521128529dda0e6ceec1748bc854777da022da19b6fe357e85379ab2311d0f4819ca98ca8c64f9d58a57d0abbf8623e230c38f3f157c477fdd53214212241cef4d5f431512dedd68fe7ed7168049354cbed045c91d4a5c6e510cea81c2d03e1f4c0da1b191dfa9098397abca24022601aade44858458e3f34acf38220d3ba00792d076e00321601f71850d7ad755505a30a45858c94631bdab26cb675f8fc68e2411a37ac87fa6fa9edb2b02f7d216a88b11ac968f16ee5021186fbc50ff4db0c54ad685202b3bccdb34037fb120d56df0a34b5d3d59162cc755f00765d994a8a72e2ec983da363a1008594d27ef5819de645aa6109a2d6cd187ecf2a5ba42da9faf542b3826ddadf1c8898f9b470de0b9fbe53c3517d9a28264ca2ffc0aeba09bbfdf608f07cc12512544541948fb0598d821f3058d92e60a58d77820e7b4dafe14c11d79633b689aaff7dea0107db6f0782d7ad555bb00870b3df829634765878b1b8cefbe1f93688916373ed792a8829f9867ac4dfcf78376be447c7a0ad7fc873f96e83acefc150456d86390570ed29bcc8ff40bc8b93a94fad7ab4874c97187f515055225ec6857355a4c718c3a142c2944a02f8198bbbdfdecffb2ecc6c8718ee2f272a12d0307a4228d405d8fc7ceb4011d4e88dc0d69926d44fc5568d7754422c4e7558bedb3a189addd886e8c4842170a6c840cac99c732fa9d7fd967935b887a76cc1cd5365e0239045b03e5916786f9dbcec362d43d69319b95153e650fba753c1e33a27c820e7472ce000626f648ededdf30cf8ef8ac139fc45946c2bb31fe3e1b22d1ba8e5be5c7f2060ad3a349f1aabea233ee7384802a789a267680bd28ed7cde354def03cf22fc4e5903ec86c19fa17f9ba217f7582f50e349519cad198e06296b8ec58a9979a511fad25da17f811b5e2865d43584973b3a45264e7e15ff2e29fecc45bc603ac163a703723762b60966fc9220558cfd1864f68a32bc1eae6e029fc3b1243413498b5f728affa4f52f37e1d590181bca8e6e761cdda5689aec55b37fe2f4cd8a2fe0cbf198491f30e22fa73f69b1293a7282aa81f290c0fca3b4770ee292c8be93ea321a69aac659c9221aee7e007d2dd552050ab328e31ee3c4adf8664a9ca73bf06d8143656ceae1fb58147d6552d78c1d462e62563f18c194b65e1323c939fc902bf46c7459cc5ad0d9db470ba2dfde62b6e1d5d6d1b7d6c07789a513a9e188ee1a26e2e2fff5a9f7eb4c7d567916777995a51acb0f00080c95ef14968fdf4407d1822b64a3d9473f3437087722c37b96f156f4486d16ceff4cae3822958e34b88902697e410e974caa052cdba626178391842d6ba1f788e9d800f4e26b1347d5a2ec38d3e88a1fc888a5cca2a260d4d93b02207dd59a39069f829be6cabe8028f2e5e5a64f774a251ee12d037030285e7a72c97f7e7cdbca7e397f4748e430fcfe900e03d9b04fc988b3deeb64c372efb612158681e55290c6763cc109284a557a7ed12f32d69c329c3b2f8487160ea19b0438c642ec6dcedb9dd887d8babd31de99a1eef1a767a0531e2f69fb5abd0f8f57e7ff0e4e1c6d46c5d1e8d52d0fa261a26d799f9ced7aaae16aa96b828effa6c77ff6e5fb28ca07da15de30821d9818bc52fa36eee588ea8227edd5a8c1a7dde975a06d3366c9c74c169e8ce095dd0b25596b40846cc43db5f8a8838e3be91ff5f7f5a9efb6ebbd92e353e27b05df0f1e4e774d8ced428b2e6a9b87ce668df696287ef6a6e117d35e5143e47394c23b3c080e41dd94b6cbd85c15e11e667002337c1d0d877713b8dcd2f4d10852f66655f0aa9449062369de70e9d7ec39214631069b52a76f116551d01d53c4ea887ad2d0d64fbc8ab047126f1c083bd5075e683adde8bddb0966b0cba3e6c65e80116b7c8339eb5b83730019adadd6198abac40f619fd7bcb6d698a1bc34bddd4d7826ff71e058b20423266e18eb6f45f77292fdce7c354492b164e66ad979947b2ad0d8d504e1b8644c2856292347b75a448376b72f550ff11eec1a114de7aff82dd3be091fd46978091a2470783335116d9a2fa28c3503494616a898bb45e3578ae159fb962a70287cc626f1d06eed9c5d32e2f6b454bc993779be3575a21bd20139f7fb68ceb8a37b64f9b3bbe970c783b0b856fdcfa8e786d954050ff73c7bb1399513d5bf536197cc5d114f8c5706506045892eb89937260c104b1a3653beab5d2769a25ed03a7bda1f5008baa8e3bd498313640ffe925331d8c915bf6ebe98831c2fe0d8b5acf9a806cadee13bab052d188b28021b59c26f9c69e7abc1f0d8527b28ff19ce45718972c6d1994e81d360414162014458cfdac2d7477fff59681cf8b5a17b4354ac006cd4b756363571cb25f0f61d136febd5d6068d0efefd6037fd1eb2a0ae363fe59e931c09367b1b44e47ff4bf9aa843b84d6fd75036bc32c18fe21f06a50c7ddcb69fa1387447f1437b0941e7a727da99bc123865c1b3aefc092bce87a7311fc5a37c7187f3cadbf40e87945d9d3ec4bd7aeb0bf23f46b63838bc45c2464fd828d71054e07e8e16deb820a23768451757c9c3ce02bd26e970563e36fd3fc6aa0bf118f47a3879f52edafe77d05ef10669f97ecd74debecfd8cb957d45291b06a86cf20dacfa84283e8846c42338f7a257ab4d9e2ddd1fe07a1c419e432f229b68ff21caf5ab64a5bcd4e1e74a36b5165d20e900c19d67d53fd1116364ec584272f137137dbf1df4146f0fac7f9454211160f6af63f44b857467afbabeb0c41b03a75e85919d26373711ab879316f94048bde1696fcb99ae7c44a5d69dc614ea1b117dc40c6dd117422c049fe19dce7d8bbb20b24783da25fc1f5bd8f86983bb23ab752214362b343ca0b3e39faa4a2831cde78d7811201b66e92a7263ce04379c6c5ed988f61f527d7e994acfc3970faa5dd7e7bebc2131ea2815a49e2dc5a1f6dc570e6d74c8ba72bfb52ed1e8fa442d2f2723c6104c373be12b10aba5555832edd8b8ba058ac6c44d52e540f0cbc5033000a5ecb25f438d31548471103ef062b06d67186ffee0a8dc71d3c6a0ea9ff4c2906164d0e0701726a00fc3bd6a958898437a0428fdf340a47b1d16ad99b4b69d6295ce99a7bc967f607dbb97b92ee6758a92a55f1e70ac41995ea6b27abad3348da71104c8dd54c87908eab83ce143f0ced0036c479b51b027beafc0961f454ce853c4189605322e28367d974c138534c3c89b50a9a17bede0b500bf60729c239a1409d58641ac83934832d310c741f7524cdffc9ac8cee5aa45bf653464c7f9f5fd050dfee7f874ffb63036e70df5bca034a0f895516bae3f41b4a4bb16caa10653aeb6e4eb9fa1765d06a45160a3a55bb9771d794f2a0725626e875a09c0a940d6a2f8a772a5076054aad29383be6c00fa99eebe23f9ac0c3419371d0689b889ce9a9c1ca7ecc28f3ebad5c6620a52f1cdd1579b9b7e2520565e3bc10c973649d13aaf1d00a03c4fbbcea91677d33b509e4cffe34ed5bb4ce2b3fc8b57b93574b415c9e6e78d0f89901fda2eb70c2e8699d5dfa5ce64c56ffbd4d56998284b1da9d49148fa13fae97eb96fd05684284ec6357402bc68365432cb9c77ef0ac4c9bf5d37e076647aaaa445328414721368471a001b88cfdb8ac907fdebc97108361a8419cbf9ed615cdb7053b6701509e39689b354b138c193d26a5fbd876203e58df0834e360ae1d832b1f6b873d86b6f9d78741180a0841ca0b0abf6a1f3b935c9834c6ad76852b3166b2b037e7a2b18ab8fae2595b47c526895ed154725bea2aa2c89f4f2d896627fe6a1193f8624b46e097f15c46d431145d39042b43630f9b21b53a59f8a211553b69803b5ffa525e488532c837b48dcbf71bb5b0d0a8c2e237270e30af673ff41233102d5f53edd6ddd1e50ba4c46642e0e77d58b2e3531dc7b29a330ce90d7b0a705212b8e9816ce89e1563ffe7c9edfc84f511054fc365de082c941614bf357b4c6c365bc33bb9527774e5868f1641aef287b26145711972d1c519443fd5fa55fb007a5f751e9fb06c2b75e4df6cf31ddaa34dac0775add84822cfdef4e89ff497bd8759231c1e9deec4b56f6641bef36ea4d5de4d40889c4a96166db4d5ef8b604aca98d81aa4c777362241fa5d3311cf874e5929f6ffaa73d4d831a5427b9a8c00df2e96bc2d3b99c7a43e12a6a4863df0764c827340386996672452e3d3c699fa008d9974e3b0aeb67db57e37ae936b0b61c461e0bc6ee28c252bddf099d4e2fcd335c8c1bded99f182d711d477ec8e3740f3dc406cd3d1386c822d32f0ca3c525512f1d94e5c76517e2ebc9a86d625db14ff9c085fa58baf495210313206e1439912455b4e93cfa3b5b693f113121a1ea2b468576f5632798f28171761d438066523726a7a3a80e271be8246d5cf2b2bd7cf3c7ee924aae74b4282c7d3f94b5e0c77ac88ec4d8dbd26fe24c157d02832ed09efc400539d9183a489c37fc3ad88c0ec237267d078f1f5b5b239aaf0f0a083e46794c40946b4f9cdbd2e3e1d4fd63ae692f10901f6103fabb7d02b4fd140fd264bc0cd704451f3e823afcba8cbd44f9c4437b22063180a82644590d786a629d7dd466b34083388fcbbadefee29a41a027065c8d563b360edc0ba428c5824b9f195f06f8dc9fc7137c23bd40dc425ea6fcc7a5bcacb91d421dc647128756729a9087f614f00930befa4b4290b741a185b0123aa676d92438b143b0e274d6cf5efc8a96a6017755012e6d33c452b14c65e08a44933d7bebe4317cb3ba7a0d0eb6e145b22f8bcdae09ee71c840220adbb013329dd248fb301a6f618db9b0afe574ce87bf7431e3dbbc809e78cd84c8201c5a889e9d3a3b7fcd7a5195f75627ac4040da31417db139414487c66e98120fa7f572fff7be65fdc35b82d7dd647b7188de32df7a932f894a7d2b61be480fdb3fd33c9621d506349484c48ac2463a3087a5a3473af9a0ef304ad9ff5b5d8da1aa18faeb86924c5a463efd8a5e3c712a4f1ac26e7c2fdfdd2c82f41e3e8eca5c45f24a3c442cbfa12e29362c4edadd173dbfad5b197688225904a44a6516670058fb60dbdf6afa0330fab3f5f2003ddf96e6fa4c9ab313ca40b33a3f2c725fcdc02bcc3a71f45790b0ead3e657df8fccbdd52123d869160754a920853ec1a926896147ec7c442500f1351fac11b816022d94b4871820a5498f21c3c9e244ec0069e9e9ed02146b3a1f3013c3c31abb99665e5a5a9f4359f1fcb1ae2f9b3fd281f38a2aec6c37b30645ef8286c17dc0f0803f882c2e8e5f362bfd13f379344df81a3eba97b9438c3eef438fc5e107794879ea40c2bc19a1c1916542aec46d7de7c785e92f48e7f08c34e653396f4355f4cf38ce572e7b614a2069e3c6b91b4a2ac6a2e8fbb7bf4685a588e9181b214be00df88a78c4025584cfb398daf19ac9b1b8ba23df794a1806918e62cd5654e3152b7a275508e02f6f6e7954281c92011b2d92f46f09c0b53a9bff0b67f45b428f5b6d05793b1a449f15ff5a989029a4aa68f673d583affa6bc5d06116a02592dbe4e6faf2ac248d4a213f91c4627131c8f28301325a9e72469f09b7cf69fa4e9e08ff1b2a91d54f961256b8a36ec5399f6f925c11573f7356ec04967708faa6c58528f737d689df269e99f89379bd5c55f9592eda25c365e938cd5ebab95df2b97fad5b83531e63d5e812d917e19d2c4a71175c46a0449a8eebe93e3aeb7a06e99e40eccf8d3dcf66b8d368df814397fe87a31f988177312a1a84286880cf745b9d077cf15aeaa1dfdfb560b4d3397485896de56d083056c9b449668dbfa472ab1d0d4dfae81d703dfd21f95552de565129fe81696a81868757d91d0e36dd200ec88f39c33cd5f6f00654ac4ce5097c5ad5b4bfbee25029a6eca225f6813c7c3f3cacbdf5f739c61ba8b50e56a2b20c6541e9e31ff1f317205af88860d6ac1b7a3b4de70e13a74c04125881daa517a52060da9ea34035db8789f12eb29fb1d9f88bc82bb862629c28afd723bba78d73a3a9878f82b1d01f7c957f2aeb34b05a8227911d8c4933fc8f51bd3c15caebef0174c0f0382813ba449f96fdddbbe8d34d5a4be642546eb6a5c614215d3ea447402089dec8fe4f01107bcf721ff631781cfd982db413dfe22644e8ae60ea23455bd2c8a506598f48d97d2628ed781e0e7e391183bbc4d8b284535625e29e79d80cb0c19690527bfe28c2ea8e29040cffe5b137934eef104041da8851627182c342aa2bebde468ea05b9a6c5ec8f12bd8db4d424aba3f903576ab4bb0d5b122db66a85add720cbf9a6da6f2fdf03b0b13eb0cd477a2effd033d9e83c917c06dc592b7716a8b3c18fbef459a8b106c45a3429e266c748e40883563b6307d7d64d46343792f6e8f442bc1873ee22c66592d57595980a3320e7513024382152839dfffd927a3849c1c5deab1b22d9a61de93e002ba2eb77175b95e5aa994d6c27ad990eac8cf5e654c647eebaff6af0861cd0a1d9913262058ea10b30c661ff0d59b76df12822f07d69f9b7ee0207eb781b52290ef51e7c8d092e8c7498e3e50d5685798196ce549e250703f109f48cf7898844fd306f8ed2a39458434f27a9c677f20c1b9ae405f943c5eb08f9d57890121609ef909b7dbda364cd890f5901548d6eb7c92d1baf31632bd4f93026288c67242fbb397430a1908b8207e59e991642d8ceafba20fe59ee9f60132b47772b1d680703b1227e82831946e2e5213e50dced773ce1fe005b2e2e5f9396f59e4ee1321848458e92d739f2a7583aeb9d64832244bdda09a21d8101298eb8fcf5afc8b8b5d7853d07de6d4f78b0386727a47c25fba1203a7a18601fd3401230171ca6ea4d83709a13193187d1ea4a8262fdf0142cac88d9ebfb20cbd035b7ea5a89ea20594692cd184d45622a7cdfe2bdb7bc6b467d08e1ecf0c4210cfb9dfdae762feaaab7d1f71dcd58e0ff2acc25cc351a19463f48dce2aa56aa763fd17516d7b92da5c167c4994c69e8d8bc8dd026278ec6a87bc703d995d147a54867484b8e6da77aa97a856507a06738d54d497fd715bae83f1fcf7b0318f6c0100cdaf4fe779a47622e5e94707fdb4e1ece0da4ec0bb297ecbe32d4b07ff9e9a6ced7ebb2cf93232e504c71e8c50f4f5d8f131dd552ea48e668dd2818952b05cd55f59583a5780854f61701d2e5bf35e986eae5c2a5d7db822d0a81c66a46c6212d135a66ff41feaec1c8b5f1bef9b4507375e68101cfba076d6fa3363b46a54a78c597d0ca2b381ba3b2c9fdec05191ce036d802208b912b14c58d2ac48b5ba0283f641b467a955d05f32248ad3d21503f3644250928d5709da1919306cafc6a0426204390587a82d7fd2d8281073f8e0a278d79a3cbcd47148b8bf61f48b0fdc6fc4bc49098504aa61936d70cc90d900f9c54cbd299de2c1a6eff2691e244eb45c07879b5b721b25eade3ff3b9e49295d708198a6917674e68cdc6c4f2603ee5531222ad584545c351387ae46a0cb1b12021764e38c7439312cb7ebf14e9a1c7dbb79389d5ca1ec754fc002a0431827e0170e22b0b440ca016b861ed8a60d772f973514d70bd447363be38dd728b372bd4500a8bb7e83df7ab9572a3e7e15e88da958d0a36219500c8a1ad8e614ee17dc0dba1d7784d8341c85adb92e4e042aee872cfdde9ca8e32ccdf527fd0db040763b49f01548665d96c0b64896824aa794aa5b045ffb2149957b5116b5b2d81762186e31573d79090679f846adc47c83e32baaf34d838749433192d81e0e4aa8fcaed63dbfa922c03dade479f556904825b61be165ac32d47095b84f13c385ac2106786747872e865524d7f05a2427d4bcdf9991f2deda3472e1f67dfacc00db32d68a85178da23ac28b2dbbca9be7991447ab301e9c1104aeef02e0ff59e2d2c734ca2cf4a09fb32748d6b0b72925696fc20aae0ed793f08c400f8807d676236a7ab0968900784d06b967f31a73002d24d2024e98fd5fe1be8e3f75643883a0257e6d080da61adf0d2a69962bd7238359db152bdc84ef8572080318cc3fd14f5df6ee02b2a98cb70ee4e1751378389d724441e72b3076a8de744699956de92f76116e833bd6e4783673a4aaab206e6830c075df0ffbfe6d2d9fd49d8d8c886ae0d05982a7a928c8c09e416b2122b5bed31cccd0c81d1510ef697eaab6e625e202aaf97df244a5733aa966d7a9ac77f78239ed2464b5a07e0e16b4177bd3bff26a84b589442cc7c61eef3de454ba055a1d284ef6a0c4fb7ef320ff0050c1952cf77a68e62291cbc692a1b8a0daa31142b335d8083ea158e126fd9cc60ed46189afb9af6182d8064788dcf6c2ce990c42c6fe16ecd0fddf1bbd286ccb044cc2228925e39aa71b87127ab9f2f80a51e74a9be3baa62f932719822e5796288a3b704683eca34c789fdf29fbb95e9277c276764032aaf0f5d921504a0550b4fb2c50eaef628cf6f6e765ffa591d174a2ef8dfeb9a95831e1632c338a1a71c366e0510f6f07d6c1bae4524bdcb13e21896e3d4c7687aabe03f6c69fd6f4c1c2c67cadba62b03d45993bb39d103f0fac4804efb20cf65bfb9d48ddd4464f0ee714120056e116dfe1cf42ecabac0a442ddf5f2604fd0e59687b41e7f9559fb8d6268f14b5532fc890b9d3c40179a641feee45c59da34f5d3429f9a25de8a4c56c45798078ecc6a42d319c2a3c574c94d5fa1941b402804b5e99f2740ad57530aec60b82878e94bc67dada3398f922afe73f0e019b4e94c4324e6750f1833c457ebd9bbdb4c1217705dccb83f9548c96f976081e089fcfc93847e1824604f1e07f5cc8954921a6039d6e52f2e3533245f39627914bf363f0a9a04889954eb05791c8cb43809cf605cd4b71075da7e17ca254633785f6cf0607bd056be2e49ef396e23feda6f9938d252022c2947eeff0b01b743c2cc0e7d7eebbfe2540ace6add5cbbd04a500fa04f104f86104e609943c1a9760ef5296f70db4d40f7868d5c66d747f141e0d7ea913fa79fb0701ee0feb204a7b289f31a401e80c5e1b41821270441bf2a17b11d0e451fae57d8e27eb3a3cda34453c25d8016754ddacbb6d9f276ca03642def821743c72fea42c2fb8ad82a8cb093a68342656d85134a29c1fa4d4ca76e3f74e25ffd977ca19dd6c78662044650844b24d7cd68c86efa901d0a6b86f2ba0e3f46a1dae42f7c6fedb678d887f00b402ed897ae63ce784d7145f259e425509e060b5511c7c57689227b6287efc75ce6c2cd1cbd071d8cbfa39f980584058d8355bbd907431f5520a818b9a68abe91d2fca91229a6db265e1a03ce308ccd573a2150c4e83cba491da74092a13ef93429e92649ecf7d8ff73c987b0d356e0ffc42998c7c09c782ad7be5a45a7a16f1584cb2ca2c987f6385e9c925287e7165cdf305da2c06d4df0252db2379751ff7a77059329041a163fd16de0bdfb48a401c32a88b0ec8c24cad1d99152ac1c16da478a6e8c78dcfffa87aefc34a833e0d11996e0673dc50258dada1f6a23d375de98f5195be83d58c8233ea70136ebb3cfce69baa62b748f8b265a50a4e6bc97f9451747bc618ee3de090c86ef355d6048fbc8bfb9512f1eb1990ad5ecb5d88536ff38b49a9dd1cc9ed109785589013c1b8fc4a7808150e4006742d3bda0c6bc2e0b36d0447f4c551b61375bb0f0c393373e6316ab53d288b6af543ff7b71755a42f494486f8eca4780c8ff1f802a92d5f41393404746eecfe6e0ef4a1bf573912c09c8cb31607f6f438cbc4cc91b4183a6602a204b5c31a1288beecdddeaf8128b43076a3bfdd8c7257ad8db98f05810adf36026979925b59d5c8611c2e41a81f351f7ba3cbffe648383fce28822dc9808059b284989946b519826b705a8e1903bf5a612f881015e1e77aa4f69237fec7090287179f79e0692bad5059fcb8e3590b64b4d548ce9a62e19480d9651a69091f2a75c4ad137dcd84727052d6f1c82f9b2b7dc4204713fa64a699a42fdf6786ccc9ee6611a3de80b38a20f885334edff67c66f0cfed561af9c0f27a81b069a3718711b930c195e805d4a04eb69ca06ddbbd43ac01e248951f1a3cdbbef2a6c84a5e303306cacccd4b9e02747dda0c4c53740be44d31add03046cb8976bd177ee344ecd724234040c904ce125db0f9f9fae70fdaa60c18e1953cb8ee6e8dd69c6ae42a60e28fe8402d79cd59db5eb5c4dae903316b12bfe9649a1fa359f2d0e19419e4b834d5d787220ca969302086f01c9513fdf9fb1ae745820c6c953bbb8f85738b346ed458aaf9410ffe0ecd9a3cec7e7a310d7cea3613b07f802f3633f55a8d65ee5496090bb080cdc6c7259616c9feaf25df6062e7e38bacae2e55c468e22e174dcac7b302486417838b023fe553187ba16a07c0053d0e6a8254a0d4fab27072a0a68c88d478ef0bf812cee26e4adabbb043a3c50262485cb365a696a6d3cc903c59054b5b93cfe245426658103a490992df241eb005cecbc8fb38ce0ab94d9a7b1917b9f424a1728e4a0b355ab8d12aa981c446fd206c75c26c95abc9665dc4b94566405f36c171adbc73bf6b55859e7f834a516d2ba6860295233ffcabe2ba3ba3139854a56008109beb2876ecacbed1ec92cb079aa43b0b6041f84cba2069f6de82e6a32920cbb3f27fd8f6558e6cf8dbba2327fbed2ecc0da24803134063e599645245c22c63463324a086a948009236c0fdeb808100462d433778f839dfdbbe7b21f7992011f13bdbac4334d55d3a16def7d279a0d11e7cb030014b86e82750cadec274986a07e9f6ccf79a62025abc2534d118085422996fde6e340d4671660bc29e94d372decb3505bfb6389278ff131f7efbcac0a08cd21fcf1f3c92048cee5b248c90a2781164012755b50ce44b87f68b1b44e826e81d5a57811f56cbe42dd30eef313b622820fdd92bd7f0ef0e9acba78e0d598aa7f7db518764d32190b1d01973a57a972f2d17af766b2514b75d2fa1b18b227343555eb21a008f34d64c9c1cb0ecf16b470db4e431a6d8cf745f9a2fd01af8633125bc1bbb1395b6cdd8bc98b66820af9c9e0ec0c557f4e38d13c058dc5e6efc92e5f10b63032f38136029d4485d38bd85e43f2d4360b882200014867f68e0b64d23a33dbc0a6ce2244ce293fa70c8085a4ad4a18df6e37fc14527f0618df0e3be7fe14b22332afa62b9fbf6845be90f826639a2714935c6c8e4e2c8024e818bc5ea2c4d71166eb5a640bd988487a74f89ded020280235bcbcf17f2fac157c0dd30f699d34f534449f3543104ededec9a22e625a8abe21a66b0bae3a53e41fcdb18185f16103bea31ab811e24bb9098191ea8d78b37d20e27a6c674b6b801b338c5c239d46875d44e12b5a0738f6222bbdec52229f41571a96fe841559afdded4cb80bba368ea554807b1969d8ec88cd902b60e36e2492c695c9b251f21e1ca520efdd783554a7cf155dc9afd7baf761587128e7835b9688aedd45b9011ede4fcfa351f0f7790c46564c17874c947d0beb63e79278cd3ef723d3f2060a52d5ca3089e7666be0e7eade8e04f08922bcecc456ed66126c5bf3ccf32bee486e47dce12217e3fa7b1152b101ae05a64ab831a1ed7f1b1be26aa6b3ae22643e85e94b47cee2c44b4759ff799ed580c8e9d0b820b2cdd0e8822a7825ef68076a9a1e9fea08b7b1a9bf14cfc4dc0f0fc5c76dc9f09c3aee7134f93f09b4c4db7edd522ac92c2094f6ead53198e18ee8ba5ab82bcf96bcc8bbdd678b607e86f5e4ab27e3a9ef898b3c9226897d8ff19a31ed87c2e0855ed02159831928752b11f82dfa6ac1dbd1759ed23b713f01f356d01d5e115965c3e9c83919d64ecd8d782e066aba8f2d73db0a01c98bfa082c695211dc55e98749dfe10a401a8539b1be3d294eee4cc5cd22f8ce48b1f34909ef933e47ed795ea4bb17a987ebe04eb6e066e5803a8ca6f121bec1724b141e6dc563d44e4d8d587dfa87bee704e97bb4e64ac34967dae18b652fc896635f38d00525bc68bba1ceebad015ea5ea9af4e421e45909677b5e984d409ff43fc74820c7e56a3639c4b378286b7dce8d113db8435d34217585fa5a17769e807eb212a7f4270b26d75ec331aec2fdb23276d077bc2db92385606af1a31bd80f8c478d12bbe77ea377b55f88d89fee1eb04395251b48921998ca3b9e125d57e4c04a854ab0958c6a6fb99e8294f5fb0eb46ee4098d8ed16f05ecb939aebaac66baa060e813a2cb3c3b22fdfe4047beca82e9af8c0623e14097857343413df37e00c0a33d15b4a4273aff5eeaff2412c8ee7412ea171422f6f9116e60688cb2add1c0f052cbe0fbf94401df4709890f371db058d24a8c640631d95d0b051407eafc522dd9a10ba4079257deac881c7bf6ab60cdc420889d608f21a6d65e3f45884b95291aa8e912cc482b42beffda8d4d88965a7455d7332c63f83e38912f2761f448155864ca8f0e852c97571a366198ed3e29868be7d9d5986a4bb0cad5cc85fbb810b2ea7ab4ee65e74e658ca3ea0448302118b3f5ae6f299dec276f0fd2dfd509ee403d4dbbc300f6d738bb841400475fda23e066e484c6b01157805c551328be3525b4dae7b0a0bca66c9936b792aa41acbaef588ef2cdf05feb74cb6d783bb905f2ef1210c051f3f7d5e841b393a2ee021ffa788b9fefc43824b3b0be90a51f98f304ee770f4c08581eb061af044625070615665f7f45e0470d233effe335d8507081d2eab070d43faac4d6c3347648157dd2ad8c541ace89864b1ac70c63a946f45f2f230666def928b61bf78c7c4ec592cfcdd6dc5e75125332ee933807f5d47ce6814cdc6b0b03014b654039edc28134bbbe9f716ce6b5e366b51ce858cc8c0dc0d2f5ca27481a558af880d50ed22f6077bddc675046c3131a68004df46dca1f67a25d8c3636889a422faae73e1537b5d1e5a815dc5f0b4ccb91c718a55529ccddc678d1753cd52971281e4533ed5020ca0c17bb1f40157c0d3e286cbdc3e8d1c1e9174bed32f571b0f12b6c2f6d3d0d7443245dbe22c3fd87f9ee399be84d8c82931a52ddfb8c6044b302649cf4a9296f3f98a7f9a79235fd0b995db57c4224ef912354b3fd6c48aa9d38653924a17eef311c15eb01514af562d30e63ff0d9ba7599be5e12091a0fde98ce47dd39188b7b74b9fc0e22795320270776014831d22d231820b9094cf96a4e6ea4a4c4ebcfed578bdd7952a782023cc3780bb372a0d5c650cb28dcfc00a033f08b9f5df519cc0adb8506d404182f583123d208d15cb91cf337c4bf7dd10fb493e50877795a9d640ca39e6799fda4c645411912c1805a897ba54ce5951bc20d9d6b5f4eb726955f1bb194e3f1483274243883d478c7e4046428fe629e91cba8d79d61f90a292cf66102af43b270f2aedc0ed5578ed60f93c36ec18f27776ed72375cda2ffd54b7b81a7da86ac8153e3b6f1fd8ad6dc0748625d50f534cd4ed07df7fc3a0fc5107d8819aca40cb6f3056feee654d822985386160f03efee0ed7aaa08c79fd929df20cd527b7168def1b1fee3d9bf9e0bc595d58027c67d3d155e1d20e454e8bfee4392cebb85018bd169a208d08a4e770797e07b7847736d5cd3e7205f25133a40727d07eafb64210d83a47b1d22ddb473b377627eecbdba31460b10ad72df3651f54413335312021be3500c28d34aee88a6083ce01501252ad559df841f30b51a50954d8884aa58b6cbee9a660b52dcc4996bcfc8a8c44675a479263eac43b7aeee082eff8ae40dfa183b277bdf4bc4f2bd6983b2ea69920c4059542bbdf065c3b4c5ba44aa5e83d5d1a098000e3f893747b37b4da26312f87a8d460e1b84a01b33b400b181fce4081e82448642806ae0a323b8ea17467d88cf65d539f07ccab4dc45be9ff478a472d8a349e009b6ce785dfd2ce7d9bf2a7dc568fcb850f2043870543ec431db38c1c2d950db8baa16e2207a6cbbbd3c09a78ae123980f4e1e5c41a0543dfa63a3a6bf48a167ed98989ff3366366d632ad048576f85c6bdade5ec0284ca0d030b1c230dd01636a99fdf891dda8340e778d122f2b9b25b78960dbf661d667ec5e7bc393b9f7f6b9ff51e2fe6913c4f117c7a2932f636138d2ca2a7dfb67a036ba74bb86bca72340a3bc838298b00a1a6765164d8e2d14e7b98cdfe5b18cb4c3ca13b5d426900e1dd1049f5a235aa0974d8ce5040c7b0b607468b0a790fa449b042e29333be5b8ee652990da10d3eaa7b49c05a7366b6357f6cc3860130eb26408552d0ea341be165d84d7068b2485c49ad7509565b533d6862ae67f0de5ad62917cbd891e9e65f2a50288cd8e16eb8827d8e52747d48aa3ff7fa1ea87259fa759780e4c854aba2f28a39d1bc38c87cf87258cec463ae18d049fde9a36bc073a8a15ce092e0728358effa47279a7e4eb985998acaa5a801ad603614a7b15ee6384f72c8adf437240fd6efd6bf1599776db9521d28b11a75d8c3dd18cca731e4c905ed5dce8fd49be92bc9715f543c2e99a5982a70354ac953b59c422d475e56c52b97d8c6a5c6236eb044f6a37a3bf93495ec749cd74889400915f7a546ad155c1071cf82f9beed2fa2517c645c1560ce0e512e0a16a2109d632da52b0a20d177569257aa868377e3bed06f2989de6c16057bf9b003e0c5f7710c3f7b8eaba3d8f41a4daed483345649741acf9e1c11497ed15906f2d4d0b147ba1eba1404a1fb7e3733b4c88cde1049c93a5f360eae3e086de35d2378f1dce7974e01dbc14a28372a3c5adb2980653011f85a615427cd84b6f3c13f1127abd4824ce7ee3298e46c5fc853d229095f2eecfd0ed56891896fe3de609736626a18b6ee1c93f054cac38fc8e4d73c344cbf93e8f2bd76b67689690bef5c3d472ed47080fea1ccd718835bb82be8018e8f89d77b99e8c0a5a733d761e10b87e4831ddccead65add7834ee73e42be2ce0d2287179450782e31ad32ab0620f0fc2b2d0227504ae3b2667c4a4daf6ba172acde428b2b9de4c962ca38830d03456f7c53810917c8ca9afc2fd604c39c0f2c99687fe0c979ba1cea123879817f5174fb6b4f2395d7f261c390f1f6c11e4892de5130e346649681cf10e2c67a80e9e4ae12907019dc6a8b13bb63bd66d6372f9730a0d6006032aa1bb8c11f6f355dd6a042ecd45ba9be241dcb10adb89c64cebca45203e0614995864ffab6c8a974791fa609964bc868df6be1294eb5f9df002a4a191b9d030e9fe9a4dd1bc4df78d3a1a2a81fbd3a08dfc140f8064e868f3d119c31e5b5a250eb1fc60758c096da3e8118e3e42b8c447b0f8b04887b0c6a7eec8399fbc1fd72aadf1d7ad3bfbf9db2610cae40bc9def92c8af872cb7e4fa9bced370abf4ba67aa85cb852fa1389762c4442aa2c184251ff91c123e519d072b986c2a9ecc09f9a3a1bd7d65d0ac27bad6159f81982c84c6e9d74695091928f9453155f645386b3a64fa664ccba4161afa064c2449c6242941f1dd281110e3adf8c0931070d472882cd0c45d15a251ffbd204bfe5b54628f104df18f313f3f1df67efb9a6b0877a11afe70882027733212007dd2599b4bae4c5bc6a33b8ef083fa0acb99ec15f180e2ea6cf0c3bf2884d356920e154c53f5d7afc50ce0e0ef941b954db088d7081f2c1ec1948657c6ec3a711dc7dfbd6dbb9d8d37aacd111d98a8a41d36d86cb3be065765f536a7b3fa3b7c18a70f12d4e7beaa7803983355283e5016fc46d4be54b3d26c6cfa519ce0c83790ca748a7172605669c9ccd4755f9c79b9f9ab76e0c7dce2b80831d4015a8fecb59acc7e042c9a0160446f5b8061143c96870d0f8b035cebcfd79b3c23f203605fd3a2ffd0173c86149a5c1c54ae87be1fa98dfe85ed8a48c98434ab521b227befff9c1ace19bd1c1505cefefaac00d8f0a4eea5cde827d824409936cafbd075d861fa3fc820d66cca126ce76e67ce4977e5b42185a1afcdc662eecf482546ae0363b4437dd6b9d8d52dc954a6b3a55cefe1d2c4b654d4184d70a1a43ea5e3d52fe57f13678823455abf7ed912fc9d1930f13c136a4c5ce1752aa4fc8965f4ac825e5213d3efb18b4885c2eadeed81e9616c3b8b5aceab189db8574064fcec82aedb8282e35cbb779d8c47b61a29fcf850292db555cb072f373507b5467d0b9aac11626169fee11b363f1e7af86746cc508743e473551fb71b6aa7b908cc9ff6b62b7c4ea8d2151db112314c685aa0d23657e952ea33cf5e72965274082267956016225a44977338840a4de4dade01b749c44648878172aa86ad847f6473a0e84ed490290d686baaab177fd955ad719f03316241bd780f7a63e503f21daa686e6caf52bde3dca6b81716ddce463fba276f39faa927943145159391dcfa7c97c1495cc6f07e29e794bdabb23270827e97625232c7e22825958c2a70da934bd71dfca49f66cdb36b714216abdfb95519312b06401414b0e02300b470e38f40f8393e1b5b05d7d51d885e9d3105d5f4103ac143fb63de86a6eeb99092536831b34ef332d1311b181fc12d6b07e17d0ae5c3bde4ad03972c3eddbb95473f01256cf1fc30ced7a705aeeaa29b56fe73b5da8f75bb4c1b61589865bd480b07bdccb7d270645137d039290aee772dba35a1dcfa6c646fb93ac1e883ca98c13140fc945101fd03e4805e2bb9c1f1824d31543b0af02a2204389d2e83cbb2982bfc5caec5d3832d278118b676c5bcdbfb5f6c44f5b2b803d2e3d854d80f7e1703ba3f0ef3b07a3d1a17a1c77c6e6b7391df5524225e6589f607c3c5bb1a052a1be5e0d478f92103b21cebf298b4af738f865cd95a5617bd674f830149e3d449d00874eace5d84cd9764ac20330f2db140c27fe7d676ada2b9ad906be64a6611df70978272f9ffdf1ac0ca53e96fad6b49c3f2d148bd7ba47783b6323db884527d2118f9caeafe835cb86cd6d1e95933b1e10dc6f3df72c26b3bc2bd8b5ac9ef41bdfb3e12c240309502675e4bcc7c1337f9a8466d956cbeb32c1e22b63ad5e8e688b80020dfe651c056f7549492dcdfe515120ebada7f67a7b6f21bb3ad7c57fb3dc50382c207ed2039a0bd8d90fc32e669c2f899c5ddc45d7c1e965aaa20240819c6ab7c818c9989ec37f027c5f9681d31de6293e25e3db571e29a98b991692d0ecbc1c8be524cc150f266345be437c00989403e285547d17f9689a2fb5116027808b4aa8160b97a8daaebe2109b60eedb52b9683b926bc54d40f7e0165d773de67dc8ab7cec4c7296d03f47b0bc531f690c5bc3ecf763f6fdad62577ee39e2af81119b7e00fd243468078a6cc2fdf3960c3e0bdeca56748ba3f297bdedfad491a9119f2f22a9de1db6c2dca8fbbcb54081ddd95c59cf267ec1cbb023c03482af0c1045afda36673b8fee0defdd9eaffca60bb870cc6148ad433f0aee09f532314870751d82e04194feb4aaaba3d11b65af911042e31a7ee7977d7b878c5d88d6aa735b5b1788f55b59f39c52a8b53c35f3807c392aae15fe9ac5fa36fc538a105356cf09915f721379b0724ea4cb1394382b1f6f4dd96c8623104a8c67b164c42eed8b5ab3da83f69db8f0fbaa57361f84bd3f3ef1762135d53a917dd752cd2ae705aae91ca2687b03c239c9a182fd677ab382910a0f370eae2a673a65abec5edf375b0ac1cf44e4e9d7465979dc7153c25d2dd359dacefd21ffdb26043b197d3138429528c75370bf45a7c4c7a84ae2b7941f901c860be17c6baebfeb25a485002de8427f06dd82257baae79db8d526386c7a67439f3e7d79c58709bb025c075df3047e64a1dfb7ac208d69d9db7d4f2fb7e20a3dfd5b18d0e40549a79498b8d0a7c44d5ef73f860cf4b42ff159cb0bbbb2b8c6583298f33dca36d3e1b8e9126fd1d0dce2f6b1fc73c571a0fc76a4edfc0f206b1ca909fa1eec0723679e8c3e088cfb9cc68fa7858b3601c60a3be94587ddd2df2e2da762c442e5885be266998b008070881fa4d416dd894d1db54f517da880a9d48a72b4f21c87b318f6e671eeecabea4311f072a27c7e70386b4eb66f8ed3419d4ec5eea4038a4bcc9c60c38890eb09b2207860d8766bd8602f40f562b17f072fe161345aea5a7f0a11767043dbf9e36f0916c1b6affcbb00fed367c24c9ac24496634dd0266e55bcc40e656a815ef85a9b4da663f68e2eca949a12255483eae9c720736844c8c927509cfdace25ed0e95b92917cdb9c190873d07bdb431a5f1c398da1c18caca4998b34b8d513538ec6dae7a3348f14ae207b3a14273aaecbe70ea31e7f100070c40f3a78317b294a1b86e75dba6ceea2bcdb71557097de05c212e6ba1a6de3f2742b04226abcb1cff36436552296f6881ee98863774d8183b8745db049159e62504c39d6f3d77de8529fd790be18ded60e899db298bd03141dfdb06af6ef966e51f6bc48ed7592b81792d0bb5f52b71a6348a04e79263a96ac4d4c6765d0378a7450602e0fd27e9b39045ffee3ef88db830e80d035ae2f880aab0382a13f03863a02a5ded736e5d5b725e7a63c317977aa1ae032030d42750c418b8becb0399cb6630a0db5b5df15c2fc074118c7619de05dd58cae1e0d6fc0d3ec4936c65f7c2c62f2dd467cb0acd6efd692d30b0531037da01c3697b26eb0e133b457154152f8ebe903255cf66fbd7bca10ea3cc3f071f3b6457d9d696aed6aaec6ed39a94eff64833233d14c6dcef6ec59ba9983603b457e6f030484544897ff526743c2dffe52f6354186f0defd52ef5e331b0be6a4d55060176a115f2d1a3721f014d72184638e379e4fd63f8925b3568c294f7b583515bb108da2528c645a42d1dbb5e67e4482c39e13146df0a01f21007c7abbfc7fdf252eec5fba0f34a8ba1168691aab333d3416d022a3a9969c6af1559b6048900227dccc417b839929c7ea53341256593569d6c42d81025728b2da9f812af8fa446d4eb764a4130c480d9c75de8017e46ef50125fec02b598eab927a3a4a9a234138cb31bc1d5e46bc8b234066dc0905400d1f631d6d948714d7ca14f228b6baa6cf108e6c533e750dd1b9fab2fa5ae8dcd2383ba43200a48f94793aed69d4aacdf38d149f0b3c52ad4d0b16d0f52f0eab12e92d6d80ff9258625a0cdfd8b6f383a89cb73f38735252f4130a2a57c3b47407aa0dab7b3b7c8e48f99d8344ac4e8e89eb3a30516526165b09a5162bc26ae27d4ce8ce1cc7b7f9ae321e911f950e4ea4bd648be17c0fc134dd8623e56ee95a5aebbb0f92ecda6bd4f334a5da0d9c880ed142b2cc547156dce9d1648df35708f797b1771b3ca76c21c26c8df206dbf8c0554dea7b3fdbe1aa34986384715ee3df1d4a594c79b2a17fb31a66c08d9bf08abf63510fa6678cb25d08db11086d652b29c33eeed5b490de7d149531ffc3f81bcf9ffecbe6b8654e71c4751fab5fa2c3b10fc854584eda95e4e9d85b44ad21b75f41245a1d24ad31a0da031eda1ec104be35ff2a3531de2762486bf3fa4c412b478734961820b007642393cff1cd63237b1b11f20290ca19f2444b35c2dc0af34617b8a498d890c2c2065caf4f6054312510634c2bb9f8361f569ed050c7ad5865a560ed69d640c44b145a32ff153a860132c9ba396005c842715c381d7d7f233d0128c322e3ea4a226f2be2de67729cdeb7940ff3e2dbac60f878eba07d37fec47859356753e424f1483c8c5bc7189f2aaf11ab6e7b1b154285ecd245d4c4a56904f0b9a0711914b2bc763ca99f384c574b7554ad56498441ba0a1c7da8c57e5b99a408de17dd2c6780f0187980c595d396bcd79f1af29646d55956ca8e663bed1266508f22ee962abfb13c951438a25d50edd5b77ad004c5341ca78f15e410d9a6ff622a14aadf07159af8defcc2139c0f72113583543424e703c0a28fc067f9db7d118ce5e8bf158928cc63e1922fb3ca6f90bd70b781de2e1898606d4581a4dbf6da38e8d9f12ac07d99f805bdfdb3eaa77a5867dbe0b8acbe901955701ae310f229b399b28ad1d3488bbe7cf86bfbeba17f323ff08a2de806ce6a97445d46012a0cd324cd7f25bb6a05c6c61e444471c0a466c27cb21502ca52fec94e811e598dee99d8cfe46d157fadcf76e9114e246f2203553128c3037ba7fc582497f3e60f75acf2fa1f216efa5e7ba1edfffd870eebab1ace869feb9c4b99ed156f42c94156ceb88edac9ff07fef8330053d589c3058eec2e6a94a8603881869c71c98ca5ca803dd207856fb0ae9a998aa99e4d67427b9bcc656453cb84bd7be1c80567a3d5c6c33f4b5cd3c2acec866240465d980e7be4058195bfe7a301bdeeac811823c41c4f5ea552a08024fcad01980635d622f96f4c190c90fc2533b7e8f48aea3639de671d36abf538eaeb8c77d0523d487377f1f482d6e54d76525e600b83e126e3f5515243692c9a4eea6c52f15a0aa1850064a29c4e444276bba32983becad084b24b77789e591ad48fffacdecef63f2f5f499fef9b7bf2474785732e1c53a6e310b879baa0daad43ed3a69b86d2546814964bf974fd015e0a1bc7e1e587e7297bfc2653440c64f0999031432ad5a4c8c823f0961551649aab02c5d0afe45e1e7092abda1f14618af46668920393c1ebced054ec253a1df007087a19324e3d2bfb608f6374d8d77d92a179fba09f27f8e9d08d36340d963463359595a63bdadd227db6f71ad677254c949bd409ad2513caf0bde39691fab7441557395a6b6cfd1bda5e625166077da58e9398d5688b8df0459834eb1f0cf929a7f58b7791e8e51525e546c2b650af0cc11ec8d894a0851d3c7404bf344be0efafd417700f80ff01a28dd6c6f968de632b0c805b46bd3ff16273d53bda8261ea2e9cc9a0f641ff966048bedc1890a7b680dd2b32b8ac83009b873e136d0752b270e0664fde5af434b280f2d388ca0e4ce87ab0a7b109cf86b3c2f0b739e41d3df3132d4bf1e4d4756d620a07e1ad5c5013a6e80be9c4271d7668992d6fd681077ba0d040b49c242413a38c37dacbd53a91078ce0a50df0b5e34b00f8010b6346ec367bfcd4c4cf293ca1faea56707df35dc41e1d5ad7743c920d593f8755f6eb2cc93ac64ffe962e172ea74fcf41ad5d08e8dfcc8cf368bac8285f8bace084d6b1b1ffd74e7594148a3a361dd4c91c333b4f7b44eff8cdb0f2f1938e78aea7b66301234ab309071e4289149c0713667df7ea90131fea0f845234ebaa9e702304bf7a6eab0b25b950019b8b628d6c5a3e22a03ce7ddd7f34f50fb191cf03c2e64b7550a5b53b151e94950c14231c49b994bc1d9249e888f2006df2dc1152fa9d8ad11e64d5815af43bdea0c00ba7fb96db477b9d5870ac160b27eb8e4d913e8c1424120b54a92b99423346584feba2c17840df15b73c93b110e0c4b7a9869fa7132970427229b838a3bebf6ad6d919a95eb12cf4e09e4a25f4960ec39c88961a14e9d7e9a21652a9ea0e342b56da7fbe6fe585ed18e37589aec0ed76466688fe05cf6f825aa85911e53895642a0184ed76cfb7e1504844391765d05288255d9899a0d534f226128560cc797e76cb92039a6e5ade42626bed300e9b87a2c2864bad840292363d32f3916e32b7e6927773cad72c8f0bf421a8791a80b3d416c82836e5a763c1134ed8fd4f30ff0d3fed97b20ccce3cff48a96d598aab73f09d56bf7dc5f1197ab658dd1b0494a8dd3231d5d9a3028a46f04f9a1efadd3daeaaa63e603e58398bf5681ab4978d2452d436f06b08853041564ca87dea3676b61b73e850c6747352cb938884312edda2729d5c8cf49ece114de7673287c2e2583e084018f24e1cb801d90fba5b7b09ea928b1a0ac85fe382621c5b07a5b131350c8c673c2f9de101cd26650f23e6a14bc50243d3c1fe8b454a0d6c2c73f0cb2abee41ce2f08ea7e4c44ea74d7d48cf0337159013636ba5338220b2d26e4447d3c47e9125bb3549b29d3bdbdb8f7ac876b33e2bbcd152febf5e9069f3f88aad8e5f82ad88c912c8819fbe7605d376957523491d4483b9d1a314765a7d1e82c647c2b11e9d9535fb02073a0b362d8cdc5b6f63e7214db43d8fb5b8b9733e74bfac9f02b6f5feab15cbcb111ddd06db5ea7a9cd5685bb5eb854a266375485bc06e266c35d12a058b4ec908bbfc299d7b700450ae84d7c0657d7aa6fa55026d2b263fa561efd88ed979583b7e162ff146ddb806814774bb736abb9b58b6d6c61002d7f0ea1720b1b1ed832bfda99260db00c577d48897b69537e1a0157a6c559f5f42bfce7b0ff15f139ff1340e49a23327f5bd54c42465805c08de766023e9a4a4fc8b2ffae947c6d40433425a1e5ec0bc28521f32b3fac9a1f2d2b7179aea9b1fbd12bd42487ab5a763f9e9e95ff5ed0545e4eeb2c71d86bc56d981f60f23cc34de85a87782bc38a30c02ebebcef21d9851974f0251e736e9e516600304948f8c4a1f9bfae4687d6e41a84355bdda96883dfcc5754d7547fac2b5ebf8dc946818a6cebe607512e20a918bbf77f2be15fa487b9ee543ba4cb8ea6b1c34c66a73f11d88e3387b63b123b1b10dd765ab4d22431e6efb58b3af8e9d17793f4c4ef287aa8532fbfed3e8eca0124114f93164eacc6dea6638ee7e7063f441256e0e0896f293d893982d5c90218761ab53ce279250306787b0e01773ee45e45b248f661ef2edf963c471c02a952a1bac419a1f6a1579ddf3c66945d6d8c2aa1120a77161ff90e26de3e9eb295634d7b1dd388da60c35050941186a0c1c3bbde07e83a941a785d8227824f809393f6259269fa7a65da65161caa2507f1a229501d0f737c7bb7e34c7c0abe3e05e5781cde574338293fb710333ae5f404195e96efa663be8aa254b0b85f7a6318f602ad2b91959d832a5b3e3fbac86ed4896ab68bf841e1c5b74448a6c22374bd9415498d6d87abf1f2fb53baeb67c9598d62f1b9710bf2f98e750a770d4599256aaaefc2880759bc85fb8b69f4445fc7db2bcd0b4b3302a041d01fa916b07307c2cb793b4f23d6c9d363f1b19bf587bbe62200f7a4f5fc6fad0ba07eb6acce758c4f27e04c190dd8f81ea06a11b5020fb240f2cd579fc27f360ad7e65f9c4fa32e64304ae9555edb4b75a2690a30e70301257d4fa6610bd5279e66172cee08fe0417cefd9887797fbe6539f7df8981c101b3a16ad18c6ccb7a883254e7ae6f28fe10e00d03594dc2a31b02fa25a2e3c277572b4833de1dc79ccac4facb6f389ba7ec17a60b5ffba08ab2d671ee47f1141c6b685bb0874bd9880f0f4c5a5bdd91df85918965a9e1bf0a6b018113e1309cb65cf17dfe5aae8cc4a7776afbce5e200158793c22a7d11a817dc0728ed42720e43345cb44daba859aa66e0bbdbe31f8b25f6eb8eb2774da3221a413a9ddb49e90a656f62721e720f54c113c9adfd74aa9522cf2805f4d84267bbcae58333f5aa3161b87fc281a16f102a46fdde5e12525ab88875479d2bc75f74d352d49154bfe4e6f7a94d5921d5eedb58638e48a2852f23bddeaa394c7ba83ceab4ddb47236ca5cbccf695ef888a3775a6a63ce30e321fc8594a8b55a8ec58ee0570e679ba47045113e374ab6be5de1a34dc8cdf95c93157bd56897824fe1c01cc2f9dcebc803f11604770b15c90a4e4c7c138fb1039eae5ed8f56df29db21726cf1f9b4a6ec1844120a23789c786875b51fc37a15e59529b77260e71632f09958e50fc3dc90a00cea0540a9147358cf293f98964a76f2a565c9f5c71538c6f8d2a955d903bad638329a910f31c1ebadaec8df66d416988d8c67ec1a30725b371e7770189fb373ebe3c214b5f81132bb90028a2a2efdee65bf71ef03767788f431e03f2450303598f9049c4b25576bd506c6f686282b6f71acab1849885822f798bdae8dc7e5b172b103eed95e8b0ff0ff2c4dc25bf8e1234f15ffadbd2b45faeecdaeaf5cfaa5bf7995a1f571b3ef0dbda852ad35e22f88ee4b31c4d5f62bd938d49be01d676cfc96b01bdc00ce3264c6e5190af2757ac50fb756fbabad868f60b1876d0aaba34eba73e6af8da5d68e01cdd855bce8df20540dc9af6ddf7c514c7b33a4761ee8c607c35fc1faa42edc69f695a85cb013cf6de6a82eba8439658e0ca2ab8143521b3639c8f3e3135296f3be700c7575e28375ed5d0dab5cf5cc48f105d7d0218f4b0e64ef471281f09d687accbe6ef97aca358ee1bc49503d4db0b3aa9e394b9c1d91c3c88278fa6c1083e64c409feebc26d603e712dd07736829563d415e611688b38e7057349edceea7c6974dc2c8898f64714ce71d005c820bd5e8b4dc212a824653fcd6956c92582e02dac124afb84c703f0588c623a84c1fd4e2fff3cedf64fbbea5aa85f8be7c8b4fa6829b5834318cc35e7fb7786e732073c06a01a32fe2d4c0e2867c208059003d3ca543c275b55f063523107afa0953cbcd0eb53c0bf9e1b5831f198f0d66102944695899113fe1404181d303109697bcacd7e000999c03c893ded4256cdd9bfb300348827acfe71a5ddebf15592eb680045a290d0b77e8c23b220f0563f3c1c9b54adeb88f714c99b636d02e489c257f8900a29fbc60c40a5f86ea118ea9452b127c20dffd82ab906260235084d0eacc2214bf33a225d66dfd6746d84a33f2d540634d7df01315b382e1840e2291810b8fde0f3e38a9a2751c3298d7c618c4129dedef75bcce2a89c408a3a7224ccdd1a203ebd03d7f960e8aaf08432cd8dce1af36679fdb59b2b7b7c35c3b8ca6522d374714a12fa3cbd346ce78c722fe51b88acd3f7f03449f6716c3370bbe4a39d28f0648ef761df5f69eaabd4028ab1d1b4e7ce834c0a2dbdccce0831530627ebcc57054b99f686b205e9d8c80dd7c6a5bbafd2c33884ab01da3e721fac93c93063ce4813be15f029236d29fe904b17d8a00d6972a7f2d2dc7a636cc14941f72ccaa0f4887b2cbc421889b1728271569847eeb704ff2320ce0df8efcee407d82cee3eb4bf08c987fd62c96e086cb7dfd9da1dd6e4ea2fb439725b62d03725762fb86e140b153395451b9ffbd7b01a31973c58c454ba9a50a4aeafff2d84f8b89ec0fced41731982d8c70be2ddae9d1447c621286c85ad263838a6578894399fded44b1a201974e7e9021ea347e55d952832545d1aa4b0516428424c69b93aa9ae2a7cdac2c42c34fdb85324e51e7c729fc4cfb4e41491a59e19a25ca1d6caefe0ec219505941abbef070bbc22c7670e7b185ed61bbadbb288dddb9c10c7d13b966cfcaa67a9c0a760444ef1878d28ac918703dd62650ecc165e677592689d84f6bf6501d1f8e60e93a996666a8308301355a048bf2a83e4511d07d7c9983259039fa6dbca5725211a4129e296433b8a26b89728a3e8e0905920cbb579bb3384f18de4bb0f92f8cab2ede194e62e6eee95dc87424f8038fcd0a25ee8106d1646fad010414acdceef9f200fa4c61282ded32243d171163288ad680d6f167b47c64da826253ee45700092600ece748565202b3b9b75bfbf49cf455eeab55cb211c4e424d789df34919e922e942186a499960aa9ce3746b1f0a3c2c00610eaa07c68321f3b504b8dbf0372e240817c359a4144b7a23575d1c98e2908eb6f0242376377761d68a580c41f93141f80fe56006372510c1a45de52d5edcef20cc618276e078d2c4dca9981cc326e9148b4aed6fdf37e5d0903fa117576346ad50fc7ea83838c380c516d313d446b5e3aa5fee35f9800c0cebce3ce62f61e007cf93bff6b3e3917e7880ca61abc40a3e393142979f34d1f339684b1893b18aa24c46bad5b4e5a2948cdbc9d5632660b49e30bf08928bdb7068bc6e7fa6ee76d804b0c3e9fc89d6cf55255f3881cf42cef8c2f7b88d196049fee1da5c3fc863bafc6afd6a252c22465683e48a7c4fc1ea8d6b8b40bb807bc0b3cb452267d2b1bbd7312fe044a5991ee65294c07c1deb182eeb75802d57558572d88374930fcc5636b23492f5e0cc7511fbed245bec7e2774c7fb512badd93fac6811bc4e8f237257b0d3171f71a8c78f0e069e13d61e6f1d1413be912daee54411ccea5e9f3a5978bf0be2877c7a29cefe1d428bb1eb2bae0a5d94ea5310c1525e5c1a80b8647ba56e0eac548239b4751a778d3e2440ce5d40e9a340f52136b28e1807d423e197d633235f12f3c76a49679e5cd30c89ec779461de423b4cc27caf1b20e3cef8e0f65ef1fd9181a389fb885c9d725ccafd3204d43a528ef47f9995449066f5797924037a4278987b0d40b816c90a74b6202f7863f94a58af0c7ac646097039eec8f26c0f3de0ccaba5bfa12c06b06e94ad11a0e1860408ad22c4944acaa114ef0d108ad6f5764de38e8cfcac1c1f2794f440a8a70ae7c241c99934a521cd72f2863b26081d7dc6c13c0b2007d09e18d20fb75878428d4c6ec5b7e7a3aeab00a832cd46a244ab6042f35ea532505c900c1e951f73f24de267fb5764d5502d8332935992f5329cce4f5cb4c2a2e07a1b1a06268949b5824e5731f744f447976490789e3e2b5e68a14f0c64264074d7217d17f13f6ac6e11919fa1c310295dbacd60f53696511ca0d26555db00dc974fa30ee74620be0dc2b46ebdca67723d29c4f0efb7760c8e852c2cc948b3ec4404d9ead5628ff0105f6ae00a0b89d4c3ed632a2526570922e36212d53e9c82477805a980b398bd9d65846ae5d0ac42e4b15604b9cd0574ef8a4a7a54876414363819835a053396bc4460310278e2b60b482c06728674b95826b4efa5e750a66dc101d8a9fd47ccd45a2f615040469ab5d0fbcc9fae1913f3fb8ff14f3920865db73a37ec132b90d84a3066edeb073856ba6d70af31d6c673cf3c8fc8690c0a615f73928215b9d50df8e1e256b0026a8c380416de1e275b1da5ab63aada938dcfc5f1b05b57110cc881a9d4d715e926e966984bb17f7680b4a2158b84d7f3f5ce0c0179bfa49820fe5778ab0c13aea64c7b62ce8524590798e0ae7c3c3fcd04a7afbc327279ac107c477edec913378918aad05ff8a536518599a1ec380d82814e6970dccbac7dc6efcc9073fca6ce2219ac653252b56179a4cec9b667ee8c2392e63af52e78d3757550855f104ebc5954f22910434dc051e41122c797e7f029d761d349dc4cf70786954cab1b2d69b7403b9d3703df4d7edecb08f1aff9dff86fbe5fbcfc3b185e0d95d3c06a37a11e692c8387da69400f86fe9920ec7615d2296e62d100ca394b3765fe8aaccb34c86a58fc0df64f3071610c32d8431986945849ef693f4a4b2e05bda489fcea001ffebe93175b6c7e177513a7fdc2bca447e862c9d87dbc3577358d651fa75b7bd7df510ceb8f0d6459b76f57205b1ededcd67a305dac27682749a481d0811118c420d67c8f1c637ae59571535f9d7035c47e3323f525cd3db70f4a101a75e4b6fa308789b28e3e53fa723508d21bb1638463155f55bcd7c6554e1f5f565ae8b7a6d48d9f6ad4d3d907f6e35c18e422aa5074c02c8de8abfc3d67e7f66a0f8b0cb6f826fdec1dffc413051d956a69b76bfdfaef935b901b93e7a47b28aa407ad66fa5f1799f171f725dfa1bc35b9051a1763f3ac9406c3e67c8bce6c399c2d5fd9c4fc0c188fdb364497ff01b0b907d91e1454801eaba95a8c39c3c116f575546dd69f3978e1c278c5af7e7cc1adee1f94350ddb652327ace8799f07ab3fed95077b1f2eecc628af114877de3de28bf93c51b1aadfc1cfbde5a0849062bb321c82f972b1c57d1cff118206a9bbf2eef3e5cb3f88dd5cf6e3c8780e3289c353ad09a0120b575345d958ed7a0e1193d2efc6a49942ebf299ca548804890c02e9390758361e218947efcc819d72142f7eb397b833b0b92509e3650ee666fb3823600351e9667b6be42ec0c637e57c062687355030c9987fb3a2eec63b4877e036ec90e157a9e2849a0c02a08b371e488373be2fc60ee84f6090634aef0a01076b7d2347fbd689b2d86065613aa572c880d262629e6636b32abdc0632824e721b1b7345fe98aa2ec4e850e5e792d4a8caa1f03489a60c6e338c239da32ccdcfb98c8960fd85a36304d8a98ac64829d6a92567ded89ad7de43f9c3b05303ed66d5f0c4d9fc924926c1461e494e852fd92c870df5aa8916952b68108837b493a76b4dc16b4b0ab327266a24589139c2cddd6e0cb15c3a743d885e7355e2a860d78a12d737d40c516093dddf2ee61a8126e911b79ddf9557384c5d48ebf6a47ed97a915cfa62902ef208f3bb44af9b3419461aebfab339ab6f67aa2dd0811f8b244cf3e426cefac0ff6cd786fd49bb467682f9f638e8eb0bbc687897d3318d5a89eddcdfa2d1d0641a57dae0328e4945d8af93e180b586c9f00f5a2a6e1aa8d77524603c796cfc34a07867279fac9555bad08ce30cd5690499d67433711f98cdcd20a5737a628753e9c9501ce9074be655f0e5c0763d7617e8835748b20a4ca88772f5f6da7ea9277c95af8c846f983f47b5652833538f16fbdc9e16054b0134a081a71f59ec41f4ed88aef7c078541563353c409adad9faef186711c7c396c0f0612cde3b6ab34862a920965ec6debf85d49387fd15cb41a31646c3377da0b46db99b68972bb1bc7e340a3f59d68c297d71e8454627dd51481f11fe233bdec4cd9030a7e0ed7d205b41982f6036e6ff9ae7e400b0a742c6d233e8a8ba0ed0eb64aee63a1f844650548f6067186c7bbedfe6b71bbf65976afde5e2f69abd2e8e3e021c70942a3d35348e38c258100e90109a736c728519d5c939d6929f36c9d4587561f12c53d71e6631fc9adf0d70660a6fcfdfca80893ab8292730ead4f4e487d296a0b42334d8522881b401224d9b4472297f9369440b23a7cfa895ad491e6175a06a59987641bc66cfe687e2fd0a4bb3190b8be88d2b662bcd08c50edefcad07ab13e2726bca0d635a42d12a000204fd1679fa4f42f317d9c53b5c804ebe415ec5ac9bf105ab74ea4f5ccc53a25c948c8d79dd51323d1d4e19c8a13d7ac734d40e10136b00b37cfe706331e22c42c055f0b64e7be317834beeb91118d948a4ebc9c451c456934839f9a830eee5575a5201251959459b6bb5a2a8d4fc05231ffbd10425b2c922b0915f1b6fff20fdc0f4ce749898aeeed073e22a211e8930ec1eaa4b612696741bab6719411f46ccdf785420097efeb7c79a968f54b215a703ad92ea14a52170711d7b3e64ba6b27c33bd9702e081e8124e54736c864f5ab73a587a047f754583ea9407539e4de6692e85f4ce2cef0961cc420205a630d537023f85a24fbe54657280bbc49dad4c987d0b57d0d145c286d8d91bf9f0a4bc027afd91655819b3dceb70201817bb4a6c39409c968cf0ca4e8ce747feedbc90f01e04a049fbb5a31e48b45e2216c72b62fdb2eedc4e6d839270c702d77294ccbadc1ac37d61a545dbe86735b9954880177a106f89c14f375f6ae98e8f75fb5d2bd134edef9beabd7738d77cfdce26a3c872179c9125eb65b434d91bb0f31efe16174ecac6c2aa0f0967d32d4cd9a9a2a5c63213f95be709c1ce623f3c1f47c24b7a95b7962f2dd5b7be186117e0ddf61c29a931e6db263c18fa86982af25cb3a78b8f9585e453fb973db4a033b31ebb1a1679c37288e189c941e9fa6f8b908a83fb6a0225008f8a05d1df234535f8e022d40c4db934daad51691d49f8e794739f2c2d73655de40ad331e484d77c5cf80026747fe795cf2d09a28d6318b6240200a46786a6078232cc9730b8ae9995492732e14bc2ad135a4f17d5f59839db603f0a10db96b2fbc301eb6d620dcd78d149e48e916d42806bac6fb65b49c6f3519e9e34811c3edca9f15df78b9e8ff88962f6ff20a12ab4f43cfb2af6226c08340e6871f08cabf60db2ab9aef6f62badc81e2e7d0db5498ffcc2613fd4131e1df5e96dfbfd41d314348c8df86db6b410ee39abe0c2cb1a3ab6445d067ef212ca2fd919b95d4c598ba8ff0057c43e6d202c5db17f465f34b8a91c379453f7491c74e3ae1bbe9045942acaed76dc8dfecd599dcc4ec50b7c4f4798d2086e60e083defe9fa770acf8053219282427e229c4328680b3971447d5903be53633dd33de92e9e56947a8bbf4302cf48e973c0f3573c5def1d409708066ad465fabef58330ae5549b693cd179d55043929df2333ba36d701fae01c80d4775a8bbeee8fed64092f31bd41bd9433e8373e96002d677dbfe8f9eb9bc02a8a2f47ba5a8a2faa89342760b2f0e212674a4c74b3165f360d12de7c68ffe09d552d9f34c327d9b1dd8ef2842c50249f41546ba90ef10c3407dc8a73021b5392e52b7f4f37d0575ba16a0da6cc6734f26d7c8528a77dd9b1b20f53a5daeabdaea5e7d3b2b7ebd5dd133937fdbd0dc3e3d63f00b6d7f5e27d4e0f08ea6894e8d1102813ccad50dbe19829671073a1f9251d5158b043545050db5061a8c119891ec36cf398e984b2f4a5301a482efa70541bed53d35b2db4caf5aa4582b74bad1053213bfa5465143a8604d640d899ac9da70c06b0f9d6235173643c66abcba88f6080e484d56ef22ff7f7ca17c883407a7bf2c95343c551933dc1f5e7579ffe6aa88d1bdf8bdd2f359bfd9f26c3baaff2d22e6b9dceedc34b6b457bf38a0bd72e62a72c8aa6852cc3d0758e08646efcb71631a8e79d3ce763afd1c3e85ee005082a9f60ae40259e96aa8d5a9b5c0ff2c2dcabf1d0274ad7ffabbbcddc5f5bcab247a4a7894c02d77a68364732339c0be5f666874d404becaf02104dac2390fa628f9234edbd4b1f4351a13d5d785ee251b56d3b68fb85a0c5241040558d9b482f90d7398d6c17fd2ac2db98e9944053b2ad4ecf1a3d1790caee28bc38327f960c06024d08bd0924f01d8dee33c5865b4e298668f6aa861a7247f1c1662b3776c8f4d21ae88a68b67d6b337bffa3fc72d1263c7087abba41a6d97e45d438a4b15f54182f71db7a6639c5562f8262b50eb08cd32f7157402cd97f853cb391451eb4649c178d8a567484d73537f20f7594a9455760b6c351201e595904caeb7b46f98dc45ba36b3a2b0d5207484e2807bcb6f69a6784b96b2a3d60d489d728b8eecdda4506c5e9b745548e340375b60a51d648f19ebe0c35f16060a3ad666822d1dadb7ff092616927fd3c0d141fdf84814b2b579fb5ead257e77c66121d8c20c9bb8d7c7d4d0fd5f621fdfb8eb4b4ea45abc03f0fc3a6eb4f6e630489b6a9ab90776878e12c7cfaf371ede73d4bf3b5be17a78bb5d7ca9744244ab929033eeadc9e8662b963b111a08a397de28da38b8b7db89f67af170c13a90b2cb96c8bbb7a1544e3a1f1eb0baacbf4ddf7d1f04778347e3625ab0e7b3541442ef2124b912463a787b46ec439d7ab7dd732ef8d5dd58f3bcaf945c23d87b5181f8d4a26de7624e32bac0b8760d5b124666b1d0019ff896512c46fd61f3e9a478c07d6a38446a8f8bbc152972e93f56f2b8590f63128afec45e185725209f14a3433b74d77cc1878116c7db2882363420a2c3944328eee0a900f535b6713b468007938ffafaff364cc825da824e4a62291dd45755cb59a0b1a90bfc50211d2fd82173fa1edffb5974aec84a88e9be17eafddf89203498e9a354c673a8ee01cc0986c36b97231e49342dfabf37c92295987666434d2667105d45433405082ebb4520c483ada53bbe6f17f33ca3670e2d3f6e7fd49ae44bbdad2f2a638ec5889232a94611ab57869bc767add6e2d19e5fa33f7b8d93a13d75458377948455590b0414bac327f9fefec7ab03a906eb0786b7a06ae28d4843d4dc104dd5842e233e0da3632f0aba504f5111817db2e99de37842670fdb1050412d98b3aec9ced5c0407ce904edca5fa5bde8c1beb5f90affe067ba657847e8a14b540b7993718d10764d5339530b3276ab3b15b38838cae66740bc414de859b26c412115b78762f694fb497cbbf7c25aa45489fff9c6470d2fa7a6a63668b803226fb63bec7a1396192dbaa2ec864ae3634699575f5777c23962f9d19999ce5ad51495c13cc81477d2c4712d96f3c21c103987df10f90a804c8290da1e5fe74e902f8632cfe2a03d2c18dd151c82fede3644bbc93a0c6be424b981047a2f107edae5b970082be3fa1970622a61cd54bf53dd35b7ba06fe6b2d67160118e525fe4bbfd03787976c4293da26f83a671ada272dbc99db53037af2293c890ae3f47a848bd416eaba58710947f80afe743890be35e7d386a88c01fa034b27434735344e4618b7ac80d160750516b37c0caaa4f3cfb83e1db819279ef14fc7e70fa12fc5602524fc58cf27d544393edde6808b34f4c4025d51b5befe450d26c7959a8c46dd82fefd35014f4ec5d4f81b1d9d5db85a842357cc2ae139117b3c64971cb18a1215d99414125721aa76b9c4c1d859198a2e6251aeb5a303a6e3de2d49615eb91fae1eeb1f0a3f0a164aba0b42133ba6216e172be4f12b5db6a568500d105c94d9170cc15188f7b3ab7a5a17e67ffc60bc4792a1177451b4e308c133fa80427cd3893b21aaec18dd6d772b54c237e2bcf7e796ebab12b22767dd2700852ae3b668ae204a2678e8bdda93773109efefdabd2e408c69886864d6fe1fc0f6e8dfb5073742ac62c5a86aa3ef464623dc685ed22555b31b2fda97f8f7251c8f5acc4af748a6b5d367a265cefa5254ea5c272f129059d62a5e31dc64676c18df2a1c6d1a188f84f4875bb0caf80084c8bcd4803fcdf191b40b8e45fff94b34b07116d625949f4f9d13609554dc3f142ef05d6894e09c5130eb2795bce65dd04f11721e6f647d6922fe03153d5a36ceaeda2f122fc0475ad3b9f8b8793c92b6fe26af650d4f1d2ffed5068d48beea97a97f4d59a72b99735869caa18ebc070041de612abfe2686a81869f351103baaa4476d9239104717df853ddbfc7735b387da0affc99621b8d902bdb6a226c20616619977a5079eff548ff6d1a3990cf9c994ce720e71d44ba8e0a2ca82e070ea15e3284159193391206aa286ce8f50e064eed923a6380aeb45104b449e5102d794bf799b480d2abed7feb6aad488b76d0c27b6f39de3432e83a9c571fd35a70b528f0c1663f747e3ecfb148a6de15a4602e199cc052b6f06740619709df1480b4b3d61c3882717521dbd77955e79b9a409cf08ce9193274585997e244bda8fcfd5e1717f6e2629f5cfa3550284b67e7b3110202a21e29f58bdc163a5f6bc8bb8157dbc4f34ee95853dae56f98db6e36d9fa14d86a77f0c68c138adee34cc5dc454fc76b71950c4f02abc685ca2b3315ae4cdc6aba0166f783a48bf2c76cd5fe7ee1c13f4b7a877d92ca15357a1927a89be24080b2dca9ab9d03b686336c845e130703c2d262a5b546cba0c9041bd674d6edc4543297bc44df4c325b365f06284d7ea69a0bdec1ce2738b25c34b13e022b9feb87c576251a915b482a217e2f91c60f37ad01093533071750ec5daae721c315f97719a496d661c44813051ee21b89a540dad982b434d44a9a7d6caf9d0384b0a70707f526ccf7695c48c5fbcf74bb049781152443fc1b2e42de8b1d2801d0f8177955bba8a69f5fd44e490f15b1dc44fa349307f7c9bc261c05ae2b77c390bb3171140eab2bba16388ef056ad333f42b761b49dd9a6410b033bbb0d5f75606ff5bdc626960627e27906632650cf696cc710fb6530eba4c8be90439aefb1863e0c70ca3e8c2c14648339891596ac50bb5725c31cdeeaf35943a8aff282224ede96cb45aa03b9480f91f28d3f7d00219ee75d18385ed4a512d33b049bf10550add45dd40180c7ca8f487cc7607b712333a32f368d696d660ce126510e9473bb1094845ef8b7329a5ac132fd8141bd46c0331dd30497c9c040cedf119343db5eb704242d61d7436247b470ec1b733872429fea8a31bbe0297e074bc5bba9d8df009572fb4e627661bd569097a927e3da1e83622df545a5bfe5d2ee7125274f123ac4a66c7f4f284d6a4403d8a3eda1d23c4c66ca08ceeb63802b05e8263d86f1e3319d95970e6f6fae321bd645a40df4da871df2526f8f60871d770e160100af38cfa855bc7a22e9946bb7747afbf8af7a4918a14b6a28016984e0fff01fe7582c97484e3644ed8b45ef366b0dce807fbd1c8499501f0cba52c54584bd651276f0caebf306366bb5eb147db749c9d7b173d01f8e53243c494623f770849834121439447e05d3acf157f185ae54809dfeb18468330dcb15dbd32741c9a7c7f075f320bfbec078492b6c24e37c73e8e05255e7f33fe1766abc0622ee1671ad2c992611afd0afe5c6c73968375bf58c73aabaa0d6412c42158184d524b026ae352336d003b486c8d5cba6bf9eae15ed34a9c371114e435d57714f10efbb44d34b6350b59db89039f60ef888b41dfc0f30aebc510fd1d70754a67659b58fb74495463fe09fcb333a37e33862775c1c1ac5ab8a21c01b8ba092a582e6337113d55420fb775159dc68829201570be3becbc4dd5426dde3b6cff56992ec19871346d03d4c6183d3f51a2c208e364cb0a205fe5dcffd3e5bf782a43cd306b0165846f452acad27eb8fb83dafe7e5528b3869eaf162c2a992491544ad4834be9116d6611e7144a284bf76fecf2276ccf43f26295584b7fd22eab8074a427e240735fbbd959ff030250d7492a910362c8963e6afa71896ed1f59192ca58ace00d01ffc0dabcc88e6776823e343011501c189993ad1f6fbbbe8fe87d6496e75064f3ccab6959b960080737dc00a76fbb800af8b7b90be273f8440da1b86d8cdfd98950a53e44f7676c96fb42642bd20dcf96e3ffee022b1e28cff89d3438379c1d6273fdfeeac4ee2f782d4e780015636056eea6837af6b3ab68bb10c0be3bf02248b8b2d06fd843bea1e2c16c73518b00f91ca83eb73453fcd4df3a309b7d376ea5913c257a9b397bc3ca497c76350fcf8e2cdccc60b0707020cb0d60b52a8ccb792a1913390963a07e44a21bf61aa4fd8deb3b8d7e332d0189068fb03df2ba510e6ea11085c4a54f0caf1a861d1da61ba8d2ced71886a274a63308340f4c823970ed1c5671b0be09bcac0f7a3060d76c67aebaf149efba0869d4b211f1099256083966c078b3e9c1259792266574693ffb41f798c5eee2f993fc92c05e142eed68333b1a6773d03f2389d3c8b1cf4edf60e718aa56be1d4f7c0b9ea49ee316164809f14c3a5bac759a25860e0340d337a0568ba4555e8cc4fd3a16e0337b70ca2735dbcba79dc0fa596573d57eec1b9b643f845510821aaac7f5475a146bb83e2d50ee30378ee34cdd5d6fbcb7fe5a9a87d41118424de3574c4967a9dfaeaf361e550926705db9a4e947007cb989b6b102a371e9cecb0b9704e5436d855d95558aa4f78c3c0d6e66300fda7f3f09dd986618f81e2f77a7c25aae6863e27a837485535f81cb9927bfeac1f7b5d603a843856d5b9d6d7e96503d251c6d6e63c16d408963a3bfee67e31c5b6a8010869f9faf3456f4c2ce3ea467082f490cc0ec9ea5b3c050a7e505c43eb874e1579e4d5a5b766f1a0bf6327311d7287d1f2bc7f4d6de7bd00240d5dd929bcc3937a6757832bc64520a124cd4bf95e41acbd755bd5e7407e9cac6359d1321dc5d2c4f60833830af6a9c42a184197196a23190760c4cd674d8afa7220f2640204b6ca196579b5afc654e2d3191a19652745e55f7e154c2ce232f4d73023cb07173b9850a505b9585fbf1c1c8830779e6b8df66596b635a5de48098cbf4fde4bdf67434bbd9651c00536d81bf61a22f0992aecc7a77b204e693e86eda13296af7b3f87416aa7b54fe07c3be9fa75f2be3a8115edc6f342f288701eebc3aaea562154968fa312ee7a74303e14081128d84341165b9fc8d9f92bf5c6c9671971164c959d710b66c0f47cf14f89a07958d5155980e449f219b11a050b75c4511dc1795b17504f9b993e716481104c31f536cc9422815d2c3cbbb45f0e60866a159a1c8707fc77127a94e0b133a746a1f5de98cdb125adb3bb9f7fc74caa3ae7bf445b6cbf7bf3ec98d5df1a147e8974e1dd83ecaf9b2df56ed1e22dac92d7d592c21e4bb0e6c6f27a6facdd252cd539842d0841719b812f6adac381f7cfeb7ee55f4f0ef4a376816d71ce765476614dee504f910ded467294477fffac8a69048d59df18ba2ad7363804de820e4512df6ade469974aeac887058bbaf7a95014e0874f660326e80d13d3e9ef3aecf8d94ab27d691d25d29ad32523d12a84b3d3eaca6cef9b26ac5e86f454cc66dbf4798ce28d4ca194b905055a11d90c7fa2b182035e0223c4e8400f6996fad609feb2a1a58816359eeeceff42c9a3332ded622f16c532646b578bd6cd750386977fce4d97d1350eca1aadd2a63258e620342595a81409f522fd6999497afdaea22615b1a1fd760a61eb357d9e4452ee271b61b19bd7af0a52f1b5bf86513093246e07d5dc6b1150af8adec97ca50cbc6fd6ae44efdb6544423d8492cdd44491eb7109f51bad6731d394886082c8f8b1efc05baad60291a8649f16c35078843618878c6e9dd03803ece1923616502ea47c266f9b955cda903a690232c0d93f7a185d523d639d9e562bc50de248da30a80740453dd5e3f9fe7a509d18dadb5bd3517004f2a26be4d5141b5562dbf5ae747995dd0bb1b8d40ca952d982234f0032e2e88def826074a6287e08e750991bacb4aa1b460486adaddb6fcc7a8540e4ef0292c95e626bdc0a5bf86eb5cce68d860068e58a7c22ec2d32f85f8c714ba11dd6ea9f47d43d3aa242123bcdc46d9b9d0fc57d0f8cf55672200e5ec387d3ca1e9f113511fdcdafa212fd4ee3f5f44dd7d11005e238e7d9d2a4c1ced0311c2128017c26fc27aad36d0b309db1a4c48b48b7f10b8d2153d3c7fc4b73c75e9ca59cde72f4ee7116758b6a9836d2d0816ace1a89f8b53667b35c9f324b5f8089308c80862886efdecf3fa286818a134a0fbe53cb9ba8c09017b4ec02dd71cfdd2299c0a324a62e72b16918936d8e339ca4194ec5c8cd2ce76de83deef14710a564809514dc9cd44b1ff76ceb998f4aab474d33511dd5064446907f341e10571d51c133eab2fbe3fe60c5f996ef5ecae50752e4891c51437534efd8e9aef8acc538b85326e7b681026c381e7359f3bde0f5753030ad5ab49968dc4f7fbbde336234af76a5ce00da0c63255649d5f2e959fa9007664c16fac7eaf1888ecb0b545d9f38d188444ec2bb7765560057d4963ba13d57faf22ca5b1798361e3345f9ee3779fb397dce9ce444cbb591d36530563e557d8e0042a11b846445b98453134303423ecfb2f66744f8d153746be0994cd479a90de3466d23958cc9808caeea4e10598a33f338de22d8ba86a44c9ee95266a0546f75ca89bb7fee5495879273f34721b15d381d2d4ac03913985fd4b13ec3ee6a7bba850f32f713acfec828982e9e99289a81780287ca51dfa1be5acfec91e32ea4098ed6c015a79a0fa472aeed82edc7c0011c99e445b65e48a5bdb93060a9e467bfdda8f77400518dc3cbe321993bfd21b01ee0c3dbb09859a6065b760499ba206f22b145db9f7bf4bb3e7c26cd70b63117a3abd17a19c3e47c2c6dcbfd09a319efdd479afd107b4edc8f5fb4c0deab5c3e2dd58007e33cd01584dd4bf97ad7c66796b259ecd1fad542a3535f59645a75677600b0e98001c4ad0e64b4d87abe64d9534cf8a6033a813262039ee0d6f79eea7996298233c831af548f28c3e5bc18c9bd30095b2492fb7e1aa68fe6c1c610702ab8027d1bdff5e0b632aeab02cd2fd7e252134a62cfd5d1679ab182ca156cc86d97bb43ed0c758ef5fec1222843f758cf6eceab8703181aecc68a0982f34abb573c316f3094e10f966ad1687e408a698c1fb2b391b1057d57aca5db38b15413af96c1775819df3c87f7001e35339fb445e0cb92e104fb7fd0b33bc49e6786c31c87c21640b8a6f605c1defb212b8dd48abc7baf25db93b3229ae196945e59b3899af326f565f4714db36a8c2b1ee77395fcf415f6cfad462a63325528c36a708b630d08a2d20ab6e06c4ea9c2a32d6c899f263e9f245b3dd5d7af3aa6fcd3630d57e07afad35d2974eebcc2b4c1282f7900b775789102a53cd1a4bffb4c636bfce6beaa955bae35e16552d0a4114ef5126974ea2615aa9c478eb745edce9c9e57c680729424aeff3110dc54b374034cc2cccc8e7c07763e70dbb0794a1d39f19b27f1a3c47d08b7a9ae6b460969f513cdd8abdef1079176330b611eaef033b96c3d34e802b02acbb3125d8f9bc689c166bbd028c9b31a7491a5ca96abc932f98e6d4a16abd6e9ff9c30b4987afac0ce41728cfafaeb3d215ab92b7555680e41477cb9117418b7d95ecb7dae23a644b7103c082cc30c49bbb49206090c9de59d3fb0c9711cc6ff1b1a31b5f061c0a363bc432500191a5e11d09527f5d033ce76347e1bfdcebe5181078490a044769d7dab14869f84f925aa39e8a8ca1eb4a1e6a9f47d78d4b2b8951dc4859462d624944a1eac289f4cd17540fd4b4bd8548e51550005a8e76eb35c09b51afd17b3951a194eda57583aaf61a7c4c165cdb7c741b4f7e2430b7b9359aec89d50ba47d95d5fb3e87e8ec8c3dd23d334e290336a3acd1ca364918ccc88c636a4572ea77748a783432590e8b2982fb01688438d597623a2505e42e3372c3799d40c964f8e98a66de9aca4605b02f3695cfc993d15b2355dce3e01ce7710e5633cfb2b7069bdd01bf043cbb2994795288ec889adc0f7ddb7873cf47f583274cc29fdff2f595060400136d07da11a5bfb5df6c4e42f6f7bc7dfb45193582c93e4cee41a9ef8f4869bc62efd2a87308fc195b4059243fe0cadbb58ae1b0be14396c91e282330c50f4b60819551219cc876791c5fe1f3e144593d424fe209630f0b6ab60e9223f6818fd3309871d7db1c97187c21b4dd3a1fbf6958713ccab32901f9d770b20e51ca53012f60b9c3108158cce4559cb2d3eaa4261276375ef09d93da52c2e1833c1dab1065481bcda8ea70b0c4a103ecdc5ba13693f05e291ad11f0c79d48f5447f15814396adb4eb3ac3f437600ce35a88add2bf5a3fa86cadbe87e6aeed21b93dd08cb63b1267f049c915746f34be3b10c434b04642d809b15feb0e2d43a55574736a8e329222e57d9c7bae84830b8561318c82f7698abe152dbf5a689b2d1fa626f2ba7069e831d2ab09ee4904478a47a2eb6106ee8cd3a6db797918a5400f9cb6b07b41a18dc1b8238f9c12a9cb3cfe5a64d1dd0c8a97a74137be7f9c77e45c01e2597302cd8129ff83eaa8e97ae129698f25071146d80f1aa6607df9b8154aede7895ec22e5707b4f93f45fa41fe8b863b5b8e4483715dd896b7e589a1756da4d1877cfe3cf8ac46a6ba2494fd1561f87c4b55ccb1996e0d0e78b65eb0ece9b333a9d05effdf8d9bbb3765ce8c502c9fcf8ae2fa1241a51da172ff174c12e06edd326d16216a434f5855dc345a1dd0a7356a4409d1ec57dac27f5e046c2ea60033a6469ec4b1644189684101566c5ab62daf981f69ce9d126b84a726af196baf7544a038950d1369a6557c995acd57e3d176a1f860c9be5a732f8fe91d19b785d4dc431ab39de54cd4edba31220cc69d818517f25f1441d9d4bffd53af8c84feb074e06afbec2877a636521f8cbcd572724c78ab3ff56c320e0f9b1599e62f1dbd2efcda02d6474fc4ea0cf39df002b11108a9224f955140b246f1e52cd4aadecf5d0e0aaf1246747657982a272540bbc2fe8518c6ee6ae8aa0d4d7c986cf29cf97032fd3d86e70471b9bcf67a3e4b467727d59059ef95583898c330514e02ec2a042bda66e4f271a49252910f3510f7ac6b94a948dd6cc081649f425da57949c4edbe9369926e093978cce94d398f8b2029b6bda7d9f8d4df47bcd57709d502a6d357b2ac767dcc422a260f0477d0eadcab53eea802fdd5a624b18a6a652c4fe5431d7e4f95282f6fc2276e3ae823968ea0429f8f6fda7cb55d0ccf16902d08fd91e836715006ca112645994e81e4376634c3d8949d25f53affde76e8fdfd9fd61cc03afc9ef0020d5894882b5f4a60623e35efa9bb534580a67dc8f959a4c056955125395d4285136ab5d552c991500dc93836b57d399547dfc40c311bae3ba7357389692fe3e2887624ebbd809dd72276ace26f32cc7967386c7ee8f75aac8d61ad3c9fe68de33310f1aba0cb839ead1df49d8ce55f818ba4246fe4cdc55e19a6d7a0a0d885653b7fab1cc0f4795905fcfb5307a0d4bcf6cdc5525bc1400b6cf61a547d83fb58b0fffc236facef3844a19438ba52f4c118b7fb2ff42f405e8e95ab0543c6101788239a626d3ccdd8b3fa51c2f7391b25f0452387f1e3d2fa5eaf744bd2473d90990be116d63cf723f1e7184a11466c3e41dea9c99e19ab63bb0df4fffff31e0626a956d967b54be530df6cc0cfc30d475036189b8186818bb0252af1c5c68abea91555f0cf4ad69a4996dea325bcec096fe955652d029db8653fae6d21f3b60ef1ded6b9b8be13ebb618892c8fd4a09e8f4da32cb638ef0425b3f4bfeef23945401e4ac54084b7063fbd2df7280c73a1c2c4d4272467448bef9581a0a8bbc5e41c3d12377b279358a997e2d92468fab8d6e6237fec076727fbdfbaa8468a5c5c85bc3a67baa8d31ad6c92650e62118b144d1d11f801a59fe92149ab23df484d18c7520591f1156928fcb11bbfd7c3b5b0e3b9937a37395421ab90da88c8d336e70a93c3423f227b3249f004504b46ecebd46817fae1e67a7bd6977cbc942810c233574177af4293570e9308130289df41527d45c5757334b5c01b624bca2b736c06900523b8199ab603912029769bec7f1a6e1ac4005cb61c2643120219b01755aa409843ff9236f782c62568b68a43d66a02e55609abd413bc7deb0244e685347926b7b44f4277f41527245f237c0395f55983d296a5ad1d557cd61f1bcdcdceffeab01692ac9ede7aa6918830a0eea9b4e2f5a42f63a17bf65072e7fd477de199128a408d0c0390a378e7867950965f22633ae69a9f8e0f65599510ee79717f37d8ba5835d4eddf53269310dba5943e845d69688ed74ddb1d3f8f44051e4bf9d278471c967ff12acdf126afe414291024ef23b5bd54651598f7cc86e83e351f8a2c30c6619173c29cd2b87b83c0892e3fd8cc0f16eba442edafbcd121b1c4c333c71cc055a04c3c8db0c00b8c328142cfe0bfc25a74d296f11b834cfadc74e19ac4eaf84ebb2a2491d5dc21b42554e68331160ba5125266827883487fd78a9639831683f14c154c2e7a109ea90e782b6117fe47f5e5d93b4e76a4b32421d72947d20ab84df415ec30918899901b3d71b86159a7c1b99bca7e1290f0e9cf6214d1d64bd0bd38bf39b7d906ad30197634ecad9d9c91614ec1065f352ffdc470a68a8229b521ddcf3c2bfd6b96f47b2e08789fd033663d0c7f7710edfee200dbd7d8aed5c4f2b584f68763beeba297ebba96aad5573a2d832dd4f6768190d2e474631fc5a17caafd7d88c64c06e2dfe4105c0dcd6f50b642e08dd06cee731e56843f9df46df09b77b64edaaa3d886af79476a6982c7ae2533acb1224d8bfa73d48487e7eef52e2bcbb5d4fb5e28ebc1d3d913cc562f769be521e9a922bee5b942f2386edf0dd966c1989a42b9fc55c0ae13d7718e785f7e43eab4cded835c01381a0a7ef6c425180e415070910b96248dcc211989372c62c3ab8f92182009185f4a02f29c97d1728c3b2ef06b9bd3c6f1bc554474db6aa297ab0e4b4321db5d0b3237c491ff63bd8eff3535a8791f55003a65ab2e9eee2d9924d203200ba55321bc49a9f5ce2ab2430fe5db6310e91397228fc9044f259c7ba1fa9d9d7aefb8a2ead91d8dad5480037f8367dba650f5ac981b0c2387fe01e2a90525e28cf82720910e990c8be8b117741085d5aace01667068fe19322e1a5390b225bc385637bb00cefe31755355123176668eb58ac483eb40e6e0e720993deabdffe077b67a600e3a29b2f669740be97faa2b88323767f3b88fd6c3dfb3d1c5fae3bb2b8c415badebb066acfad434ca009b2200329accf1178a0be1135e3673e6b79ce48fbff00b4d70703dac4e268d3e255a0de5df8a50ae5a4ee2af6a28ab767dd28f1b7f4a43a1494b86f48cf1b09706064f2230088151cc72b4c79a7cace52fb5991a297259124df018dd3bf2e74f952f44485a573602b563194ec074a4c74275a126d8c3a0801e709f574da2d8d72a8a119f0dcc4435781b1792a2f5d0c0cf14ea7b21e8713506bf97a707d92ee4cc5ff2f9d2dcd1ea842e7102292fbb8fb173bcf4b8529076f3c1ac3bf6d5fd5cb36c08088018c0a3a27f6a4dd47092e1c92b24317747952e7eb8d06f21e53b976e758753352782600316364186337b3dcb1e1b36859a1ba2255d521b15b5b51c365a26d0a7a789277fa6325e78133a7d97900671227941f3e72f580b2a887f590cb9114c8daa13dcc216e8d407625872939c66de52124cc02a537779d1fc95d6bd478d559d35704f0601d04a1194746fb459c6ca17e0a854eb859464efed3da9810c7639954cdf164735a04a8c5a420f907a07ba23101de70fc296fce19154fed76835cba63fffe604cbd2bafdeccf7b0466c0ce9bce687bfb991130d7cc6aa85655251b2d372e8f5148519e92d5bc356cbaac4a10e43c0251f30b3f813bd8e128440dfbea72350a66a7d8fa7028270fe67c1db1696b320603486388c8540b64bbdb738d9e01d8df91fa9c6fbdbe1021e1ed4254adae1ace38a1f06d4ba48d0d56af370a2361daf250730d1ad80426d03c8ae7b8b2c9d5cce32a809c6223290cda414939fbf5fe6982d477246c202b453da30c41711cc197c312e08e2f0cd3c3ae312eda8ccf7a2d6d5bc7fb7266752825ce0d40429afe09586adaeb7a408c03ee8cd668089751954a51c374dbfd2b15177dd4d2a860a1cd21eb3fc767c4894ad69a8b2e9dfcab5b39f49dd5f7020b3661796be9fdb983b6101867b1b5e6904a8450c01445de3e867d688cff053ed8460195603f16f788abcd66125b634ff25f5ac42b9a850bd5d371951e623fa61c96acba2e852684477bf805ee8ec57ba3208c7bc3fd22b5f02cb90946597691f2fa0aebc1aa81e3f0bf80606387e29960d4ddca6766e3c14b7e67d5da356a4058bd4289136deb0e766c0952312360bd1231e8e7cf94244bc66764d3aa0b3fd388f0e60c0611e46fa8e5ef43eb98cf96b0b254a1f0556d1b42189f22820fb7b4af35b3c032390e2e81fb1f218885becbc6e2daed10355858e4fe8a1562bedce5cf87c819d42be0eee232a5a657ab98b5398d2eb9c8955ee3f53b30e9aeb941a2926002e20568184a788d57f42e46146628ab1163b00f2015665e40802f38c77cc76073116dd1b09df46b26ea0c3a699937eb07d96e2a9c075d34c41476b3f91ab9ef925f541c47dc2f97ed127855794223c3031450a45f0137d6697af5871120477563385b12f6f37816897431ceefb1c1ae35d716014e95e4b4446d64ced5f2672a01b9c972b588cb5a7c184f45b5fdb8ad96f576457cc559f123733628a5403bda5841ebcaa9077ba74c4eb71c15cbadc585ccc4527654e330c9b4cc9380eb79e91bd6859c78244361cf711f1be4defe6cd583d37cf03e90aa017a796a9d194c88f6fe84856ef0227bec70079f8f992595ba6ca4974ec3949bdc4d43be17c0ce3a2bbc1d579c22eabf6ab9a5a2882c3f0e63f0116cb5e5eb548108e4cb99eec4d2157ccbf0fb759418ae0ced8080dd10e138bbf7f1d20f3714e33aaa028ec251c96c7acffb26b5904b0be298837a347f389233e86504ca81dc1fa082845f04dbf973873dc42edaf7624a10930d0c71149866a3168ef22431fb36cdbfaf7f7c94aa6a89c4564c7f6bbac15d1532da872e75fe4ce5a3096a7f63bc0e428dad30f05aa174c3d5b0a25f8a6ea074b38ac00fd352995d12e1702bb7f5c37b172b75fcd737006f1f378637a61163a43e4e55376659cf6ae6e9dd99d542bae9f4285e5614fade864deca20ffe5a569c16fd95b5102dace88d829a3526be675ff8fc4bf5a34462e80ed5f4660a940be1de3d69fa5cd5ef050c61e9c91e7797cbfc1267f663a2f502320f8179914eb6643006b2db50ab773fb7642793f5606729cc5e69660b28c34483f4a1d3c040fafc7097ac255e5b5d1bc6f3c85f15756559288904ed00083401e92b2326d4afceda1c92345a0010717dfcdb320eb0fd319ff194f05b755a8b9bc75a63644cfdba75204fd0723fde1c01e5c55071a358b248321ae0ed73133a8939039dac744f88cd031c358d4b0e666d3caaf4cc2b1c54c72431285cdba38f13365167d13c4997203e08a6e4c4b8e8ce5bec513d69e497f19586db24c4f68afe9a85291cdab72a1332a4dbfe5608d2ba668c5e48b9ed215f338ed1d0e734aba0163220725fa87050f50a4a0c9e07900681e208498a8fb5159c1808bb97980bb8df49c1a629522f44780b5465059ed69e824e462f85abaae8dd9aa00d72917ba8676a31a4b4f09d2ea58fbb912f677ada191b63c0ef31fb76dcac04024ed030e0dcd9ca9d434f30550dfe83fe8aec21bf77005256c277f61d57bd08975e1ec4acf0f7c7260ab3d7851e5912d04871da1ba2cb2f66b9bea3429d7b1153bce449f9f7a684366f6f7d5614dc9677b60427b7cf0c082485a4405a9387af3899c7a892b8657c610393312475eeacf8b33376190a3bea3987eb6a514204358e61b3f782c25910b594a28d54771fcffae47533ff4d4426e96345592409c43a1fd16d918325e226c4aeb4e485ba686f0aed4d90195c9ea58e7cb6d2ecfe97fa52b9e695efe9e106508da7f3856f45320a6a95467784a7b1988b91b2530f839599678c442d3a63f8626cdd4290a4d8fb46031673e977b60c0d73a20782cf65d320285e838ed169c4fe7701e03d132dd0c0c77ba40ec199a1d7f02c98d514783815d1b12d655dbd3618977bc851dd3c9c30e82888b2a51c3ab730b635acaac35afb6e7090ceafe58f68ff6f5b8edfc94a4b580dcafa6c6abc1d52f95ec9e0762fc095d24cd181c9192493890783bfa75e06fc2214a41e5a543e634635ddcf69de034197c256778203d3f9091379b1dcff3edee161954d1ff55e1bfeb65693ad034810a25ba53f85438337d132e18241eff54d56d892df2a041ddd18194dd3933ae38ba25bfb6b1573a61d3daaa1ff1973b483e86038a2c5f940910457a412f6193c6e9eafc9282adf64fdc91f799f215373a6da7d1ed92aebd9a4bb9a9ae58c1336bcb3ab3c630b47a905f4297f759362a72cd8cb6c42b8aea5639dd97b2001a3167a24b2d5a61875bcb92d3a34d6fb5796c77923dedc2e1be0b793e50136eb2ac25c5505817a3417f1d1122efee23ed7abb9fa25ed1a0a9688984961877c402832fdd8d3a0aa1fb9fbe352e8066db05167a6d9840a867510a06d406e12a9f1dded2217bdd11f08745d8b21772df0ad38a663fce27803fd4f73c063603c8c38dcfe1d1d30d7a91309197c058dac712b1f03b8ed89c100400014b7e99ddabb49ce846db228e322d2276cd3ec590ad05b8e11689477a0380ca42ab8394b30ecb7eff6b9d531eaaabcfbeb1379ead3937bb8baea5a4f7cf259e19746be0ec4c3603fc3d7d49391c133c027ea0dc35660865621afbf147e08890ade269b10eba82d5774153321b1f8d3fd54569f482c96099bba7e2358859690f87cf90189b3df9c907ea2dc3892a994be09f53c1991d9cc8385e31c6af2cae9819029ef5b7a22ebe64be6f97a6cf61f1ac4d7d69945d0c58e8e4f0b869232e97f825d62a2c8178aedcbd533827281970155fceacf9660f5904868dad4171220236824e0c8b0d25c0453b3dd32c2545b3661a40437c7de08f824b14449c8cf456b81611b553f61fc1db45cfc0a18e30391f01f44333b02051cad14e7c51589803516ddb51994e6a99e9e8f50b2b6f26f7a8a8eea9a6fe09f1b7840cfc61fcf58ac06c6b61b5290f81e07f4b1d1a2414bbfbae2a9330906f26f102473f52fc72d54d0f639fc2463b4b0ff1e0ebca02500e12fc1935864d3b742becd82dc593273b5339639aaefffccd63f79f3feea822ac6ed5102fa11cc48ecdcbdbfcf29cd62dac2d2f7edc5e2e64decefd73e57ef2d790c86e1b0844e19e92bd34c9fdb6f0e3393a8a8efd31f1e853f97d59185c71665f31c0d474beb64adce9250b600f33488ace948c3e56fd1ade0a8c29fad1774d031e32c09b3d7ddfdc934ba8ed6e65282af66ae1eb18b2a6f8c5303bba74bccbef9df0846941b34a76acd57e538450dc36fcea23c3eae9dfcf61ad7a1b6e732ae3088c9cfae5fcea47524250da8e20c22a42c9bd0105ad933c6e39965a4b2d640b914ed92ced7cdea21f1ad722e14ca84af568f1d988327b395971e77eb8f4be11e5775707be6bb42c4c372bdfbeb38c857f33da29f7daafe1a818595d1f5383b3d6a32400213e0220bf6bf8fdb0069d70fe0c7f10c27fdd7eed9dd3b903f1f2adc888ebe9a1f0aacca6257dad0efc29530475f1b1a71c4b974518aae94c9f0b62bc0d16a3845f9b04d6b2111379f28d4c2e3ae8acb557fc0c65a8e13679d6e144e474c569a817cfc6d46593df5c483305c5cf436a8d108128aea3f6b2f164f93062bc5a76a5e839a24219e15a7042bc2bfe1d593f94507c45eb79fbf425b6c1001e2d203d5ffcbf237978f13c537a176ad7048f56158d67cb4dc753432d8f59d38a71a9349ca1db2a5b7e80e832ff506e7230ecc879c556392573479aac57ffd725557d948e3c27f53ad44a73be688d03da7a7fb1f246397406962071aaf67a8c01148dac7ec61d1dff3b82c1d2f8a7a37cfa3fc5a9dcd2cbaeb2c52ea1fd2be9a223610d814931312beb496a0dd1e857d8f7b687dfb2e906bc3fe6d4a165cd6e2d82f31c640f269cae63e4f071bd123fbfd0c4d7b8e4f5f5b5619aec6efd91bcc3c580e0373c54badb783df26e86dc1d3dedd47e3041b6dd30437f3ee31bf60d9a704b6b8065b45fabf2b694acd8c8fadf627318e813f792bdd1b4196539208a67f76ffda1a97dbf3acaae6edc22025a0577048dc0273e4be6fdcd763db16da16fb2130700380fe80784789b2d33ef94bc0dff2fd8ca7bb17c752721fad38cd8363848ab6ff3722ce1be434794683d0471742dc41aa0de8a6824c570404bcbc3517bc907fa558e0c98fb4bad5cd379a47644a5a62208da4de9d0ef7093ddeee74890ae616ca6ad9f3d5c521d28aaee5f925daa6bc808b0c0798dc27a0f03f376a7ce690498e70daa19e5df88ea3d4a88654d92cda829310f950e88dfea337f90784ed0da3db8a5446358d2de46e01a242b294678a69340c09da2df2c8a761706b3ca21ae235eab89e4c8c594ee098e652950e34ad261bb8ef480b46c66b40b3962cd5f6c4b6057bde5214fcc57693062049cbca802269d55fdcc79b5e8052e5e253d53b129c5a7d9de8a758dd07d24826209a91b4fc28952082754d199b6eeb20eb1f57688aaab7f2b1dd2986f02bc44e3b524d922f3d3e4f1250e63d873fa862ff2e6f0da72ab9d8a51407989b399c734010ec3f524f6d3f1e3b874b7a6677a3f4ec9dd4e7c343aff457f4f16b5c5220e8e45863e791114cd10ba3dbd8337fdb15b74846b362a8f55c7d45a7ca143c549d74888326310a6be3efd7959a2ba973a552873b101f02390933e3be319104c93261db9fad482a3b8ff172952e777615585e9df8d8e3e8acd8b451f0ce11e4ea6730426f85385fcf4ab1d2a3b22740a4a921a294c1e274cd3b9465beb937f4f217ee612848f506b67ca7251263671fffc2c2d778dfcde0632009e0710d24cffb1454c89222ebbb28d35728b63533ea4411fc956e8bb598864115835965a8456eb272515ab6c8c267c85a26851fa8218374b7d8026ce3d3235615d614e673c8c4c3e10a34511da47808d17c934c497b6849457fbc10e0f0b68fd6cc3993c24b4bd4c9a37ba3d93543483bf5026a34c2f6a4c17703682c7fefddcd2a6b6bb22afe65b25287a7ff08f1d16dada051eefc7b514ac9aba4d2bc84a95533490e46fba007e26dc17beabd73535ac9c89abcdc4886e5768066611988c980d187466b9a6817829bffb4b792cfd6663b3e2b64c6447b3fc37022e2fe82e9b4be5d44acf9e54654881894bd65c3419270739bffb3eec4a9acbd7a417ae84991f28148197e9abf476ddf37624135e05bfd5105dad92dc792bc5f6963dacd60f23b3f133398a1fd9a04b149fe024984df1aa50cf0adfcef6169653916b685b8181e8312d1b9c0438099677f7691c3ecdef72bbcbc4a3974f18798b62f09775a970f05055ca17150d03c28feb3cd82b506bca1022f6025771937e26758c58aaec1dfacb3279b904a93f6558cd615027696033059d67dc005c9745525ddd4e2eac06f1ea8528e7b3f2a26a892bd9847526e253701f61e79072e88e3b556b7df4217132aebb78bc894b774b20720f7eb444ff57a0298314791a1070944ffdf8dacb5eeff3161adfdcfe1ecc87105e4cfab9ed1a6afbee450b0c74a921d288a73de7341b3e50c62d15decee13836d2bba0848db028d39ad256b28e6913c89c435f493e36bc640f8859484393dbd69439711057657e688285a2ff1bda56a04a266d242115c74ecd36f1c1707dc1149f9f96339ddc6b04249bcc33740deb14c82ec1aaa76c28260437634535ab9027f52f478165863e3bb5beadf2b9bc37846783e89dd909af5244acb3554a063bf9636fba7c94088f8c9f5b8c21c12666b57d42f73bc99a48c4b21d385a31559df5a1690287b6f5cc0d560c7071ac5919b92c3b8cb65fec4f8e26ef4b1d351dcdf724ada3571189ce1673dae53e38a7e43a4c0d4a8ba570c5090834458925eacc932206fab3d786979893455f54588f0a1447362ea4d390eef67e800f892e069e675bb65856f6540d9d1278cfb5892575b4226a6cad5c2d0c746a273082bb381c3fabc509ebf7d0b0f9d6bf209f447a01ce0f002658436375dd65da440a1aee1722a42ce2d05d3558a87974326cb44190a2a2097e441c77e3657bde5a8c41c2a5a1f004aea5550e88d6b32227f94e8bc6fe001e7673b5dd65fde71d061778c0e725689747288faf387540e538f9b4f07032db971a66ac88dd887209d99e1f01238540cdfde56e4fb86177089ec1cd38bc4edb58b5ac154f1886d5a71a369cb6de8a9c8c7b1876c08386f53c3c74361b60ae495cbc8a01b574a9555e62f6ed4b3183f4cdbfd2de997a58d61d5e7a45837e6934791a2b90d1d81fa89e1d05ccb752a4ba9cdf26d6373e5282e595e7d78b4c84ff8bb39551e3eb1bf8ca9a9402403be2a377b03dc8474e467e9ec114ab116407e7ef75ad84bb65f1e0424f04aa701cef3a33412812d764a33ebd20afb18b3856b106f04885cd5986b9a793e794cd5765d8aa6153f0bf995e4e95c49af7258d2fc97ca6c8897ada7e4d9310c7cb026c61cd9cfcae2656a2ec525561281f250cc9ff674b35aea07210f30978d1d25ac7b0a6f87f8bf1c1d7ecd9024b7a57b7d446e5c6d4babba1c8971007e9d3c272098d02c6be262a286f5fbfeb5536bb8f85caa91a8696c9e0ffd6462c19fd89a5fe3bfbf783d646fe6dc597ba76ccbb697529315eefd191ec5ef8ad9c9b17db42b163b63243ce415ba84c48ed6fe26aa9c8a274a04ed281bb6eb97a86d3b05e772f8e4644f277b7ed0b349893b1753e5dcea2dc81d78b79523b3452b1dce440360b52b333a9b340a891edc8d6268204d12a96f66bcb0f389a36422172c044aa1cc077ef97215e81a0610c9b9fc3ee7c7a1c849ac2b76579a15ef5cbbe9af10e405620463e58abfbf6ad480d01db29cbc0dbb2a932580cec3057eb23f831c406991edbb9976992a4a237af97370f835807914272f4d29ca709fac11938bd97ac429979c47b1ecc1449e3c5835237bd86cbf15ce25f80f5ec788d6a439a41a67b23d753f4533dbb2fd719d4cac7912bb32179438dd66a87780fc5996c742f9d5103d0e0227c7dff6ea7a8d002bd15e280bd880f6bdddc33886cca5881c04d49142ca41644f5cd503b52a0e5fba81ade784dcc89b54dbd6e8add0bbb457b8d845b18efd4157ed466581088c462fbba71ab7040020f746812fd2e946629c7a31ca78cc1ccf39a8a7fcfa4b36817ee8c012e9fb2bf9afd5e275e07af8038ed38a87d415870b5db164234fa281cdbf9529672f3a3974ed653784ad5480432ac34b84b8e747c95e7ffdca6cb40d93a04e293407feab0a973b8c574c08b87313d2decbe388e7dddee4cca041637f07bf6d5307098e3b8fffd95bf8005ac6f6fa4d0f5b39596f5ca646df081ab8d3b8f888fe51d60ef9a0c8ac8bc91197fe462fc0c0f6395aad9ea23736b5b2e9597e4f718f7cfa4cba26be761867b6c66aea1c44aa5bd480238688b1e944e3a0784ed34df03a40fcec2f32bcfea36659dce6708801907fbd0f171c2f85fc548b1a861e3f53a061f2385f46e70ec1b316e68c24a9802cd631c7922f283b07901883ad97d83597b3cd849820ffad8dbd2c43c70dcb584d6a36d927cc88b4351daeb40a7722510c1e353b1729642da434e5a974811fd7aef093c1d43a7164d37475953ee3ce6ca583e8471a324d780b17984c0c3a2e7e6b8ac730de75be4f60dc9c055f3ea54afc7052500b4e2db0e7419269a70386d26c486dec7fcd3dd127489b54fe21477c1b6566b96147e8cd48ec5924c9fcf8f0414fd562c499e69594e59e1b4cb22fdc672caa51d3f7fccfcf3df5be77b734e663738a4e0df5badf6256f90220d578f02e591a552e832d1682db2a9e006ff79a23777baea2054bae25b5a02a9ad5df747ed49d86f5b5640e53eba6029fce5c2384db8dab7f80d9c4c72eb8d2589aad06556ce605a924076277303420dc851432c41bc535528b2b03dd78ba8b39a9f698ad5e06c695daca4411e6c7fd6b582d053902eea14c556f520eba0f5d9abe2a2253cac5dd3dd03fa6c916634a73da96f118a7510aa31fb5e0a113465beb261ec5a6edc467779c48f7ba66a6cec713bc8c3391aa8afec07cf443ac7f66ed1323bffe9854e667c82e7bb689ced20b3b64bad3bc228a264bcd861ffae838465f95971af4327596b25418244f502fb515392f1e15df92856d20bbe15f403464173cef790dd235db76e5a57f6cc61cadddd86d02da5494f56cdd4115387236ca4a6b550faa8437bac433b2c027c5544b2736b322ee3c3bed2db4088a0976391c046a31d6bca89b41ea4159dd859eca0615178ffaaa2f08a5c61cd39fe8a210408e67edf8db7017b94f5cd9864b1c20220378005c7a9dc55591a0c9ec88f46dc6255e63f1ea5d185c39d57d4c62103117fdc67bdbed4b3f62c5746c3c60e0772052178cd868e656cb55f88ed14acc8f912d526a705d456e9c86e1f3a0aaf00b6fbc1b34e76354de084e61b88e312c6719cfced9ae87c26838330d771b7d9bad6fad3127d05299a991ebabecfad7cf8feee5ceb13c1f5e65bda02b38d5825c295400d2be3080a7bdf9e408a69724d3dc1ac916e5a26a214c02ae9f875c01a1777d09debf4fd03b08d1b00453d9475fe967efbef0f62f1f7baed449821c449561dbc4b61b7b4730e9de277d68451f1f36592da47eb762a4e514f883b6bda9b3232cbb7aab276857c632fb20d9649cfaba797805e9d0ebdd70b1843d5b20f798ff385fb0aba45c7dbd1a48ab97d950eed27cb0c8228b275408320b94ed708917bc5da92467a363a65ee02cf78b99311a4faca5870012910e50dbf0e289c1a63a46f0a397034d1ce6a0980b660ece22ce6015682db12108f930d5d578b772c8ab3660e1d5a103ae61f4d18fdd25b8c4ac848434f1e6f3c60814823db536c9cf67ec369637ec5a7f097b946a35c00883d17e02ba01343beffdad5c7b650c11226255a50cfa93f18914e4ea93206ffcaf1f9b4c09cd67d2add35d8655710061efd9584cab3eda2edbca0359dc603a027116eaa7cbe1ebdb1f31224f4db89bb68218d206b40fbd1212f2e07bb344dab00ed690a04e48e8136eed831cdee36e1a32546d495565ea3d5cc27d898c18dadb31419de65b665044cfcbbb9632987518de42f57cd1355afc7d7dca0676bc3b3da221e8236d01582a2fb81f70135d06a0216ee34348cad82dc7521aff237fb6b1fa4c38c7381e512f06d45c2c2b3650f41bf8144faedc71a839f764bbf4cf3ba411225819db65016c9f7970ebf96ea6e06648072e058a2aa464be9ab4731f4e1450488fe6e5ef190ae7c166711ce93fee7481bd90ffc0f0f9a45edfa7b3bf9fca46f15ba40223b9b50054f10a70dd6f9c359909e3ae401104b7b4f8383377dbbbe19a47862e75a7f012c6e68483b46faef4e2db7921285a3d89310e6d4c820443958fbba26900e425a8abbe31479bf66d44bc85bedd3a6ff5bf9e350b03efaebaa3ea6a1c8b35f4c7741ed491fc4d6fb9caca3d314bfda5886aa737807b60213d126518312516656a4c0f1f14535198ded0b4f97bb70db5168a316319481c6660268e2f398ab9e7d6bea2c66fae5f3d39def32805c2e28694c20c504a2f4e4d2cd3415d7d3d00488452a7f5d1cc78466009518ca640e497e179a1381e1882117d8bca88edf38d88ed4a6c4677c250e39a187edd07c81305b7c99545fba6eeac78f5215ab9e39e525adba2dce1de6ca41f30f128cd0302c9ae04a01fce2775683bd8a51835bfdcd9d0ae3442ac3ed4b37b03077167291e0d8e0446bccc8ce6a390f0a326930bfa333bf2cc5f3f008128a3935d6bc14768fc9864a6d3814afd530d5ebf8f5af48399b0a14fd21b3cdb6df1badfb7fc31aef4ae6672dadbd8c3c784236d67387a837463cb9fa9d55910e6f31f258ca3aca165ffe0f4f9fd9e8640f4085f017d027d8466e782645c14216020c6c00a37779422a016cee77eef8deb9b0374c2e7e460b9a92ea96177daefb2a8e9b06df8519989386374e6b48dbf571ec704c1e7f42bcb13c6e984a46d3ae5a6f1b573dd17703c878f9000d08463148861bc71701411325461a06979f08026d39e23fe2881d6ab37a478bde799bf4c4b957b1d4fc66ef48a87c0d16a30c3add814e385cd2da5f522d307ab528be54a0be92c1e5b3e9a71986dce510e0f223a7ff9155ccfe4e7171994bd3e3f56396a77b270977fc0ded7a62d2ce8e597dc51acb1ba88ed7243d0ed230fb32040a68a06fa9e313659be5409d4172fbf201d507adfb9d33770fc5e6ee89e36de0e42b48caa4446c7aa4b25f977d12345c24f60bf36686b5c1c0dcfd043a0af0a0e56203d8cb32eb436d83288df50192a2226df8e03cad41d699d1cbc33126a6498eed5f302c57ad956559f9488452578514b095938eda50b3cd1399ea1fc871c8043e27ab0bb0ff3593224c8902e45970dc15d7b1c4606a472e26c912f0a7675eff953eb5fbcddfb7c39eb88754facbb6ea53a83b28a065d52093bf655b6c5b78fe150641e3bb794c2bc72972073d37a836a20008dda608da79c482b4a6266eccad11819e2483593e30d206197956de7ad247b402a103ed369d465058e52c8ad8d3cb22f646b4321e3b4c36dbcdc463a5121258af6efd8ec4b944e599812a6272772e2eb9beb9b1a1e34f7f0a5cac7a6d1d71a2a10beb6c5f5d16a1dca2b2ee99711457d67fcb6b2248ecc104cbd64f2888a9cc3a6ed15d058fc7981a378f194088a742b6a78319bcebea4b2443a6e535a24d76b0d6ea7571dc112b8a19f89ce0cbedd8ab24784a6dc97c77276d1c8b42170a343e0b930f837002eb80188f44fb342699eec56bc11442557216c42f70f8d62f2044ad314f999cb0f1858a826288399246b6a33b3535f1c4bdfded49a55a71e5a4d8d4f6acd1031431d3a314118554749c8ecd5b07df548410c76264c493f7bf008a76c540f74b0636c10c146c4aab6ead1937201b775d8cb3a9a8dc726c9588a7c399f528c274d258a826aac42540f3b937a119ae74431dc103019a6d0d7e339d1916beb5531c83ea3b4c536c4c867293fcb470ccadbcf32fcf25c31bbf074e2ee893a1615574a01aba55e87eee81c249641fe3f3f918583e31c1db63192400adda54572f648ec0c884e4314d9ced215f446e718f666f46669d62b5fe33bc135c5e0091f5a3c21e4cc1e36a0d0f6463d7b87da3969d0dd84227970571b3331775c4d3379aedf6605f8ba2372dd033a48bce66740badee29e77740e60734eb45e1755c8c2a6b7cf2a6218116cfc8b2413c3f23ea13d804f1ae27366cb4c001769fb026b44190ba20065eb090da2c0d01f4dc4300f0c2755b55121574c0c9862f5a1d98be3e881e655b3995b954dd191528effb5e918105d31bee33067c5ce9c13ef89df74c796c2601f6ba3ba3ae8aa21b97d585d892334148eba6d61878e0f1f1cd03e0a17d6a00b8335eb8b8a7f177d0ba1dd1cb28298bff1c1c167a645a7b6f9f0a7cecdabffc436c254d69dc9cf5a399dbedb6da43ff2dfe7934131420ced8a902e2f50eaca1a4d811f4cf39c3e5307c361434690511516a3ebc0c9bf164a46b46da825d5e1091e9f4db148dce997aacfdbd76d127dc342460c053581f9e03036eb205398fe233a5a3509e7bafae355934681efa0673cc8ce3fddab3b0f6780598f8d95fda361d926cbab0a236c70e9d27b8452d56fb588fb9e4679a49a7987045221520cf6b922eac6454a02050eed7d92b33b42a6af4ac3a91459d8e5e505edbc14cb66336388521e4362ba2b5469c49526e5c61a00cb12fe72da6a894b599836b1a397b784f2641f5b13e9509cd1ffa641bd9ca589c397341d2b416bc323c9ba57a24b60dba7d1c1ae9bf1872bfac679004f53f34619227cd1cf3936937aa3c0f28a64784a2a5465748df659d565a0cb91637a0839c0a05e600a9ec3f0d3fa2e98bec1977880df5221a3d72a66054ec69079ab18fa6d62c17b10943840ffd4bc6527de4b620e53b16656cc3208a7705ed6348839fc1f6d698ef085911f6a8d841f104a8c057c948f350ed2bd18f3b6973c8be9c23004a9e89507010375db7fe26afc62429c24d7fadd9502429be436a86fd4183d7d7901bd320427b61a6eed5c050c15dad7d37320adbf12f7cf20a152271f103401e0aa51271598c6fd3448eb7b08b63b8aedac44cea7880ba6971e5c66bc5f5e8ea88e836b93fa80783466d66ef4aa3edbc97d48d108d96a6e1c7418c7373a95bb7b36f2cd3ca42a2639220fd32572e235537fde1446a631e33c045b5f949200f131b70b9a941b54a98559230f6242495b4f24d9e440c844c0c6fdaf95fc16eb8e9622e8e1137b8b6a8c528bb04b10fceaa7b00670ed90241f5b5c602994a3a2b9879cafdf5ee87dc2774d89839977cf1069883b073a287560a1222df1baa7cdad1161489c24af2e0995492c531027f4e268201c32817aaa43265450b0bc97b3e9e23688580ddb0fc3138590c31c04d8e88b5f0ec7ebddcf0b224f5009ba29fd615f514c97c06b731ed51173738496e6dab69f5719ed81f62a0e33ff13d2944319592dd95eaafc552833a7cc84b5adbd12558f71954e7b14fd8f622e63b34622cadf8d0a00148aa2b10178abe0cbac62a9b3f6f77b08654882f0ecd21dca6524d4645647364f4dd4bdfaf6b7e816f20e1f40abf854b9c36794fb562c3d8ab48c1fa6b59fed96ffd86ed7c4508893956dbe7deb1b238a15fd7783e887676829135bfdde7c078d1170d10f4b6e32635a5e1a152154001ab1722a40e8cf53644d341e27ff42cfe50268ede4217436b1d53679a505b4b6cb8939d8dae86b7a593dfdb109518d83d15d42543424c50af22e96da17c31a6db2abddc8eadee048072132fe3931e8f097a4a95268ef6ae4b09a7eb82191ca5d23455dfac2c8fe9fd8586f5aa43b7db444b1c80ee8883c91a41f694a28d008bf6e8144e939a7a7b134b08cede2cc1c6e6ba10fc5100b5de37758ed50aceaadf7abff9f566fe658105a6914a42bedf80119a1ed6ce59f9d91a4a2096cec6e2e5dd645bb849c9cb7366ea5981e2469472b771ad33e0d6be24791042f4acaff69855c9a04798f9fbe3a8c18e0ba53fa73fa56594c2e319053d0bf20d0ba9452ab1d58ffaa1a0249ff1ce5e96e8e2df5fc69b4a40250079f3b6fd9b3a54934419811d1172e35a2ce25389d94a80502a9c6d6809e58e8b29853b39db6f54f8cc20f687efd1e49f71d88ff2bf9b1766276730cd298162e189fa7066593ba895ba3629f39ca6623982b1b9949a96155563c9f7f21f85fdf52126786dc4493daa4452b6257bb72ab1e1a6f2fecf44e7f3794a885ffcd9e86c5a639aa7d97c6ff902a77f60c1a19f497e3172ed6ee3ac683e0b78bb8a82da9da1a63a6da76f3ff42fece69a5e436580ec84b683a8145fb95ae0c33b5b384879d40635fdeb3696399b7e156d9f297e1b619b62844eaf56d9e52a8131f65ae6cd8259edd51b533bc06a6ed0babfce4048a8a7470b21185af25387716a7038c9de73c79a3f291ece0380b819fc379f6dec9b3915ff8d366bf8c88f374126f46714a86312633fb26e47407ab107a1b7d7ccffac6b7eae844e4b8354142ad9caf79c44f923a5653e0bc87871605a0fdf4dd82e6f99d065519094b78cd6594e4ec22c65084fbeaf04d953006cc5428b5d3663f608d185e733ec8040b765d254c30df121efa7428bb586ea38deaee0f6cc6a7326825392ea9de1548c7b98ccbcbbe752f9b90bb6019021524d2b4a3a027093874d6833b8668b435242a1a5cad845bba694d7368993b5e1ed3ec9f33446dba733ab1151cfb39726e12030c0911efa890d53a4ea20cfa2c61d208604fc3cbee10a47c3c0e786bf7898271916dcdab4e6f3501105859919d91038c09fcf6f23992dd7e800d3dba6739ac99c76ae532beea33f4b5efef6ba62d3fe1dd9165729d083ebdc88c266afe031f53c7d756eaf599df5433b861a72e41f5d93ec416f9f525b81d3cfc27226bea3d5bdbc139980e74f2c1fb656be30049a8bcf3f0ea112593d804201f0eea9e6fdf23375ed70ec12e418abd68b1a4d13a7110a77545a277687c7c4e9f142d662cd001264984bd09d703e31f85e778c146fa4d87956f5550d335fccc8b035b4ba1df61ac8e81cc30cebfed687b2d41daccf2e17f534c6b41e3af78e9d0b482c5b6ae701f484b3962e810837575ebaa09e540c3861456941147428c971baf155df24fe1518288703ee4b355ddfdfe1ea84f1dfb4127180ef606df95bbe427a3b130eda1ca0c4b29cc7f5e1b6919a04fa65adfd3af8f0d4a464915b5d6bb2b35ee30b7445322f7b78e7a2e0adbb38c00ba720183cd6128739d5a1cdeffc34f09ee97f27a038d69a36ebdda2e87f785146b5448a66fd5b64366f47b26223a654a5015b4f8e43d36acfba42219596a7981a3302dae58b89ca5aa7f3d76aa68ab3e7e905ad3ac875836b1d99a021fecaa67612dad18de9537ab2f442448a964d7908b51256505983b0c30ac9fe7f001bda5e161d14fbc9abdac8eb2ddfec775b79f7eb2a84586e4e21d16ff9fa335985f99dd3fa43fd63a0aa762db6c4e7039df93e34abea5f4c1f0b7c6cf9383901941c21cec44999722ced8a32fb8f3d64f9240859adaea05cb66b63c250c70365ca1f87348b44cfbb648c7842d57b0f768f8cf65a6c01d97c5aa192404b2e329689accf897dba723cb186ea60b2b9603b27d9ac67d65f57bd769df1884653368c3962f74bdf8aba50408222654d3ce980a9107bb982665a12737353736d8f67f55af7e1d192d955dc1e191ae3505ff15d86b3f2fb727bce29f85d73a2e35a91ff4c6941eb9d386eb68b175cb44cba061a7a2fb43ad0893c25a543b929283377600c34aa93d0f8b58e447ad60a9fc531e514cf5f1938603d72e130af04413075139965a8a75b02f510cf68807ac9c8a4e19d13d496092e8645ccce9f87e0201239a5798a157d7502fbabfc2e992fe564cf49746195269ba305648a30f413c86060047f363884a6ac1855e9558faf9421ad9d31368bc2d6229f4845b512ba671a064476310c8e450a24d74a2f3d8d648a1aa1a44cf3f05e54229baae57fe6db5ae27294b996584dabbbf02a0b2c1a1842f433f34fa57d9cfac7e8bd1a17e5d1c0daa15ebb2136e71e2ba95f2357569aaea5677ff9d5bd002c9bdc0fd0e42d261411f95caa7047a098e7eca9f1cbb27e980947d45e94209c2a76363d4be31b5a38368c3fd075da56c19e11c4fc43271ae0e772f97070be5f7e976ea83c424a65a6ae0c5e5bfd4fa31b34e8b61a9640d354683bbb9af9d5b471878c747367dbd586806d512ff49d79c7f7c2c47f8720ac240346789da7ebd239301ab19496c54f3087357b28961fcfda6d63071a5566f10ba2033f3015bee1cc6004f24462d3b19321178f96485006343b78a373a5d142c0db6870b5da58d57b4ccc68d78a4bb5d2f2223292fb38798dc719f8fd7b8642d33fa2f69def7c27a8e0bbbd4fe7f598fb37dd19be1da07bcde3af3334b783380f2d0401ab6d281e00618c22a199b76a1f8f3a3086a4da9941fe3a5177a9a8589f12f1231d7a6c06213c9ac1b8609a6ca2a12e81b9bf4cc3e835645686aae1ec6e52f517106e21fac9c924b3743280a298dde50762e3d00a2f914a5a8c29573148dd74cad804684251d69c1985595782099164c5043b2b915f6228e9696d49a17aa9f8475d2c4e4c18553f19629e209d044ae813b175698896e65185273f8ff4869401607a7994a166e0a9e6e0a18b21ef0a616b3427aa4cc37bf1ec15e2f37468d264dd0d4ae4ed09b6672c558b5238b4317dee534cf0d677ef7eb1462d4044e1b52e928c3bf0c4ff34093230cb247101d967c5b5622f70091e1448ef00b55b8e5a6e39463300a3da5a54a7ed24b3625601700053d89b4550b99e3d20aa6ec1a6b6adf448abc8bb60b518fa7f2d7a092844c2f8692f8eb26ba3434e39392bf4ba0be7376b191f5f603d04e6b8b2002d172fa5b780486213faa2e7e70349723216079d77149dffb6000265a052dfcb7acb04610cfe473e1b4511808330c97aefdff9da4d4f7d4ff9f1d693ee54c7bb1f014d45f19d47c147055857e9257c87441a994a311a17b2b59044d703555dd5fae38472439113d133b86aa98ab94307b1e4959a30c64fcd14656838d4b8a8ed26cb92f2fe6c4597412eff8cc2367f9b72ef57ecf37b1bf495eb79bd304e91ff9beee48dd869da0d1731aa1a54db9b12c6e6e3a09d18f51118f03ad80801db434bad269463d0250f334d4287e4ae81ec236012e688ba07046c75aca165c8c235bd14f023c9b042fe59b58b29a6a877c9c745d1f3cc59bbc308ba13957c319a57474408aeab4a95ff5b6ceffe7fdcfba140323aebfaa646e34214ded67839f582a19f5da1f9dd30f24966d61c35373fd87396f95887d6a8ea4d0362d2c29cac3ff32fb6574ebc4a7bb69e4df9d2b70e865760b3f787ff49d66ca21460f88c6e4589f6f68025af866bcf529de0f76875adec03a1968cc6d23b2de88ebb2cab315ef98e6a670a2df297cca4010e219112f20335b665d2fcfde6083a6d3e2f9eddeedf59b9ad32e88c789813cf070712201c76d344de1bf7b2f534171a9e4fb7d540a920feb92e083ca59b8540e8c5080ecb468f3e170ef1ad83268d65224570f5c72b5598ae17c8311669986223cae4fa80f58fb86f598887cf2b49d6223d31da5118c9a9882ed4bd055ba24474fd76a57c4250e339a193e47bf7fa3c7af26a03dac3ea2918c08c63bb60a6c74890ea333353d21b219996e42f1ada19c7b6355e4ffb88a6711b8f4964146a0f4c048d368566ab3bc43eb6d4ff9a7961e979feee2f7c97be11812cbefcb85c38ec2f596d24e2924e53bf883446f72a9d48315dfb625b890e4b572491cfb855b8ea7b5b3949dac95c10bdfdc8572ec9819101efd6f00ea0bcc226c919c09f984ee1ca0c42f7fabf7d265ee5070402ead84cb1962cc3c3331e087c1868d6e3b7edce6870af6781db33f5142857fed559198d9c84c487d8f272d0345bbbbefd75a0667f8509c926048ea170b33bb844eef88981162dd67f01c6c947a122dfd76cd78598209317e7ecae8dddbf045f30d26ead1a9f3d3e2b6482fba661c1f7e1f819163a34d5901e448e835b4fa9e967e93e338002e2d57853d587bb1c1e8f82ea6d3574d7cc5799359a63e60ad99d12c99741011eb6181762e1872ce205bb134b5db78a5aeb42231e86792ce05d1cf2b6729a75d99655ef0662a8b609d09c80a0acd807e83840aca7a689aebfe2a163429992b7c498edd7b922fd714645f7466d1a1ef0e9b1cfa87e73e9e85d610ca9176df618a5166527fb0501f200c21f21903e00231846c006a1b27df9d198a375059f473aeecb6f0d208c02b1b3475cf87ae81606a74f85fb307ab0a080c205f076027d3983dbc48912c518eb5ce610e0acca4b960a2e6bf9fdc89d81578512fff782f703bb6400523fd5e2ccca223e6822263b904752e36505172074e83999d72d0bd1ffed839b123ab99901755c47046381febcee69f6639783a56222f8c436d54cf5de34b2deec8a6157df47a6a46ee1c3e1b81d25f93dd6a5e4165cf79e04d4a809ae5560ba552632156569f8965a89c0557a2d4914b038a4816dc047a5d5c67cb91f8bcde915ef79e9163a091cd16baed192171dfd6789ceadc869242acfe6d7d5fa6443c1a513407232faf0f405f5d53d4e9ca3e9e39b97c42da2da544adf3a724b6c26d40a56f19a595a2dac0832ea9699d2a3edb289be3371856247082207a15f1938cbf1d182e62b73e1459fe4348f74b10c7ac231ae154e738898f3558c9aefa06fe6a576bc3e845fe99ffca536cfcb6aa176a0a9d01f8d37db5bbe8d05afd56b51e4b7e9ec60e8c544a687dc771a828a6dee9c91a17cb99143a02aa629237c381d61a396a67726c1e07dfdb93e85bff1a28bb7589fc5f56516bcf4cc80a80c0c426752dbb6ce5b00d9db389ac56ec6b3bf25248f7e6849766099db6cc2dd9625a64c2ce199883fc270af446a74d85deffcb5fb4742b8f9729c580fcef12d1d9dfb5cc9c3e2876aeb04491803732dadbb56ed6607b4cf66e71e2f181d6e1d4d512faaeb22539d772eae300d82e3529f3ddee0c9f1dcef8d62b01ee97bec35a6fdf1cfdc6329e7513c26cfb0d1c19ca5492ecdbab9e1ba548c0d6745a89a404e9e58576fa6a44a66802502e247986ca177cc1c28d89cbb9c66773b1812f191e67d4f47bfbe33f3117f1940d817a2bc18926bf3b35c67440ae9c5bcbe1f269d75bde9ff74844778512048985790cc5af12b6f3221c128d3cfe13c4f986c1184027f1e996215504f581763ba2a050e812371f365ffc77a081b9dfacb3b542c8795045c2bcb25949ce8ac0f77ee9d3a004584f2e06c2926736076fc802b9f7228ff8e11d99e63d26ac65d472ee6058b9a2aba17abb363f2702b4afbabaac5eb5600bde616f5099a6da63bc38474911910b4882eff1d2421c2e83c899f2039f2d1c46c90d57109fc8a366191b7d2c4735309225f530e780b5d5c27f362a79cc55e03745589074cbbb45cb2d5753a0de2414f0573e39ee78780c532a3dc0fa90807c783d8e980b3b7395c1f67f4ecfc808fe3ac67582a133ee4f911b657a82727cd3bf5bef0e5378972e299ad025694970ec04d073edac1a6824aada1b8cb3997a3d058aa1feda430bba34db9a43f4a6dc349704cb0e8e71336587dedf92704e89b58158759ae279f98086c7dc9eee56cdcad34718a8ee994c346db8f486306b1787ab7e25bac5bb679a0a96b9a6503580d83221683b148fcf1528c6231b1e7858a87536f62843022dadb42a0c665915ecb7f49e15dddd28af13899bafa388d30be46c34a5e45dd7ce47a9c9380d6277d1228bd02a28d30b2610266c7372f90169a0910a0aef0833198316af264e3b36fab37935c830bd7ca4b0ea8394d69948d85e3ff597f64512486736552e2f934d321cb764096b7d36adabf6888539b954e94310de27db4c02546708b770c2c73cb514dcbb20156e236d0e803f663798a3553bf4ee23028e3ee8d0e9521df0ad6feffaee374bd1a0973a2b83aaf99981195dab2e883fbfdb5e7d104ddd88bb3fec76da900076b22148676e8a98cc06f7bf874e9b81f3a03b2f85290d6e88e49a33870cbf739f043614e3c54806d4e6d5aca352d36b876a87c9fe83b6b91a587d2cd8ef0645d33211006d0944b5d92f84861a8560de3af52a3170cc1dc96305bdf362433f66d467f717c4c64e4a3bd59b12153bffad874be48914b454fa78107a3ebb4ea672eacb47394cea0882c6829a13b35ec3e00bfe648b971f76c943e5ca889f22ff7382e1287b76c01cb16fa797221552cd7744eb34d711623162f60272c521c22b41906d7607f9f0429f8e8d2cd4848007d2bb1b219f1009e69e4d176dc2b9b8a6f5928e33a651a3208c08107e882485c21eaaad0be8e69b1cb838f644d3465006baa82797f0975c2f96296d5fe5d48344003056700b72c23356633d827871fb2edfe24eecfc6efd436497daa35ef3aff7e011135a59224ab86b65fa4906cc314ce5ea29d3ae16a0b7aaed2647809e9e22ba291fd6de15fa5ad8fcde49faab4d4f69454acfa65e60a4adb7918f9fafa6f3fde453044097c90b2ec2d404237a25832272b111efaea786ca85a854db28156a69fffd4329e0c264136a4ca6e7a9703cc75b0ad3692838331bf1308b2dbb1acd9a7a8eef8e5a8c007f3b20c06c563c0524b76b2ff446d8280f9fb3ac32edcc28b65371b1f847cb1aea8ec3d8444cc34a0e4dae0a40c729592bbd7d76f01b1cd4662191aac45ac3146b10b132bdf3a31ae69d28c91205b07621dcf832cda90c9497cf733f1a938279c7dd00a67716486c65dea8a794ddc4e8f0b8b6b396e83b8c1a462e68277d648f5f4a848be65e956b05d54eec6b1f3a10e7ef74e51157dd1f784753563a6c1827ba1f03930de9ff785b0f0856f482eb2e4eb7c562d6e0f0d86c4bebb50a07abfd1d1af148dcb8b9f4ee90ecb51f9136147a3a0321035f99da55205191a2a7397d12ce9f851349962a02314294eca285918eed79922ab45c3af194b42e892ed508a33c0ddf6f2404d4dbc0061acd18b0178b5dbf07036c6b1129df2af62422ca9a611c1328e241d3a3410067d4a39ce263096e0cb3e967f021c281a1fd6128c6dd6ae16899055e060d5ee843e865d8ab738be6c952b8651dd30695899860d54cc8e472744d0802cdabbe323ea30ca7916c64ccb81fd69fd016033500fdc9125684eb1c4646c425c589eacc8d0601510b695a1dfb1bcfce54b3c1714fda567abc2012c240cf39c9837315d024009b462e9f05d232c771aa70f8e4446c9af6f5c5d6445ed656569c74652422d5071b3373991e8bac531cb67d41b63294680d8f0d6766519399090e2547741edb4aac09df3d0485d46bf9d80cb3a053e828342b14091391ab5a1b886722dbdaffa344b354f3e2424bca3728ce1228c552cc54ed0d6066046f8d93d2584cf16832382a1867e720fddbc7404e876352f7a17bdc55c1d6f0a2ede8e077acdb6b7391f5b71278f39c9718dd75d2a44370c9ace75760be6561a42ddc25d41f03fd58b244fbf7f8dbb5bdf8a59327a30b42ac46a893d6813f347ecd891a3baeae812459a9ccc791eec9f92a391f9c76256afd2ac52dc933d1eb33d860f620df2f58a0d525f6b453839ac4b27e643612134319fe2d9e55ce2387cfbe68f8798b3b603f5135fa71e356acf947bf0b7a6f76b403be0d4fdcef1bf2866b82a713212f9fb8f8fe306cca0a5e2de82d6640c3427177ec17b5fd6bca1b6c2a7d589d8398aaea81fb5d927b606495e9f43e879224eba7dbeb01edc0380512f9a08f6fc52f33562c7aa0762ed952f36c3c5bc041ebdfa7a5b8e765b5a24017069217d225a4521323e7586c9ea1779473001400acb96028fdb6ebb770f85a86a3ef053f1eecfedbf7f14a6aad29d769aafec8cd2f3906dc632e625b39bb80a853672ad557e1d2c4b55f5d8a7990c8ea32e4324602fec42f2911f92bfd56290e66577bdfa2457a051be972f6e50f23bbef99ffd2739f8ecd678a7ceb9d48e93c757598911498d4b9f52b33509efd5da2cf177f584f5501e2b9a0dd860233e52d5df435da2011acd97406ee77e2b6ab23fee7010855f0cbe475e6ac09dd5916496e46da0255ee5c59f1ebc6dbc782efe70eaed9d21eb83ab9fe697c7002ce3cc630634e868bf09917ee52c95a7e5679ddf06e85e6e02396df204dea4ac8f5155a63741592c2b77af64ea851b7703a00b952c579b498b66ba423e6275af6d5869784b456f02e0a329e035935a23dc95e3faa18669b89815d6c44d83b3b2268c3057f41bb3cb904bad9c7d3826800e8777417cbf067b1924060bdafa4f355987d1248a40c40cf9a0e7d88bb7d66b3f7c627dba0bfd82f3d684af195e8a63740e80d3a0dde10cc449f2add66cdfa11e3b566792fc4feda7269d618353b380cefbd8e5c094b617898c6392f15837bd9c32443452a4fd194421866392c471799600d3e431520cb2d09b3892b6b19b1158faa55cd8897a6fbd5e3cd3e11ccd004711bc7bdfade5f56a7bd280da9d9e416be2d4533d1351720a063759115fcebf97ebb4dcf250addf285d9ca158669f475b9b22fb3520d0f7bcdca8fd4a60f4a7088c7d5ac945fcb756952f12ed391b97243228a7efc00be3909f5b09a69f4e92f3d218942baa7eee0561035350727c751b2f4c8a3b18aba3423975c1c8cea94933d2c2c05bcac0218fd5fe14005d60a9ea333bdde82b21778439ecd207af8a9de86386292d04e7caeeb58f9e7eb43f4be6db2bcb9eb7fa88bc87bcc2e890428f5187162e4331fab2f7007d349d8aa2d34ac4682c07a5d72ba8c398d36d962d50f8cc6be82b50a0778600435d2bf3ffcda1a46a8a8f91c246af2352e11238caddb36f336f110a4b39960c6ae1cde17847d7e1b9a3cc25512fb244a74acb1ad68d1045d3bf412f03b5c76e2ddc50d1130c05daae9ac2566b9334388af3071edeb3670dd2c0072a29e80f1c6ab635a8bca89519d0de761c8ac8a9b35c98fa87309c7e1654b6e131274f61043c44dff82df4e0ae11f4d0d6c497a770c3305e6649749dc85320e945359d8122776b3462af19498b0a3c2704aee8a1d21835cc2fa4c5807bc35cf0828683c904c6bc1c06e3825613c9d9b3b54eca7900f40b788a2e627b3fcfc3013de703664ff78716276b80f08f010fd7247564e748b5a2f307c30ce96bc3242648ee7c9ebbb71fd0c5396df826927a40f94d3f5e441f97b9472382d4cb9d40ef53e25a0c7581229dfa8c9b7b8e0adbeb7e2bbc3b1956ad684e0f74db1cda48006cc5a5af942c9c4cab449a107614ee6d580de60006f9074f5f5ebf093c73db79fb4368d2abc85f6e9c652a8997d223f1502f85c92d7c8a31afce8c2196125160762f9ae38610ccf446c85f04d29d4ef69e3d3e6c76660a34b705c5eed29b89e2b49310272565218eba3dd209411c75d25393da24a6d732a6fc7d306934f207229c3514a3fc9127f9950a926eae843a85e94061b5999099df3d337d327f77d2a260d8d0a486bb46ff2a5157362255adc9cf5d9f562f45e0a04581dbe711eb9a7fa430d6ab4275ca27f07a707a943a03f04d20970208e16d38ca56975a0e363a1b7917105ba62e0e3452ad59cf7f2fc4e3cdb5c528c458c5a07bf70a36aac9b8c74976d145551c798483769c5767e9a0570b56dc93e7bc99003b3098977ead9fb326796a4286136e691df520283aed448a592bf9bc60dee0e84944f24f3d9565a2363e973871f30365a685209204d0e175da8a140dd630c66821ee12eba32b23919741d307941d126d02ae2889592fc2e07b45ac2a331ce3173d6f47e092b56d5a8e4408c62ca7729dfc6cab51d21a88775ced8fc7330b306b5bcf2ea7acc42c32f52ee97982aa6227038f0c104095c4b46d0cb6973e1241a49131ad8afa64da3ad3896f36422cdd352a4a17fb63e017509d7b4b742c38f08474c043eee95431efcfdb0f02bd44d8ecb44cb50d896fe0ce26de91db7b97d0d6c5b38d60734d71d7c5ec39b832f07034f7b22e01c9427319e9dabd4ca6a898c07ec7bfc14090444335604e7295c781a4dca0ab1e527c4b2e6e8ba22e0df4bce06bf13bc4a9b0abd0c0fa760ae57279cd2a209d8709873ae540edf5d950e46f53aba633008824c99426f0c9feefc037513bd80f061ce6e45e63a1821e9a541e22c86c83e5178566a3328f6d05eaca936630bc79312d0305ab640127b48f13961957380085aabbbdc17229a5deb2a130586b8e6a49459d4079769466b75a00f94dd087f213c147035a3a20bde7383b9e7b0fe79885bd2ab561cf8ef04f003bff53b54534dcaa96e245cec464b50a502980337db7a6af6992840e090d320679bcc8cfc5f58dbc943424b202050abb45db16c6debb9bc6bfc922fcc5586bc8220b79f26fc0007de34082e7d9616c0817d91d89454e56943c36096357b53dce22f8ae9de3b6038a8407ddd52c9d24930672713c87cd65c076e285ff2ff864e539ad4569be69bf024b7e67330080e45a388864d0c60dfc25b99bbe6d0121664b8657ef08ff86dd9c2e3b791eeddbfcdc68bf738f129110f26d74cf7293312f3f35fafdff38ebddedb1b7e6fc585d681db2a0048dc7ec0d3f2a62b08d03de12c85aa85e6d5663d9826caaaeb2cb008b4ab047b9121e1744edcaffa335469263b49216b569e072fcc4ca3d281068b142c9636936bf8a64d5b4834a5cde97df969c5857f7c1ac50c051f78858dabe6be96122cbfba62ac85d7405783615398634cd4d418061bdde1bd54cfdea05e722cd7e6455dcc65dfef101ef1b8e457e31a76f355480c45470e61d9128fa388a09fdf9cd6d57a70bfec8d63cb313b158c392e7e3571e83e6bbe23fd88b9c748cd42368c8798ae6fc80420a813ec2b6e1bce8a118009c14e47923b464a48f82f25ce417fb7f88f08598e5281a511c676a945bf02772447e80e780ffc0814b5b63a6f3162fe3f2addbcfdce63c6c6132f5a44b9b4317f9b16281626cd8f34443d0128c1727cf84b6b07c5cf5f780c54f8dec5072e8ed4b24eb0bf3c178d4c6a5f6069ce3330b6bf7b432897ac7b72cbb39fa92d083e58ef847cd0b115dc5317253e1578b6c45701460a912824bd94b852a24e0b88f0e66bcdbab9d199751619f264d9e4ea4bca215735aea875c09711bdb935eafbcc13592e06333d3f9f548deee5018751eb22ccb786bfb2fb819c4043129e04279c3947b1da5bd657965a63c57137aad5206bb221f971956b90fe00e577f27cb74f3933114570cf5784611ac50c919fce7929ff33913396234dddb870be7fee4094056b0b8c44055d0861b74e0876c52d1fab182693445713b0cc3226d6aedb14dd3faede857bd9fb5fd961aa31ae1be9968c2f8585c9a4dbc9546cb2cc7be2a191f5a8fa90f7e69695bbdcb156b75287285263e6ba28e01ae0da24c5d727df137d2f2c5be3005061560eec57f340b94b74b7707c125816ec062ad74d13b36ee05705ed78cab9ebaef7682f5c556be87c941125664b17a63a96980364cb592e8c9f5cef789d9d885f337bbb08f1624d47b474efc8ff5bba81e18c34d0e6afa2cb005eee4bc511bfc5241f5588d1b545b3d4409e03b849d7c0291d9ee358b9f59b18f9fc5b84ff27fb0fd4295b8604d9cb24aaeafff6aa71f3aa03dce2129eb0c366eb41c890d258c798e8cf5e37ad2b00d3acae7c683003ec8bd7f3ad30452dcdeda8cf971ecd500183276ea325b3a374ee4b47fe14fc588dedd455980cf6e94cad92d6bbeb1a597d7e6eee8badf08f75f381a0448329875761c9f95c7006dc2875c555340b8c7033ff03880c989cb740e3e131536ebe6a398e3adb849b2bc2733c5e8c2f493ee236873266ffff47e60f1f4ac4b2582d52e3ad0a41b83da82156ae6dae976a73f6cfa0b47402aca9975d8c6b35f1ebfee93219c2d3494ca3edcf65c8d00b70a259523132ad8100448ed5a8a84e884b2cdb70038f8bb13666a649b20d691fef9b9fa3c984e39478cba565885f184a9d2474ab3a1d858c01b0e65a336e25a35210da9a57db55c0f49fd75fe4abc075691cc0652ebce4f2c2f8ccd7ad23d3514cafaa47ec94f56139b476b6486472ae360f8e784aaab4cec5e86643822de7a51424b650f29eb078448d38aacebe7e9cbac58d068f6921d59f375763b141bf3a11138525d5fafd8e375f76163d67ce0145f957430563729a4ec1e16929efea95002bd92f7b8cc6e1a8de7156e3c6d05d2c1e7ed13586f0ace778c8e4587c7b1f71d5d607c579fd79479da63d712a28f4c6a827786e57799020abb2da667a4aa097cf4072f837690c4ed96b2af15bfdff8fd22cdb26bd48ead51de4440a88732411869fdd76b46571f859532001f98799fd8993cf870cd05549d8a6aaabe938a95b6624016b82308f025cdd69b2850c6e91631e0d7772ab0700a7dc0549f106365946638f185bc394c486eddec1f8edc14db5dfc5ffed4f01fad38eed6f5e52d640a818e4d2aa8e16558d4a02b2019e01afb752c8b8dc8883a9d1a2d1f148d79f524291e123bbf886078defc3f1e7ce1a9686513384fa258304ac23607eed34d8ea855d0acd6eec7e708b6ef18f9bdd275fbd03152211c2f577848857734a1132bc60cc2374556d8ea6a137a14128c139b67057c012679ebd52797bd3eac27a39712ed55f00a5903e2a83641e1cd5faeaba42401e7d4cd04c7fe2c9609064dbf3f4cc7c086e56f17c38c1ff831531f37573437ba204c1260387ae79ed810dca7126f542d4bd75ec8bd530ce690e25b68cbb890a1781867c491931ddbebce02066edbfd759b1e003eb1c2169a1e7c0c681612bf8e45f52bdab10922262164224a037ebc8001249096c1770ac518870540c329f0c1e32ddbd1e7c4d9c3f538f5da58bc021a9812aca718f65a3f7f790d3332f648dcea4a15bb3916813af3f8e63e2b5124ee6e718347768e572a71bd3e8ab7df05a7b096521e165f4ef883af2efe70756aba617cfff53f93f88ec977ef9992be890c2adc2bc79ebd871567900a29b87f0c3bde3354e7a956eb00be79ad36c72335df472a4d40a362b81979904dd28c2e8c3de7699b3f67a183549250cd7ce4298dbf7b059f78b19b9a8fc5ee9d0cb02fafb3b8d9ce48362069645e99ee57c6a3bb5b34021fd8273d4ae441d7eadc39b8c4c47296b0ba6e943fd4e622587880c565e8187db344c5b561ea4830e09da916c618465b07f6319ece158d01b8ea876fa9df0a44582c5ab8addfd3afe25a7f1d50e25abf6496a52223405dbf3123c97166c868eb6d2668244ab153420b255b4f4469db02e7c027474af5cb3ca523cb59080ab11202107c377083bd0889071edb0e466b2e7dc3fd1842099213a39b4b341b61e1977416961415af76a3009a1120fbd0a04c95fa03cdf29ad48450f6495fc7c3023a6e19312cf97f5ab0badf3bbde65c6eb8e877c6c6fd98dd2307935d597c73ae6818659e7cab19debf77112b9dd743c45fd46f780da544482bf2856972de19fe21dd83e8cb1492bb782fe3cb52881395afb6272cd1f9b0a67be9cd82eb57a6d9092583310efe14064c03025fe75fa5c0a01775c73d9a40ee42872f066cf87200dbb4be013ae30c29bd99874d6d9134f010e15ac20d1ecbb87d6a7ad53f2da5422811f0fcd854cff08e08b2f783405635deb385508c18fad06eaf868c649d28ea68596dd8ee4233c0e08f9a0ec20e23a8e312b94ebf659bf1e1d16969866d39e45061d0441f23e0309d2dfd48d535e643e59834e41ef2e313345ef74d244dfb0ffb7017c256d6de1e0bf1dee9e6c2edf10a88d38a3b3f488db872117c8947c504bf0766beaa2240ba4f4e7c5defd596b69033720dd2418b705195e584ac2d5a7ae66e5bbf789d2cc9678cfbc35e111094fd3028fff8e28f78c9b89f2db97d873133202d78fc878b4b263410dff64867a99ba61836542f016529a11b3431619d7b6426f9760f729f470ff62ececee4d55140474c9b9104f3892d925f87f1ace4f546b7f4f65eb072cccd290ecd9387fce99a532fa616de5dc36a02c896e6a57f9c88ac15d7a902bf8e7fcaa5433602b7b43609d4e4ce60f9c2bc4064a61d6e318d7fab241586a42bd2e0651c78345f9ebc131649203773bed57beeca8142c88e8950f0dcbc5b84bd9bec105c3545db147422b6a4bfa59d69e5f9d350c3a0825b251d1ad55f8ce56906a75ac42e32f10141f621847cc34f1c23037a9434d56f0ac790d6e3fd49d5b33a398d3e795639c416d5e028b2d54f0b28aa283368be8f07f1932270d225b2023ac2d609ed7ef5891574f2b1905b71c9322a997d39e0b88e7f017dc24a702372b960df234c105c138706cb7f07e89fb26e41ab804bf4abb4cd7cc1937042b8624a536952510ebf73bca40604f5c2eba82e0076e464261dd8f4fec63ec52e532a68a18f95bb09efc85f8b878d00c381754061251deee285ec0ffac304c93d9f9ea75122ef3da107dab9a1b57a3c49fd4a6d95a353cee9a4a5a5e92d2a481fa168ac974e9f2b33cf868ea9ef1776599b21cc45cb02df33c74d1ad88a7d1d79d0b12206417e76579e047751f23e11fdddbc252837e87eeaca4897477a55427d46b9e730e6875eac2cd1cd624e6bb12dc39b326c9996e25e4a7fb7f373373edc60b2ce87ba978fdb506937bf29d84a6e5e3425ba4929696b5bc8029aa87afbb2a95319e4daec1d8ca5d3485e4f4634a3f8243ce4896308019c8fc20ada012b689a9debd79b7cd9199db57ef77376fa301398d1328d8a6c0be37539024155e53c7d97c52c7966866998e74243bbdd24be6b3e69f9c151415cc9ce6158b5120f2dfeeea5bbbc9e205449e7a83415fc0989cf5078fe499aba664489d2be8b8832e190bb9f360174ef7e50e8328d7a823cca49b03d76496a5c52ae1fb6444c6c458d383597406939df615f630f12beafeecb358d322d8b0ba47fe51620aca3e6288cc4abc9e60cc2546f15564ec6b133120019ea2ebbf6eba2a4b9fd3c867de737741325e0aef481fae6d8ccc4ff8038797b3e34335c7e5fdfa99db4a0cfbf64e17c4c1a97d5c8af4549300305a2511739f70e55385b5907071beeba057aa3177ba28fc9341dd482a495978459ff135560045f4fa71a9cbac8a88284257217de603e691eeeca6332f1e40c5209bf33182ee2e5f99c6ec8de14929f2e42651b899066f3aba2d85b619f2eeed32e43f9b13382bcbebfcb9231c1db59071cd46acc9bccc974b826870e169c6955672e2766b71fe8b08f95795f6a2af117dcf89e56b4ddf20ec325927eee68e8c3f84efbcea5d0ac3cf94cec10a9c66f5eb1c4d86b15bc9b42c0fe0b678d48222db2dd958b0a2565cf356e5680ee5a968eb9de1b3912b765338cf0ff2cc4de995281fad5767d32e871b6f13e345537031b66e7f8470028a3261fa533dcf35c963bf4979bdcd2777db7d636471abc10d2d532c41cef1f3594e7687e5c3a69daa38424187c5a41f73e33a94c96b7090b27b167da1477806a0be62de8cd4b358fccd2289abfb460dd6e5870bcc0f792e6a58c4f6d9313166dc368d3f4c9d9153fe5d11027be4db9cd5177a6d76282d3f38b987a3724bd483cd5a610a68579071ef2e2fba3bdb0a84cf1f8b654fb182c4a876c01306d7b7244cc1e8fd7fa5ef91bf081d6140a63e562bc65e7c4cb95ff77bf2f2a984a5207e61633a41fb13bd578cb7ea9008f905d7f8a98e25436cb70b49ee119d5827c44fd4cc039aebb42a16a7f3cd085253e5cfcdcc4bab2eabfb1dba96d4834a212d44289e5e1275b3579466e4a2e0904b5d6f1a6cf59ec6391ef33f822115254855f9c57b7926cd270503818dce14ccc3a3b96e6f4e875c69679d23a69b8f47d5b861a5ab9330f6e26ce55c382021e25fedafe68728e4c73043f3ab4bf896d6dc0dfbe50801207254a8cee9ab87372786c3a53a3f03a0fef3656795123f8ef72c38d5200fdb9b4616dd381247488b716d7076c81c81f1d084cd64fc50ac18b395b35ee0d5f77dd6c7fa53c8704f1573a274b24f1c1cdcc74660a2b37d336980cdb7734f366be331e72d85167a9fd4d139fdbb1e8a35172d2cfe0bc9e7d12476d281864bdfe75c54b82ffd76fe02465794970b77044f0f7fb8d7c0897ab4be1736fe679db3ec8967818f66a62f811bf646fdcfc057921d46d66b08427cb4b19ffa0872b0273bba5af85c516ba28c3434b84be9601e25bf7b97203d2474c19b22d0545836d282bf8691b8b6102043d0c276015823f0c6859f0c67c61f823e62027918021a290c9fd9987c2ee7e49bde076d6e01b2e3152e92e2abf13324582c9491980e55e0cf2d256a26dde470b2ca7f0c950aa9090b26f742bee12ee642c9ba31659a2583558b60d33339bc9e13b75602fae5cf83d975a11d078f8ea4a2dc52f3d4b3a8a9151e33d75edd0ee7f55927e94a55a7f35aaeddf6d6534f6b281248ea6fc6632c2fd167d54660ca8f6a8a26413a71cc3bf6d1379b9627276b63fdd42f3e48d189bd04b7d309a092f0aaed3ffdc32bc778fe1aea6c79202cb2a7f13f4c09de4fd0cfcc0303f9f26887af8db10f967277c442287eaf216a4b2ed4d611a6fc7c147390ba4f13124735dff25019c9b8815d5a30c2e813363c38fccee0ac850ab7edf00adbc5fa94a16242e12ec30ff193db4e1380b954188e3693fb97d9e6efd945f5dcc3e83f89b6788f98fd1ab3b0bde855426eed757a922799c8bcca95a6b1f2386ab6f60c299db922aaf493444ffa3fba4eb29b8cd761ec4448054cf202e0b80fdd65561175e756fe38fa1f389308b7850247cf9973dcddfeb294767d1dd5dabb41e73bee77c3621ba08434335b31353a41714f05e548765e9df194a23d6c1a5d7eb2911c7469ad91d3af0c0c67f2b557123a235647ec2138a5549a8e4af3d67d55b5e5b673bc9efc372603ea4ec5bb6de6b1f83fa43ead6d4681b24b7b8681186d95a365d29290c1b2570cb70556f8b6d3c5837b760538455e96c86006f1e4ab4446360e111e8922cd12a7d76625f217da4bb4171c87e7ce7fb5012dbb4b976cf9cdece31f9dbb7d0a48bf4abc80ce361ab335da4ffe9293b21736e3b78912beb1fde8570fa72e7a8090c73cd85f2b53c7fc6b8cd41e9579fbf5b9fb6542cfcfd83991d53cdc0e57dc2a99f98c23c1a3c1ebee80f87952c54ab4f9583fbdd19009365c15f23ac4527698f82ae30181e9e7d8be2ac1f908e836de59cbb3b27e3a9fac362a0685809ad4ad923789505c611898eddd4b6bd4dce6c85dd9cbb0852b402afd14a623f656549f15963f4544932cc5978c305cdb118c530c28b41657765d0648f7484cae94dd4b8294f5ff973a70903044db81c264ad7c9cdf7114dadff6980bd269a388c4e195d3c8de21eea9b029b5c8726046e31dfb05bc8450a30614bd2d2d2580706b13084b8ad706147938f4aead14060816eb77d440d4aae95af3f41db53809c4dd46310b622be31bbe91446092fe5aa8b5081a681faeba7d6793bd2d16c643da04ed2681dd3eeef05c736d6b89febbdeba946baeadcdd145bb2a2e79c59729a21ed70b003a721866db0ab31216f743205ac37750a524cff7e69d0392b775826254d8bd710b9eee705189f51eeb9406ea7b82a2d55c8d0fe2708b033534de0d0c16537103189b0a2adb046402fbe08808e808e3656c51998ff0d317338a26bf8a6d4f8bcc0320a244d10c4f119dfbd1b21dc1b07253cf3ace0226060ec3e610a41d3284066c6b6211012b6aa93df4d797d0d9c955a7c5718d35858a0ac92ca22caa32ad8e5f0d216f1cab15a1b73ec31c7ae5adfcdec335f3c48d263c3014b7a75de5017c8ca74b8dc9d120b5b95e1eafa0912d38f846c631bc67ae93d4e6428695ca0e9ebb29cb8881ec4b4c45c27cec8df24c9f7c68a2e7b50d3d91a093172efa23d6f67e3e6daa35af10e15d943a7a8c814755f7edf66b3734d9471f2423240474a5dc83bd585951183ce9d43a9765bb74f6c0fe12affb0736b9e3b2109469201fc862d25057cd115679e1fafa47a1c13d30a414503a6e816fbb5b9e2aebae0fb6f4e7e2f3e3b73e85cb1ad1bce0f9f956a862c401a2a07bcf5d8eaec4a9babd23de6d1288b57deeaf982ed8afba562753e2b1e4b4ebf2bf4376835d685b4a1616310d57baf61e2a8796b0ec576810c254930190df32111aeeffd35a28d6814150ebbb2cbf7f6b46e8cef1d9478888cea4595bb858f196cded2055d1d97823ca7a6805f2e291576c7042285469c45c0b797b27fcdf91f8be73c1e583cbac1080506578979e8f6bc8a290c96279aebf9364dce1990e7b0276643d8f18c43583018822b797c07d9c4090e08b0b2a662b69b688d1ea9506d8ef5d1f4298fc7880a2ca57d6a4afb0ee9f9d1acbe3062993a82a043053546d6a79f06755a088526cca22f43259689d06476346ef0079e159ebbb89c08a4e321164c2b4453e0d1c2a9f8c6d442f9d3c3a0ffb0ff5eaad4502c1f9a05db567fcd7a858b4a61ae7636c88dc680ad0d635a89a020600fd9c92ceec130cdc2e519dc906747f9728bae2a3cc7bbbed0550e24dc27b2838ed260aea4f1dfece9f3a9633c939a62a4428908acb2133bfd5e6ea04dda8cf3fdc70dddbbccf1b69c871a14a4ce0523917090a0c5e8ed47abd59d472819f03a6ee5f229c233f9f5fbce93f2d7cbb0897f2fc8c4daefe7210c83c2bf15436af6fd2475877c1dfef4d51d599780dbec70706902a6d169261090dec19261e3a3decdfd397d752b48fbfe45949448e1ccfbfdee970e1820e98c2ba7ada12f2851c8d37db7cd160e55bd4b5714acbb11770745468ca3f6b6a4b4b8fe77a9fdb112da3ce444cde69b1986369c0edc42d02ef5de9068b8e68511681e7eaad513348a874849091c68ae3d1aea677d38c0de3595f6a571ee78b0ae2ec6af11316dac5c43032e45dd7433131f6a71f53c3c2a82b75b44e9047502a70ecd5173662dacb70d6b87377e845bec6a5d0d54c14610d8bb19a4192945ea7c5d762cb750bb1ff795a694aa8e16efd2dfb94e8a303540e6d094329a033c116598b791d523687063630330d29679e1db3fb2ac8e797d2ff90cd052f7164bc62e63f39bc4d553dbe147710d5564cf71f17817e6d65382442c48aa5713accca72cf6b17030181065bbc59bb104e5d4e72d52253d601c99259a8d285cf48f4fe6e0e4196f6baa1fcc24a707bb153a1acd78a1693ef2f78e055a68bc6b976794a63b986996f0a6665a4a96011336888311a077645eb37e6f70e9dd537d976aba6dd6d97cd38618d88d9b993ea5294acefeb318abb6f2ba6277ed4fef2d083fa7116c0f786eb5cc8a470bdfdc3c9ba450347b6e830b763c55be47890804e23e49e4d0a000b2c548e292eb42c8f96d6f6b211ebebda6badf2cd26d97a56d94aa0411d809aabb2f6f46bbdbb0a05b59e811eb7873dfb286dc2351a22df317debf8223f016308f0f01a86c82816a39519847d94b8ce1375be5ceccac5893bfbd52da1e997293d8de45920cad6a0cb617bad7bfa165948c33847959a7150e26e23a3e0bae3f6c0d1f0983a84e233ef59f57e8fcfdd8542316ddd23918f18bff9288a2f5f4226c27aa43fed16aa347e9baae36792d998af5d6f55aa27ae429fc7f56285e37868b39895b2898342ece29615e34b5a31b5e0adc0c4bef04bca0f0fdee60b7f9cc8483aff06e89aaf6d4cd765c2a599dc5d7a72d91ec947d71427d1c073f32decd96ece0ffa4a80f0c0509fdde0a6349ff12f5ae51eb43662dfdb14f8ad34abfa22dc96b88bfeba1f3243596dc91f27e8a8d8279059b5ddb88b44a78f676dd8b0d32f58b96c6505d93a452bdf4b06fa31ecc6a6dfbde5d91c2302c0605e9192cad0d960eb3e3fb5767b5b7cebbe91bce67a508ed067d11373dcf671f1f989d922d681638ee6e723a329c174d63ba635e29e7f9181693855983c99900d96b161b07c021d64c9fa44ee153d36137b2fb5e1ebc97a30d53e0b2719716c03c7333ed4486383e33d33b18f612acd5cff179778c740a25571f4a98ac89b77cf28a842d95fe8680031667f75277850569db561e3be78df6b102432f8df212d4937067050cc209bcee3fbe73dc4881936cb2caf7a2819c3bf47c523e15927bb298184249e4af6571cba2fb770841d5748186cd96c288594be2e71fc25c9209502f067e0f373953d56662da43e7f3488bfddefe7f4905c857d93f25914f384970668a8ee0890fdc6451192c293d1e891bfbaebfa1912986c72966df4b6c97966768352ac74844f0235c5943d38fe7e34d2bb4b6e3490e9c7fc0dd5023b6893baa07baf2f52a0f83c8a033c69a3539c27bd2340141e1e3a29487362c8a7d7ead64bdf2526e5e20b4a92de0e2cc499349643e90b020470e8e4707150c3bd2f5d90327ebe19ae9902bb030408c61f40ed7b015d14070293eb1e4bd93f7b23f95c35e06274cff206aecd913a07aeec65fe4a3a238975f5be3564e87f725439c0643ca8764c8fd2a42b04d7704c4acb8ffae9f66318d4735a799af3b5177a70e1ceef7ae14ee3f4ae7cef3c6b513701c588ac1e66899193210e7cf8b63415f5c6509e6663005ff41a3bedafb37dae01ef67ee168ec85f81b61f57224f0d47826b7dfdb2b1e2dda23d0e80bc0a546bd943ef9d009e8c2672ae08eef034527765a264e910a464dc77a8ba80f77fcfb4dd98d9421ca9f2838d4cc9080ba1b40a43bffdfca6f3fdad76b65ebd505930fdc90e07ec2df7f767451e881f8a8fe66d3c2399de6e70e9a96a6b33d6b883485dddc3c22122e3f3ff4f7473a43b0aa6c0ff91b198384a98e9c42712a6b28e420b1f6d64b9d57d5d2e286d375cc3bf499529c2641a9977e7f1145c17d36a930c75c47b53a92a82a3e43db5994ffc43afd257ff30da915382a839e43f17858b6fdc4148d6bc9d9d4fcbfaf77b80442bb6bea1d6421a9dc86dae8bdf2b4d7cfc8e19fabdc48f11b7c4554ee198759a155c102503af553306d3bdf234d5014f1b41e4d5993a54cf738e16627ef0db6d931d2da66a57f615029e223dc0956aabf7585614627f535713fab110fc1d22cb6772e49065461ab967b885c30ec368920d75f444d8123d4b4d2742ef6d5c1d993194e34b82bdf9f18f934d4cbb82556233c2c88cb8d2751f2d46a36cc561716ad475652ff399de21da27889209863e126d1b72e1222e17daad87aace226153d93c0b02573d79364aa6e0d432dfa5b741cfe9c9200502581f06ec28694bd554b8518a1d0f1838956c65ee636af7c588d08323bae9e9950caf94d80d946cba9eb949b132b2635a50a4ce855ad42ef08d1e80e18b36cfad9367d6cbe2b5c1e7a7e1d896f74396f7053b062f6354d0b16929b57a6f5b02b1948ec03804044356e274dc11302ac22cc4fb1316a720439272b318b5d6663234ef45b839847b1b3c3db9f709a11e75468115199932ad0bfc3e688daf6acd1a268b2cbffa29007aefa9ca99e33ca092e2bdb3d765c9e5b2040217a7906a7ca3006414f9567db771e5e43534e8aca35d4ed32ca870e5084fffb0b49b73cafd196c17d2ec5f0691009bd0489acdc0d67e35f20f2464ccc29a908c8e4ae2b6cf1ceee6bef72dd62a903b9e5e3053435a58fe83b3dd6f8b340f93e88bc178ed36d88b07fb75ff49ccf524c23cd3e7e8ba384a89ddb9fc51592ebe0a00760ce3a7fdb232e5617d1bf69c572edad4557c78e21419c5228830d7cd1dff1987fd26a33f44daf4e4b0eb360858ecf447ac4bd84b222434ab5b9219a30ae59b86aa1b3a38296fc2279ba993062bc37b802d807936960c2cacaedc6bd9a4b7047019df366310fc85ad024815818d69e05c46da8ec74b5b47f6907a45dc30ca29c3147fb6218b49ad91129644da0d5adddad1532269c77c8dc0c9a47a0f725a4488469779a3bdb0d1957dce0ae661eb81a75b4b10975d4a296330bec87eb8713e73111a0b06a76d21a24f4c31e50d0c16f3d2f9a57a04586cfc1205642255ff93b5501d2e554242001c84e1a4640a64c23d4620736c252445e14e3129b06ad06227f400375eb6b23391bb0714c0ad96218a55c9540a20babc3289c8ed03f6caa2825b0dce45175dff471a943f4bdaef154945c3dc5b2048d00b7c79919cdf6b9b0b2378de6c248aaa9923349ca637b853f54226dda0ca93a164175d405e11960fbf01ccdc56c14104edb60ac26b2829cc0d586a3cd835bf6a05d71660b80d3b44c82b4921e8b317507539afeb7746b0e636706cce496ecd6a63616dd6f060b5942d644839e73de18a1e0a4c638b315c629a3632d67b200e79ff472490c2915110ce2fe6a0aafbe39fd177d42992aa553d99e83e16c61acabb0174653fcf067a8e1147d1812d054c241167ec44f6fb2cee0cf65b0fcd463cf0e9dfa61877529d0d53cfdc617fab2b98350ee5582aad74e4c8107af091f2019e971732629b909414f4e8041fcd15371e86ad23d3b910361283cfee995080323135b5803cb5ab2d52e9c7b15e4871e2f5f5ec42f6da63c7a2b37478589309705b566bb2e4c14379f8d9c63738ac977eb626e4083356c4b308a9a90ce9ea57e4ed2a9380c2c2d10591944ac664d0357d8088efbe1ce22e4b4a0043d4f3bb1d63700715f41a2a347053ef909b41f0a358c6d17f1d4a7700e5196a6bbba4cc1bbc01d2657999165b8ebc4fb9ee03ae2b94e9941fd031a4cd97b0f7ecdd4bbadde07790de2285f22716d763ad056b442798082b4d2ec138386a92cfcb97349c8f0448a07727d231ebcf0ce58777eecd32698bbc697c6ba2294b28c3f6c40ad596aacb783b8b48a0360df7f46abe1e86bb90568646d1a409117746aeee550dd4c5280f949d3a78b98a9868ed889b9f8a200172deb688947928b3fad17ebaf8c1ed036792c4ac1bd886d3585c64f2223b8ac1d8b8b3d3edb4eeeb0adce3fa0003c6594a05ab2b5ce6b24266a6d048871fd23448928fb853714cbac45f48674fef28751b2111f14f3818cb9f61155963f61f6f862eecf86977463f2518e38958216d75818b2a273e301b6f3f7b883fb085f29bf35532b44a8ead3ad3395ba7c007801ab67abce484ad7e7b31604dc6bd12c699ac1f25d1ce5ffd8604215266ca2be01fe1f98f63e78594b28f0db43de03f378cdfabc9c987dbfcdd703eff8d58a80d38d49e0b5793901e07fb09cd5c3eac60ef6e1ffc105774656071a39294dcdfdd6aaf2efbbcbb8c3fab1cd1433ad9709cdda3114540fd1e1a803bbea450bda5b94a9b8981c874a5bb1bff714f77905c3cfe583c89966ca2adeb4dcbbc4cbebaf685d68b5100448ca359296782f89faabdb71967c0a0e9c38bf8f4376bf9b9045e0a34c78ad187fb6bb1c8658ef39cce97c5057d5c2988605b4dd0ca7611cea3ddfed834440bddb9bf97c01d9283586ba64fef8223e76d296d4cf3b63f18b090cdf2b659d3580c94d95bfa6dba3df8a665e141d39626591390837d8f192131818e246be87f0eb0de65369fa0dd83104feb31c76e0627a03ac606a231857b3e785d7b53009a098dd32c76875993fa87a047beae394a1c3a81e83498e495a1b9c294926aea21ca3f0977e1b7385e8042643a051ce409af938c7ad9431b23e88c32cbdf71d15083626eaa723a2cd3bd0678af19be4211990b598c6918cda2850f9a21586804c3be2ee7a9aaffc374d13916bb7ab8554475f60621f1886ebab3f147192b6d4c9f840f551a255cbb7322a4bbde471f4b10cd688ea04c2de765794f3b1f43efaeb722fc1eb975cc2a6767fc27f72ffd7e9e25a2d1baeb124a0ecde338feb729e1e517193912709b3f904eedc8521d5728b90142e0152068159e063a31bc7a7b9407d0de355b6cf60bd84d666061e6cf4727918ec3da8bafc80f4f3f11339c5afb3a5f7d874671c93d1155c381a7470802bdbd553989f049b17071793fd3136dc3ceb5d0f533ee6bdb74711f7953500eaa3ffc4dbc4e621ca96f7df9c1f3ad3b608ee04ea667033c3212e1f225f6786374d8d6cdac5c7aac3774bed828c22208c2e63e1548d8b9194d6f6f59e673d61604b148c2fdc530aa57f291fcd2c23091fece3366dbe30f4b28529dfcb6b5b5960c3a581627ac4a1b94b26914181a4c6aae309ebb0769ddc39b79a781b41707593a4800c7d4c83ee78a9c53ed3532091e42f5191532b2bb2dbee203d4b23b34e90be4c3493b3162ce1453d5839a88c11d4a8af85ac4af11f970de7837e16cc61fa9d335684f829489cf82c4ad054027be1a45ffe11188cddc0fde602718ba6b03d1f8348b197e6bde7a4e8c79fe0ba72d770d913cccee2a764df7f25d4d68e249a6721112620f6e4f4f8cc80a8df84957ab37bc34d95a21c5afa8f4256831d1d8eac15b2d28840e55060db79f82cf7ea659069ea9f7c0715492d3e72d534bc6cdd558a1a8b2921e8b089654b757b6c514d2ba62cf9785f2449c67403c977da389f80e0d6d80cc63e115b41b67da575628fd888b7eb5ed69f8b4f8cb9d7414c4b6d5ae700e3e21c587bc6ee2d5c07ede0b740b1ff24aa4da98508d02a26aa5215a640a1da26dd95f9f6d4ea480e4721fcca89eaf646c20734db6b393e1acae738509799f316eb7aa2382ba61593d883ff86e72dcf1a7b0406350512fd15029f147cf82872511e9d5f28d61afb59725f019be58438c8e747a6505dd6cb6979e2669025fcac96935dd64dabdcf53513fdc1accb26c879a09277a8c91444cdf113312b0fd24d42d114a2328ce6d3c60a622f641e0baac09dd0258f20fe56c8fdcd0cd62b4ee385c999ad6cbcf39cfe1b5e5f4e90611a2f1efb29aabccfd69ff41eec5d201158e5dedc70174a6e0bffa62c93437b258bfd19f8b51db852208f0f9f71330ecdb51d8d62c89070d48ff51755251c0212ac66121fb0a3e10465c4dcbd704afe4b03306b977795d274936c533e04687ee0307fe02ca4752ac101e5f85b87b6b75ae6710affff1e897edc1c803cd70d38291b30423cfb5865aac1b6c4f9451fed38d6e80d203b660443dbd3092e0456842d6c70ce36005e62465f1ab9344f6cb378456d79ac36aef270ce7870c4ceaa262fffe8c2c6f4dc221a0a7ac77a9db76904aa069fcf35b4621da9e47d70121a12a1b6f3a97a449150ca4399449894c60974c27879fe308150c82c83e5e7c99680797795aa7812ff0b25aa8163cfca36f09c550cae75b474f48668aeed62dc7c880fa44701e5c11df97c946f526b92834b6e4d48849d9c20b0093420e2a8c856c9d608b02cfd5d91f85004f16b1357e1ea62922445fb6282da4771ecbbbe3ef5f7a13d42f9b6b62600617fd5f8a8a459fa91df405c12b507d1b4daefb2fe112f54190458bbe632f356dbd59423f7b9e51219b20162e8b22939469e84922af68ddf94024f2d0b598b9b9d620f3a477323a1868afb36a5dfab04b35288ff7b9af5d1fb75a5897936863a466b6ab30f57d597f7e45326ab4b0aeac859822f26cbfcb6c03a3c80e56236d75a364b49f0d7ba3cbcdd757f8fcef17f21fc1a511c51d7b91d75428ab9e04aaa20f7c95177832c85f5b2945ec9794e862b6823c263d7df9d3aab1e921c52a172600980a9a48fb0fe69ac148b155ec4e026ce29cc70b767cf1f1fff373c075928cf38ca8f70b37d1fa557123ec27119f62ddca7abe422f2653eb0140766e004726471bfc0fa0a3af858196a413da0414fe243793e42d28070f447cfe4ff78d9bed9e327c91c33baf8b2cff912a0ed217dd87ca70ce77022e8af7ff12a50e0763549ccb97b4715992fe8a4b2f9d4ca8567dc6859f151132ece0d1bccfec3f40c4f22fce7086d71177dd69c24b634c3a2869b02a6065e48804b3133ec33999391c2c0878d81081a90b91a332afa7b2cb0a508e998b251607e14a2a08dba2db90e43257b3a9882c385c0cbcc04ca470776af257ad9509ca1a19231ad095a5493091ed8e90137a924f46562181aaae724a6a4e679281bde3c42b444f8a66330e674e85b3328092fb127a99bcfc76741956e0541eb13e814d585cf7fa29cef000946350f5e3eabf13678751ff430ce1a41fe208775de7f3c59e06cf83778b45ef03d80b921f255210df226557bf16c36ee9fa89fe813eacf1393a8734fedf29ac7355304adbbef8311cfd6336c40823d0541f00f40e535e748ef84097971924bcfc4eca6263ff9b5358d1fc1dfddf18f111c6e9d6cbe810bd66a61531df5bf5b3483a0522b515d320f3fabbc298a360d55515bdbbb65202b1bb3b2ce5a9fbafe9887a433a1a38068aa33c06c389ca4d147ee554215878dd0bafb044dbd1c95c76be7757c642fa80c80d663a3b94b61141b169f168ab3c1147011b101c5f05194bd8c2028ca64b9adae93d1c18d9f80969d6cb9b81b15520780931a8353c5b532a0a32e09fa6cb8e7f4a0611be1d79ef0e2e9587a6c325a92cf4d8de4d92173b256686591a340fd78d35d5dccd505fd81eb5da5633d6a3a8820c0635cdc88ee26546527e759d0325d84b048f5c50b56765ec2eb7b89269d0a95e5bab07ba6920a4bc98584e63700d5e5d54ad89bde7620fdd565472cbb66a1af1482051d8e7517eb4601c00464fe1ca1de54629a6b9ada61659c5ce380031e3b83b8f37090978359e0def961dc5980aaf423f4622d662b72a2e820971b2bfcafa10375a95be349920c4d5329aa656be3d3ef85191450ab9614af0951289824c8974527fdb99f4cc388e94dab76dc7281fad1da9822772400d0ba49c96bd12c9bc57cf1fc2a45356e9c3c8958b15b4cf63b34388c4d00660b65154fca470ae30b0f84646adedcb4e1c5085083b3cc0ffd81cabbd1b6f019311f691e9637770c7e13a8983adc9785d3641bc0c04904603c59c9b80e836f91a5eb0dcac7fe7e0c1c594c27b8888b862838e6302cc0038ca1a44198723c84ed1f9513a311f1284d6f6d03bcbf97bfd585c2786b482a471c789ca8012af2c171ce6885666f8a7679db876e26a5ad4729555eefa34c0bad76100f4f00e7ac6b3c555ee37d6a63f4b82c083b76308d2cecc8035c1c7a966a0d5d320cc67011e52aa3ad66faad1d65afa675f1ac4569873d69412d1bd85119dbf46b930e41249e5cfa0939339c59d543decadb474daf263a4c89e8f61fe0835c46239d08011e4cf914adf0d48c7b64d714284a19238b42aa52b8d92cb1d2cc4488e6e8e24fc3fa0fbeb4e9a2cb7bee37e3ed2f4731e0a6279c649021b0c8c602de9b0518b17bd04476680e600bfdc134dc7bd52fab2425e477bb0b6eb0ade2be44fc4ef5442b3b4400900f1153a92ff6aaa99d8a2a63729d6368690319a5f8a89c55297bc7f71a8412d053d0f6263cd98a3e1ed30af67141beec4dbf4c811f8588b5e5bf8d7e97f5de5d43f6e7907dee2439ba0650384dffb238ad3e8b7a674bc617746da931b2fad7ed5e254a1be8289b0d28a791bf5013160e00fb352fd2f498b11205e314a16608e6f42cdefb1ffd04023fd683291b8287797f751d027e03641654c6ae61ab18ddd75f2620c1e9240630a7eaed6c6c1097a380422ace090ce486f832d7314ed82739cab598108a94e9df81eaf59c5dc117ac45bdb211d79c03e5503f5a13b16c91e54f6220e081e743e3e37f024fe0adc6b110e2a068e89b2fa0569d549a20df7f8ab8447d1ecbcdce5f97f1509a5f9a12187d0290231192a0db5946a446b1791f78da55ad4f2df6499e025327eeb0a41e027d30a9d6a5a6e81e1894483477968540bb7b88fef8bbb77e5b111aba40458f3de9bd7f7a2d196f3c80195f51ee0a4fe3dc036de5ddf8977e9a6ea225c9ac756520398f637e2d14971ce1f7070778e5d0a72463ecf39c3f8cbb1ad5552d6582c68effcd3f714440d9fd111519762de1e988916f3c15785b36fdc27f5b569c9207691f2f8edd69bfcae546d32b502647ba794262f56fa41e93dc63a7f4a082ab071ee9e63a351084241b68750369e2806bec4c740a1af20ab1471859a2b62fc621405bb6896e73628dba953f0aed727e8e48086b9b785144a9bf8b12e17def46534a8c7aab78af61d84d2d8417fe0c176b257a0aef07016f6b48bf62527f465c7ca0badb053e4bf2281d8bb52150766252125f3132377496a67396325be01d0dbe8d2ac25cbed8eea94b88dcce8a57be5a47665b92754a99b5092f0d65bbb030c3b3081c149a131da5fdf12f0a0980da547a30857d69fe1c0d18ed51f67ccbeac37263e0d7ef98d2fed3ef4741ca9dc810551d3fb7e3ca936b0a94ffea276b9ae88c65cf6ba74f26db827b7ebec9aff776683ac34bc70b290026134f18fed5f2b6029dacf76510344d7afb34afb71e57c55e46a1fc1bf6d77cabec061c0a16f31463ab3c0e14a0e55f01875a74e9ab3b9c9e432c7e56d1ac22c532acfe29c35d71701e444f152a6fe735937d734dd72cad9fa141b68bc6f51f85e9745a6611e9d99f934da512b76fd5a1a1ac7726844466d7b861ca3dc1ba8c4c8ba9c29a15ace237a77f9496b048d40c09b45ba920bf5dd58fcbafa7fa0aa08a8e44b247718a79b3aceca6ccac38cdab20308c98cd3078f5c2685eb191caccd6dd7e21bf30ddd8189d5490108a72438e2dfe76d6f297242b9fd84e601e7cd2c6209103aa3009fc822621390931d4447d8da6c7614c152fb7bed3f9b204c0fb2ec388f0f075c2b3986ffe101c237a40d592eda1e70e5db532f66ce3f95b04d96a87d370ac4dfb4f484a434c8239bfb2e6d614897d2317605a8bfc1ff493d0d2338c87eb6e13b3da3384863bcb8f25dd83313af8d8445d8773241f9627c25cfa2442079e159e1d802803e31e5dff452248b66dcf0c60529197f94c9d814eb2d329859f2f93a4d25d6b15dc8cbd9b5397a95708c782562aaaab6b70b2ccd7011be1f1d866ddcd42264b669180e7d2b716fe34ec112dcf7df957b5f6d533f31a2fe8302c67a105f420c3394fb01b12cc186b16352a03ff0c5e34d759c51dba81022908e8a4cbb1833c9ae986e2318fa50d524e61eb6a86deb7f651a011e9990f0a2b6894c86d1d86bafb52ef697aad661771bec4ba58d689e2f483d1e34cfac1a6c723e615920342e97a18e4f0d9b48e3a0021a5027d76b7d6b05fb4411747175eccef0584b6d74954a35fcccc5fc73def0d6b1521f01d9d66cf092f445640e05d3aaa86049c356df214ce131f57a81bcc75b4c9471e8a34e4a32c0dcbd7510c49cf1708f321f083378f3fef281cc703ba61aebe7d052903dee89d83ff06ee5a5762e244f8a40a46d55a188cb24c68061067b012d6a52bc51fc59956a576050e328f4b01e2b021f631af527490060d80649d35b07d12d01d70b4ff1bc9fdb2052b070b142042b1d0d93c9691dc0e11efdc830eb83bdc6cf1c8ae17a8fb2e1e122c26d6f4da727b486d7d19d8d5388eb313c9c750d328af4702135f0adbc46ff7a4f49ce8b55c089b21d763431faa661c6a750d8ad2e14d39303c0be965a6770bbf0a8e8db14d6c35db92d8348d5e189b96c30160813077b94f6e2e9cc418deeaacf6ceb2dbdd68bec947964f76ee08ccd125ed94eac7e9adbeebc1382b81e0369c4a9c00b8d898d8e5cf93b7d04874960f2a09bada519817891531ad39d051f7f3259d0b79af64a8823e70994aa292f7e7fe97f0ee53459f4543f905213c660f5221f0bcaeee29085fdabaae590cdb075d23ef7a82c231c5a61a8e531c6dd8c6952f01b46e2e974d9e4f397da3d4ce921fc3fc9c4c759ce37bec0ddcf584a142b8bad5f1b50e13c58836ed2c710062c41364027b364ff40b707915ec782c271479d672437125289da677214308c39bda673219449808a2a7c2c5d46d342a21773ed1a9b0f06f3e14476290d6991cef3bf0acfcdb2232f154bce030df3975b3bee23ea31f0ad78de511bd454027d24d6dd8c233b1dca5403c653e8385f0474ee40ce97b17cecef6d7e8ce398f890405764e48c9e011a6e3ffef8e6500812a830738d0343d17cb62d48c8e9b7d7659b52439b85bb7488d8c7ffc2523025037539d0504d69ba0ea290516bc5740745ce66bef611604d6e9ea03da501b840e7ff2cbbe7e773e5218db4e5076711bdd04216ede91b9daf619c345391e69451126bc9b7d8c06d4b79a84cfc9b248e4eff42052cba2f568334e19f3bd9faccb132bf79a1d904f6247a26abf3fb7615e1476e5211d24cbb099f179661c3873907f14677b42be9d5d2d24d2d18637e22d20e39250b121c7885992e654917b7974a851f660bd663e6aa949047f5d24fc131694b1fd1d610a4942d1c015743217e566d7b96b7766bdd044bcbc7c6f966226eb749be64690c88376e827b5bbe6e1d7cfc71112016004e8a6405ee8f3a9c9bee902d84b8490689cbe73c8e3f385c4e4629a830d1a8bbd9261e70c7deb74d6dc845c43071695829e4d80e28336145e51c28969e1f58ee76620f21576af8cde038488835a4ca95abe79a3313280d093c9395d583e281a5fdec9323e4a5a9b1d9f12ad7267ace9721778f2df8585424fc5e50006428cfd0b69fd4e1607fb9decc7a6430c5e42e720f13dd304c88f150aa48b11782f80bbd2faf89c08898acc2edf4a2aac84f39019478402f421288bf92803b625794f3b29031984714a60520c6c4201a15bbce2cb66d2bbf75521da08fce07e5822ac708b3e95a3c05b36e4e3e8015726e0896edc7e1e468e96d014e1d159778385a07427c3d9625a2d14cc401b5bb8fb3da0ad36a8018d6517a0a23994bfc0c7be2aaec64f6d83f20f2962c1bb496e2a20f0628c7f88093474dd545a3ebf49d6ce13b450c178d4f3ad3035620d22c074d04169aca18630c61e38224d2143dd90f1b74536c89de0bb3ab270ad827d6de0dfbdc896634e7cdc9778c49607d7ef3966bcb79b9eedf886e154fd99bc7ed9130b0ce7988e9d0ded787ef18aa86ea09e072255f6b637493febb6c6169f8f4d35e968731a9fdcf8f511dd2ed4084392715a7d4f827709dfc6396b8f45ef9f9e16d39a856dc181222fdf419a300d8d47fca6263bd6f2a4c37040d2b10f575455d8a785fca615396ea862045620bfa3113fd01f8b3f08e70a3b38cf2e056d3e49fc0da3660118de65f449909be06b2bae3c68be8a4d126a8ea0d850ae3ac638102206959aae80c97bb5c81c2dc614106d231a6884a15c0808896804e904d3bd10d0db816efdfc7706f304027305ac82838bb64969f72bae844ff69963a84ca2fa997ab4db0c9f03c0373bb14f36a843a063524d0f9474af02f3fb60f25a060d43a37b988a2d7f301e17d199f472fed6b340e474fbe44d45f518709eaf766cc06a892f690f1ca8f176c48e0d52e016870afc13c241e29f66fdcdc2b3cbd1a989da49627b5d6e017216938852462b1d2530e32377468e4447160677355b131c088fe422568c65471bb99861218fa418f54cd4ddbe056febdcae92c656096ad88bff9639662fe71693f0e4a0f493900deae50ab57e86cd194466074a909020d81dad7e77e1f0993d709f0019ab19371aa849f067dc1e1909ec32e38d7d7123cf0971d996cdaddfd0a8c99261bc57c4de3369d9f9627e03bc15d99823e74841aacfba127ca3986b3f0436f39c1ab17bf5ac4b6d8882d1cbadbf2cae2d5493377896fd3209284527ae463d2e2313946ff3ef861bdd0d409afbfbd97bc9931deed315700d419169e7aba0937ccbd39ddfe18f2eeacb2657f45fa55f6359571727d25450f89af02a0c221d04c712db0daac12d7fc3dbb275cf3912ef1d8ce34919f5b33e137c0f8a9822c1cc87a85f1efcb5de281bf70e32b97463dc01c4dc03a7947cf164ebd4e47c04ccd408aefc5173b7c40e8c176cc7a73d8c2faa9cdb53d0e343fd914f11491e689ec43fdf937caaacc38b6dfb201e5f145a3f0e38f604bfdf4f7d84ebfc1fce26905885eb7878d2c7d6bb7af2dd62cc3cabcb9354ac16bde8e74253d8831e7bff9869fa3828d3f8c85cf7730964e8c11c86eb500176beb009492466a71a1f3ee4e08f90ce051677405b7940ff037d95bdfd23c1344a5a0d2c6d1983c0d22b1f8e296994d7ebfc1005f4f0fa52b29b890823362d1c756a694f63ed11379e82c8adcae7aa2a79b9f1644783cefafa69604862cb5df717af16521a1024db08d9d9a6ba5d1d995f31eb6e4beab2af21a2b7fec462ca81e602fbdb8630702629383bdce49eab2af71b4ac82de7567edfb30a50ecc4f7b4cdf263d5f0e1075fd48607d2e35fd78697097643252619d1834503431a28cc186b4eb7f2734d4addc197dacad06f734c6e6f199dce14ac2f8ba6aa95842a79b8f69ce60ae8d764d286f3b56eab5c7673720da584e179c75bf8a2aeef4b87d18e45bdf99229de80d6abad7c700f7154265c502058d259de084362d9e00760421799224df07a5567daf83c1ca9e0800239ff600c6dae82e6e2c6b65199a19326879060cf7baccbf1c0f02194e489cf274b269287654c7054437a64d0ca4e80769158146eecf925f4c065e30b83393f3f74909800eced8bce1bc9d8e68aece206e36545cdfe1ca62f8ca4c4b060ea9763eb891df0cc8923d39089e39b35130558560eb16f16aa38cb6659bdc786a816bf572be214d05811573acb26bb0b243507bd3918c0efdf29c19ea87c8052659fc3233a3cc84d60a258bf9baf28b9f3e9dd5617e0727b21af062b71a2e676c2b411ed6cd1e9a481ad0e81500b9a8a8b4d007ac8ea9b983beaf46246156d5d71727c86465ff101856965b107e865055c1523971842fad854f62182f17d4d5c76b34a0912bf6da471d7acaa7d911274809aa409a35c25eca1156d4555004f5465f72331921ed7f6a2d761749d3e4335722d48eb5b97bb5c818a2517683928ca5d244bee6ecd6249a647af3678f66b5ba501d3832e71c14f01de225d5d33a654fcc00507bd18f81810fa086d6b1430ec0eb5bc0540da343e57267f9f616e6ef9552ca1fbcb25a924d8ac4d2bd80aa4cda93456af57d49871cdba65d6a93fad90efe4c15ba2bd7662b4d856255e584807e18c2f5e6e58938add65e279b32f496cc86552b258328ccc879f679e8e231fd06e40fc02ee1bd624047192a19dcfdd9288e51f65cc61cbf8b9da8901acd35f646733ac9b5f0ef9adbb6d9c0a673c781b3d0bd0a6d409d143966147fd728394c7298714bc855520e5721163c0cfb9182bf0939d060aeb5a2e0ef977c8d5864773b8233286cfef06aa0d7843773dd6ba2857c50ee1f149895d1533287468a8187d7fcbcfbee3c14500cb9ca6ff3a53b572e5dee4dd1a6583095d9f44ed9a24f590686c220e8e89965af62d92f686c682739e049e495b6a965326e90b66139837d608f1d58df6994215d7a0cb363998fd5a2403d20fe50a2415293396606eae6efe7228b50e17c7dae0dc70bcdd74caf32c00dd035b5cee2a24208cfcf2fa0b30c98ba2f9d6d4568ea27fc6dee473722c37931d4e74ef99bedcd4faa02f17bfe25e9a3cd4ba2a7f58f33bf1a3eac3a72123db3019b14d00dea0353ad53bc30bd27210bd22e709b5f258ef4f7fd2d72eaed1c01fe15b7233ce555068f037b8561fd0c05e287d6ba1d7fc9d74674bdb578759cb125dc8983316e952afd784087aff33b7d76b4f8a3702c0a7f786cbe0cc4bc8e7ca5f2b7ac540439f77b1e354c6941cb9124e8bca509cac9a6a44d53d652fe5419a9c448f95d7e5c80535957a37d2443b43c1881825ac91408c357548fa7115e71dae8a75869871ef6b4c0b3aa30af652c0274b9d295fff4aae2427fc20c2103ee00e92a71a46b8f3859343ec28bfb89881dc71804cf49354f0b11e339ded9cf2ccc620a2fd91d86cdacd28250b17d8c00f3d1309d124e339d04d5029e69f30e8025555a1a59876b33e668b35b08722eeffab37246a73b9d11f6a8b71ab4fcf1d6e0195afc0c1607894f8c10874377620fce8f9d0ead0e452434b8bf8b1cd3858d65c257b748f8504e636ec874641381afe890ba1b896952eec355fe5a986e370cd6dd4d37feeaedd8d77481d91e0cfcec48b30d667295557e820c5da9a8e8b9300944d9ef80c771c01f5d45214d50b5d68a5426b1b0bd25239703edad5c166035384db64b43d030143f2670e3dd5ab8ae8c5dc9cd19473a22a8ea725ebb1ffd12d8168417f77a1b60da6ecec6ff929f41e9463dd7e8b73c8196d7d743932c84785adfb01ae0fbae974712f2d22a4a17f3e832683e155c84e06463b3dab8e5c0a6b9f7b75f87e91016a6f20b353c4debf4f5c2dc276df7bf9b02f01ced9e38e4b288fcce77dfbc40a20a7a6e3b29b02f6f235e1e90a0a25ed8bcb0eefd526ac0cdfed1cf3459d2fd5ab7f6eb705b59ffb74d3ce44bc9da75beeac50de98f5878e4cc27eb496819f0943524b3a0463e3dd989e58ddfed9935be61cf67a6d7c30c22a21871af049b4b58867fcde3b65ff570961c000301e5f6e4a15bf47a174cba0dcaf9ebb8bffaef638796ca37fb94c4646fb5b7db2f9ba7c9b445994987777ed5b2ffad71195b7434dee7f472b77b3b39a4e4549a8b730bea9dd1e02dcd1bd24ba633637eed7908800d304e212129ca2158a8a4f65fe498595fd804ae47028074f335e9f2f80f530f247cd552ce41c61a52886afdd12c9f4170d40571a0f76c3b5a4c107f4afdf76cc9334195436d4975341a1104dd83266829d92ab199520c185221f7d7583c14beff93679147209b0338c466614c6a6a14816b544b3a879b375821247a045d1601c142d0c5c9151b351a4f07f929fc5264cea1716e8159deb6ac503c91bcc4c97b651bc2467c5c163df503176d79fc086a8b8d127a070370fdffb642465850debc55a09012a8b05cbf2e21884261fae092ce68caa99602312f7fd4f0bfa25a56fa75a0daac18a97cb6d65f940a848982590f497d77603f4ad0c83e43af79f9f1573cf3b9a841ea55a4628c867be1a0f2baf36b59bea75787aa4070c8aca70759ced44e515e9938db1bf1d1d2ee0d1b5145ff7363ddb24b067c13f17e1ef41ae345aae87715ed5f0a8b401547887e1c5036319b09d47880808bcf892d4587d987f97bd50b650855f70aa9f038fbe4f3d3187c1135c0a5f4b6efb79d00b91b917f3a7233922e0810ac13c6ee6b5e73b95ac0cbdfcbafa4f0fa1e1a6f41594b1057fb3b8c39eacb751fe3091cec1a37d96ec85443ab1f42d86f4fb997b673d57c2a1994ec5c1b8f80e90b33e648877800246165753e73fb960e5cc90bb8cf7b268a42c9f7850e1f0166267c61972fff2ebc639adcd2071b07e8a58ddface1b2044b2a58e3a6b912a3702f55f199e7d9d07a5eb990fa1dc46b294f7ac1cfe7fd49a6857d00ba77cd210bd854343c9193b97291aa0fb37419acfb17adac9323cc8474af9da7689839b0ea70a06f23c62ab2d5bad8593086426c4fab8d82913eb45251c396430cff08862eda7d77ced49cdd57f71dc0032a06aca3ec3b142bcefd02d56e7fce13a8f2c186eff4bf35ebfd63c17b32e19d78dbf8bb7040234a613bf89efb2e55c022a600d1f392e02e71e8e383f1e4605b66949637f7c49e0e9aa4267dcc73f8d23e9519a4ae6d5d24e19593fc94a5a44e0cdc88f0c99bc47e46cc7581b57be6a2f01b1322a6756e6f6ba92b3eb9e2eb9129c347f84cc43f55b2ed8828690917da3dfbab84108bd270f293ab03e47da0689c857549ca99b5cc650e214f86e68d518d54102175b3a173c2a4d2544441d600a93c0af4d4eb5dc047d6a7c3443049c004553245def4b95392be8b95ae01762b0fdb50ee837cfd5f236ae349a13bac4350cc56608dc1c2ad3faee4f537e7265eeaa548596c395ae355c3ced86f8c8c0e011222b0d59284d392d712bd6710d5df402cc06486387df5848bd338c43880538c31686cdadfb6c528b6fa7b8ace765ad7c7be16f451209a114e40be145a5e1030cf51e5b7a93e1b3e1c30a98dd7d8adc22d081c6d139dab1adc009c5104103b1ad2983dde4e63277626453e3dbe07eb48dd1cfb9a4caedb4eb5f4ff9f712a7eeea9a6453ac46bc92f21d4ed70244322128d882e9b6590826e400d06a697f0efa89765e31322f8a37d22cb0b88b8f565682ec0a80bdcd694ad993299ef99abd3bd222c348e9d34decc80635cfea2e97be62313507fe3f66fb16c81aa2aaddd4c32066befc3c73b3e13f25280fa26732c3bdd1518a1706f4acb62fe3740d7fa6e3b4dcf7c3107f5ea9a451c08ef7c2f1223cd1d5bf8665e98484fed7c4648e19a5dd1d3d44a61a5ef8250f3a77d58bf72bf5d7d6ae7fae871ea9143012b45ee228569cc57d71afb410e18ab56f6eb9dde58e8f1c42160d13266b977ca74750787c72f29175879891d82d89d0f382e8b3e130a5df347eec67b8a7b9c1b7907f588e4f858d1d1e23079a1999a6bf305ccaef7bfc158a74a845599b51cfdeacfc9ae2471cdd06a53f514fa90be85037d5175aa252a78f0d46500ecc98fc781b3f283b1d389cb1b731291d08f04fb841cdcf3943cedfc7788d6427aff6eed53bd88fb3db7ead7e1bd6f3db80f84e2803ce90126745ffe83f197253e663ff91d9150760bf29240d7ad5bc02614ae4af5beb49b572cbb0f80dcea3c70ccae7e21122e45cbc10271343f4711ea59152d7c88900dc65deeb67e2a7bae1e673953181a66173dd7ffa934502c58e769aeb405977ed0a896467a3e3ab36d86a9e73b4f943a7641651a80aec1cf5c5b6bef7ef1c3c556528e71e372af3ca3b6ef3dc240eb211900c715315034d970faa73ce99e385834950c553d1844f92d4d5bd0e816993712462dc3475a7cb6ca774bcf77ff42f9d17673cf1275c975b3d8ecee2fc032049d2a55b973a83335f81ffb2892dcd840963f2a2cba58204bd8db43bfb679327e731796cb5cf7568e5ffbc481de803ca554267ca006d020156639631fbdcedb2a55b53aa06c278db1a4a5fbff67395625e699d204f7bd0b5c13cc8d7776cd0494324b4ae03fd740cb8b538ec0352dfbce726564930a076547952de472c0c88454418cc92d1377c5ac63648013d7eb5f8e4f8c1493935b386274636aa3d365dc3e0f33aa157c69a76680c8b80b2daba67d7ae58672c623b1614f590507b08417df97bde1423b401cf7177ecac4be164da1a01c9e2ef032792b6e80cf2e5eec4c59a5bf7a1138725d106e0d80dd10f98334f768044c90cf446ce050c7d613a36032afb97d6bde33bdb428078d163e4755ddee8a12358209f94d64dde3f408d690925fd11e0df272a5a3275f79b2c2a790434940ad5013e0213f5fe92e01bddea17ff968be41de6e6deb4e7ec841ef22c89a2ba0ecb155b1e613575aaae88c52426926178b8853f4ac052b0116351c6744b4dbd7b1c98c38f7dc64151e1cb614b478a9dd03a0f9d489468537eddd4395770292de7d600acce0d04f862bb2e8100e2f11d72dbe8d053996c4e662e1ba4b9b266a5e4ea3cbb224a36f4c13f70ef8a6d775916dd998225a28c92464da8446b8332bf7ef7394f109132c52278f50c03bc18a9b9a4dadb5319b1f9ca0864b2cf63837a780efb329da6b2dc20d880096c8e15c77314bab25f2b524a9aecc3e9e8cdcff0438fe40482316afd1c376f3e80ab60d5a0da08b4aba666c54aedadbb5e921e2080e54e6ed0109a633e332f39f6035e525b1fc454649aa65a3af916b6f10e7e2014f2a6858f3f8c88da60b261a1fb8e85cfc9bf10d63a1963f6e305de38d827f508a63320556f8535e5d784eaa864de3c00ab9943fadbfdc0f5f7dfb34215cfe3debeff5b916d16a86db5a5966e62fbb2dfdcc4d306ebc634b044a970d630065c4fac0891ec2337580b48145709333bc12cc93dc17130359886887c4e2c69683d6fd049d954dcf8a3ee2a4a4049047c8cce31a387c9a214b4a89606710e7453ea3c3ad527030de2518e919287484bcbd154287ea0d52138ce3bda7a4974f62f574480b059e9d495320fd0743549ae9e83cd988206e328edda912c8dfa99173500613296807f2f1b83048e1c8bfac488265bc73cbd3fe71dbf1db62a43e3f5dead60465464553bae4e64237c423b6d2d3bd073bf836a3dc47eaef4149107467ec204718b603b8fdc6d65ebe27efb240ed003e39fe7b5252bbf77b8e33763b0f7d3be59de48d93c11ec5735ede9185e2b5eab1786520b40109178203705af679d762970b5616ce094583559ba27070e78835c68314c270ccc5e6884547c67db93cd893203ea0aaa9e146b13addf29712e41df2d4db1ecd9be77af5456704c6e21fdb75af77a4014376912e0b1b17e8104cfa223cf0325544b8d558089447430674abaa528e9f49883836f9f7a876f51f174bf5cbf2ad5c9ff32ca3e7e46fc9f66844966147a953f5182aabfd6dd37573e51b10511b260100ae6ec586fd5e0ff5b627dd2bbebe143adf4b6bd91c633486fa8694db1a1a197bb10bc113d7f33b4c7317f3d9b1e54518d4c6edcb0cb33f91fac0ac283e9aa7d35e36b9e7199ac835a6417b932d90e098a3ee7d40bded465abd19203f178e98d439119126e306ad0c6c3294ff4998eeb11fab398a5115250d08d9419355cdf0288f7fb39ff51bc91fc00c898f8185b2434970ae2198b4e43f3da68f1d68cc48692a9d9cc563e42689339db5b1447201c8364d13e164d0a73678145f44446d51698b1540bbfa9915ab7eaeca03825b97219ed52f274739124c1d8f8fb74f04fde6a53c8b6ee6dceb68f2a2aba6dc3a177576b283c6b3b20285b9524c5718cc3f0642a8846a84d693e8a936462841e4fbc46be2e4c55a09162f7fe368fd5efd47de66edbf66a77b67f44518239a9fe49cea8397d47c3f5959e6c9e50b4945dd34e0fe77daea5f730e39228556be4a492eb1e668a82e32ce29a5fe4eb7701c568d1ab5bde0e9c2d47b7a8180ca7c272a919c380b93b6f808e7cee65ea6ef9aa6edbf27ad54b22b7df84bfd3cb4d2cd8ec9a4a3b29d3c35ab2ce7b6a79dc0e6768df80a4a1d381afb14458fb2598cfb96171fc8137e76a052b18c9e177fbe0f5ce865da0f91c77f81cfd69b28913c18d6b8d094339d91ede25e72f741ba9e42b5adfcbdc7328ec7ecbda7ba0780c1d8c43831340757a7017a6b0b074a3b408430b245a8cbec3d84065ac3781350a67c4e00cb549f3edf9d4c7b26a813d7f3e6d63b5ccd4e606e40f2a45040df7cfa13a6521109ab3f3e1de652cfeb3c8c1a045faca3b2665e491bcfcc55c02ce7127766d5ab48dd0312f49f7485dc3cf3542d7b5df430e5719b87192b3072e3ce72d3e25b07bed52484a6f747d7aefe2e00dc0cc228c82fa14ac61eb6573e8e288163dd50c059f57c81062f54a161cbf3c4488c5fa6b8052a58e8169206122d490f2ed7f16fba1e543c84115afa77c06b786cb70815b0f2ea50c9f3c24436804389aefe09ae10fc6b0ae41cf8c34f14cc4bb300f1294850e42ee19e90c712f2a3834961418c7872c883a2cfc1dc9cfee21312cb20c3f63c67f0a4d9db8ef78cc4a1637db72f9037db3665a4effc0ae21ef88ac56b9078825002507dd49b3e0faf6da0717411b0a1438e71e88fe9a5b4e07cfc5a139a84555c9eb47901013b12db4acbf62c87fe78a52bda66b2cfbf6697f859a586a25d34d0e2bda7e5f2df5ce820a34f8f1445b641f9621d3373bcd99b747b8f148d700cb7f9c17ba44d35400a6ac6916810a15ecd2877f1b49e576d18aab4037599ed15ab2a75a71a78d389b68c7586165ba970441b2df7d074c9ada9d7c397e2fb8349dbec27c033d3c64a6008745c09b6fe6990ca5f9cc8a12de3fcfbd464f99f943ac30278d58c1ecbd473150334e6c4ab243fe18a684fd0ef9b8c5cbcfeb54039f56f82f0b7c761a6742491d350f40fadaa8d783ef5a1930a57302cbdd435a8a07a79ad9f96b4e9966ffab4e57766fd1031381bf82b5f636f68fb8fb0dbd5b194491cfcb66678cecb51595df7d39e3dac72354e4ecc0a064c9a3b334437f6ee7d959c54f6706c77f15946afd73b1038bb0990bbc607423d6214573bd2f4ca9de6e8e68dc0dd0cec181d4f5fe20ed766e00de3bd371d65cbf0551e81758bf86b8c6b50f09261133ab85f8c86424e4d78296e7463f19bb7b8a4f79a70230946b0a5b7b24fef7ecd2d6c7e6f2f4736ae33b73856c40ea1f4497af21d73bdb79d6117badc13d54a46c95cfa2ddf1a54a1f3919e255683be1aa7f2ccb80b5798cefddf6dfe16b06cf7f0618810303773872dcdb9e3ddfb417527e3db69f6692ce323b6b0e1ca09dbf566196f9079ab1f76d2b76f952c6c969446df2f514d0cfceae7ca54413fccf02792203682537bf225933d6bc1259d285908df76cc80ac7777d1c7cf8dd2f836bd8e622daefc73e0b560ff32eace2be5a3c42e1adf8cc0c1824499d1c1bd3f7e1079d09725abf5b535b82e531d7f43a2e05e5066ff3b168ffd023428a0ba5782c03b8939ddfef82cbbe801080608acd4e79fc6cf5a9f9fe0bcdbeab02916056aa759fa0d3b8e7d44a1da5f3f00bfd88c19547f930c3e8045fc33c0fe5af7938436719a17fb9ff80bea9064abede0062118f791e1825a4432377c773ada4582767100350490ca716422d549f8a7baa7c9192f0c1c506b63ef749d3829941c7eb6400196af426ec273200ab82819bfb0e4ca2df81cc92a9e5795ec6f4b06a63d96b1562b7f63a57e6b2fa7f6748714c07922163b5ef362c0bb1e00ba3eb958b86552f043b96ed1dcc12ca5e5debd96f8acdd70e28ef0b91f3529c9c261711ed969faf38c79670aa0ad9dfa8e66fc0fddb0d31c134df156a5f4d73549bb65cbcb2e16a78f5e5c99bf83dbe4018ba127df069e66aa7ff3f37b1e8ffbf4962753753c7b8a55491ddb2b9f50aca0d1f469c55363d2bbbb19f397661c89811629c8a96c7682714e4409f8c035e03023a8b50ac86ec24e99046bd87238744beb81ab5defcbad20d29dee9714cf15093c66185e70fef7ac93714313fdccaa07603e7e42a182ee4f8e57a0cfd2108c6bfda5e14d004a969f96adf8ef77fb78a23c922f7c763a448b90ba8a504a1a8e718434e21f5fb2698245105313b81cc9b8536101ceb0ee8e3f3ade96f351768142664a90ec008a584d475d6867a54e436ef88c1c2f0a0d52a82390c80a941cdac07e023ef2c7812d941420548c04d05ab077d0d638b214f530e9c622b91ec0f7685464d09fb8cc34a2ebc3d18b74b2ca75117b63e1774066c07d5392f4381fd027281e6b4a83eb846fee1ab1d06108bef66073359c7f64a4a0f13241040bebcc65fcc4415f69ee773d8f938197cb46d91611e9077128bf2b3d16db389608ee3bb8489cc5235d85ce4727dd785b51b514c9ff4ce7259c3c207ee948d0603659b861219f3847e5dc083ff87ede8e0f7d75f68bc6f76445a4283e69a782a06feaa89844169da15a97ef98b6d965e576f7acfddc498277775a1e01e905bbe15095e7543ba8e7f5bca9f423785dc9217864d95d8e0603656b06a7b477a44e4716684604185687252713bb222f0b84cac4de79dace9de5e600f075eba76f91b31e5e3f305e420612d5c27f326f485788293ea6569ce900db88a2d5d6f700e734a9ec5a784a0c6c72bc46576f7359a0a47f838b8ad22a756ade8eb807f692f7a097a934316db9a203eb6b2527ef4c0c49c8d0bfbec3bb5b029bcf01074999da66828a3425f5b23a125bc6af83f7fc87719fa4d0ac34f12b275f805d3f5e4afc6a2326e6e4f1c4e01d7e5f2a45568737f8b4f32fc6e03ee5d053690b1a9fd8dce936d91b1a2c0dcc14f59c29dd34e9e92d0e255a5266ecf8edd58bc702fda514af7ca01ce22d9cbdb266e1794702daf5673f345a62d91a033adaf733d2ec0f28afce43d4aec485632a687a839cfad9469ea89a1f95eaad39b7d442df5a1235547017420a6a4f5206a900c68b4880fd699a11f24cae60ff7148351c6e56e3e96d7d7b1e90904fdcc61943c9dd807a082e9f256ba62f0ab1fa587c5d5e6991733ae61969251692dba168ff3c110150675a336b385afb41e6c09372d50a2fa8d1ea4a49c6a7f5a9ebaca48e513b63a9e2ab96f770087df891debb788544a661bc18fec45eac794da3fb54aa535c8b237ff00151f05b06b8533e10423a30ed2057842fc34e2299d6a925fb67c98fd9c8e75fd650815dc732ca4fc29d0221e7447650adf41ccc2d50824dab90f320a5b8e7eb39bd4e7c4f8e82a213854f2369366e75f47a000a904d78df1c77ca663c7e0f768f8d87b9a68d4319e62076aff65f24abee279c13055f4134a578dd7fb110cba9ffcd6bdcfd8e001c8f316de67c105abb8130285d965cf611e64510699ef49a5171d0441b6dc3ee683a021524dcf1c868903e3bcca6b6fa50bbd2c9d8b90916fc6bc3feb2ce5fef1c402b00458f164732798970f3456fa062f381ce052d44b35428d976c2bab931b46b16b38ef03f6cc1a04c21d90610197bfbb6bb10c38ad2ae5a2079fdb417d5b351aded2b58517efbc1d42af8e785fff1a71591463c5ee9ed0b6e81ea74f520eb1356fc3d8fb5134ccb9d5dc858a89d92835549d988244bcf24cc15d5fa3a1f6e98003cab0b8c1f51b324a35627077e8f288afa9b8b5fba34d495544d9b8c720a046fee9a5d839dd8b41a02a63f2d2f38554afc0a369e8b5a0334b66489ca2b46460380c324f6a9ad96d506c8d07effa8c96ecc212ab85312968ea686175da67c0638d3b9fd34f73cea78c50935a937624e0232ec85674cda7b962f047c902afcb3a4a8bbe34c7ed2865fc1204f48b2f8c8de88689501c9fdeedf01e46c74264751383876046122e195d62fc68507ccd4d1e721a212b3454978a7dfeceef71b3fe2bdb1a6ada6597b5b18ae3fb68a6da94b4d4c1f76e0677caedf7fffe043a3c181b2bd9c46e39ee96ba6abce6e2c028d6a8555b78a41e648012d90cb82d142d537cee910eaae7bcfeb0e23ccefb8506c17e1f5615910e18c039e841b85b377b03eb3d32186b824dd454ac1afa17aa537af4ef60d701e97bea68593487c3f06007977893d9d7e8c2d532e7a4c58f95eb0c0c9d4d615593732bd05a2dfe94c39e7a84bc3494058fbe469f21936de93050a6c626ff80f4ef54467d7ab0e033aa19fe8dc0a1c6ce8b9e7f73175b617ec5ae6ef024177269af1edd9a0c7b2fccef07510e7080b43ab45bd60e02239a4c581ddc5ebc59d6050787a0a3b9957ace735514e243dac989f3f5390f7ec9951748fb02e1f4b9b38ff4f6bc0a2c88d0104a55c95eae0dcc1bcfa7edd5157c2821762dfd6792d8376593c087ca4e4c4b511cc41d98b4bd0380dbd3ca2150983556300b6d0816e24d93145fddcf5e265745d7d91dbf597c93df684b90d2db37a6c9ffe45a1cc5f6615fd3188d10d06574b674a00eec5cbdca7b1e520abe1f6ea5397c1e8035cd866b66653b8eb2be3aa4956e38494a247b2b24943622af33c98a507c615bfc61b641fa133a74a9970700342e379267f927b4416ca7b63bda83fcd08578b4b25ca18d9452729d0ea8a4b9c0d095eef721625d2135195de30c714cfac927131001131ec116ac1d35a68b5048f425af1581539f2765794da06857566a5df549008ea49a2857f2262b7d16d84e0a450c9246b77a9e4faa8f560ce01628480d89754c2ec66f8e96bcf514b1966e30ddd73cff13992ef5e32a1b7c1a9baef921b422ebd431b0d3b09f8ef1c24a53eab0f7145484318fa76ba0be8911bfc149dd08f353fb022a2bdabc7d16d3b073336e3563b642fb9cf14d911d5a4a7f788a2a7572a664b90fd47162c2d308d3e1bd79ff0e32f5d213881144541c6f33aff7fc1985927111ebd97c0cf3f5d72050dcec64251017f4bdc6177edb7ac2ec4d3c72c6977a968b38c6347a2d650cdb62e9d53a667dd75e890815bbe8788af956c098ee2bcd9f31db1c10d04c580f974dc4ad9e53e954d0ead8332bf295066c89875edd7016b9d6316470ba446fc826016a245ee06c333bb8db0986c1fe42f44dae3dd556f7a3e1ac395eab4b9314d41c1600fae93713b23988edc32dc43f22fcdb43c33ea8d625fef9e0bb6568d761a8f0136103d28a8f550e9523353b0aafa1a6f8654b460d053c077ad8129754a60bf770afa644b7581b5892d4cb1d1ebb938b48f9d4583f31db7ea9cb7bc590387a3e1367e88bdb283931b3559e5ee5b5e99a16b41594d4b0c6247fa5210e0ebc419873a537cce6fe09a57b2d76fa6fbdaa4a208ff84f97eda4b7fec073fe96a50356e0fd559bcf44ebf24be3880c0b8e0092009fa3cb0df306c7f33921b9a48c4c1b30a4f8a9b414ef49bb0c140e34c4f57620b2262066ea0e81ca8ef114436ebea20c2070df4159d62ae5328c5feb3d0ddfedddbe94b728e44e47c792799e72b0b20f5ec1c0551527fb59be7831a125c20252fb40f8b9b87e48e2ec73f6bcf3608b419d832e51a0be68b810c58699dafd8f86e89caee53ddb13d163062642d34fefd84da26f02e3840440a2e1b852740ed4ca7684bd8c53f608295c8df80a238a471537bf1c9b1fa6ff0b73ad4c4c9e54b882bb294e6e2a787cb6303d67d162366078c7c084ca48ed28a6d8ea8cc3d61178814d3e99d6dffb70280f0143d987e553eb12b6850d9a5f6582bb01ceee8c46bdc92b1c70f98cc07dbaeaeefa3ad68b0aee97e97b80ae20aa6f66d5d6cc39b55471abc69570682cb1744eecba3a034a9b66e4fba8cfacaf12793ed4a3a59ae765e2e21c37208d146cbe98193461e7b2e1e75dd972e3ed1d231e07a963a3002286f1ddfc371ab1417b37b92cfa009454af9a9c9d1a58aab51a8fdc93fbed16777b1203093bab460bf06360e802c0cfcfca154cdf3e719d032c84375762e31068ddd96ba03c025f93677b591e1ae76fb5a7448b92b788143178cdec342a9e3af43b6e462a828e9457c6e0a6872b9133089eeee0043d775b836820ac53d842abd679c0b515b84671b1ef50f22a7460953618bf5c0af9b53de9c5cfaffa56b723e38a151223513480054ca186418350fb8d0b8ef8b314776003f9498cb8de3ae085f9e14609514b8f452146df46bfd4cd8b1435002f0b2e5e268e7ba24be0b7c84599df9dccc10e477710757a47bae657bce7076ba64b7e69a06bbd68f2a4b4bf14e7ed20efdf18a662e623369d74490e9a13617798670152b43a3040d481cc8a4a88d1d56a9004c77a0d0cdb54fc40cae091b4ce71142e0d3366d2eea9a7fb7d92cc5797e76eb4b1589ca3e3093c73b6c433b9fb7cf3e59c4296e20528ba89e1892c6fdb379ff2a9df0b0b4d98b0f404c6c2876e42b94a2b1f9809f93f394c0d299e096d3887528ea5c33824abe44b947b4de78d1a6b8a206e078d9b9bf8dcefd94f82eb2b17106e4e3557b110d22f4a57c1858742d23b1b09018dc2dd930d5ef8ea04307682cc5c58a8b0624503ef18a815716f7a7604953b7f602ee6ec7af77c032200efe65f8ce7a9684604ae4041ccdce0c1f3c66e70d214521a853273f37962aa30297eec121927bd27f8bc13dd4a022d4f17f45c7be943d2be31a6f38862cacc10831c0c3d9c342807c808feb5d06b5dd78d9ebd9af0f69b02b0e202fd5b6045d0d5dabd1f6ab12dad7637f86d5e49a778d94095b5d8932b9bc59895d490585e421fbff83d4992d2d76c13d9a9e783f1a7c87230945c9468821fc0be220d59e983b1aef1cc7c02708dabe24b6b65c032df981191c02f07f3fc03ecbadd7a43d21edadc576993152161a468115b6b7c737a2eff15f20e7fa43b5f93057d4b7966188e118a43ad7eaddd63f2eff5ecb2f97eea8f96918245567c24b1bdbcdf9bd3af926a165681d9ed8b1f658394a268609d4b056da58cfae811c9d2d6f59d2a5e0a33f71c2952d74958f583edb5c142f08656d3b2a9c57c01b3ab58aee26ace74c0a114ac423b8ea3550c0ead5e2c9e2fa7396c3ea69e53842a7ad845671f32aa5e8c89fda9452d11a2f427ca35c73cdf68442b75496c93fd35432692e624eccab4ed6756fee407b64d8e00675a2aff3ac3c09e153c12a45cf1f4ff996c8685914c26578fa7db64bdbfda441933f1747fe3d17bde47b85a97ea2ed53808b380c2a5d14b86ca6a247f624334ff83ceba33451eb7d6c84c3a6944d2c59ac8a0f592f88fa1af111bfbf37ce02914ccee09e36059d504031cb7c4129c0505ef25b8422c105a163fa5ccba297eda4cbe7d7e42837b210d2e3e77f2bc52e7266e7af1b5906862f9d4e1f7e0357cab17bfac71d41e94596a5ec036f633aed24d11069e10702cc209b9823734204423773d261709caee690ef53d243f096f36deeb1152df7150687051a2d168d830c9e7f1b2a27f93461e9b43730f022e3a9b6bfbc4997ddaa3e7249cebd3ec56c86cd1621912f72aaf9eedd52f6e8f7f362342dd095feff913ff11796b72887b24b29bcd32efd75ca486458bb65aa9eb81dff671571009f48e837f986955f2a8d9a534c2b15362c3607dd65f5c74fc921f86694c6d4a86b2c50be79593986f7243726e20f127a369d0531568800716a49761efce19a0c065c117f625f81661c72a866a43f7069f40422f468dc0163a0645dc783c50693e5ba637d9c9a5f256ed414ad8198fc37a67dc0fe16dd3cd9aa90d40b09669cca1890b2cfa2dee0a0f06f917d7ff567cf457f8e6e9fa12303b653e8d8ecb5158e678be76fdfbd98a3f681c71cbf3e5e0e533ff9d1c64bd31971a528dbc8a6033111c17ffac2a5790ee0d8947a513d956572c35072c11c05368a0ecd4bf2d7608bbd4586275483831fc084e84fea5cd962c1e83d180f58df6ef116bc3f19ab2b4d7bf8d08ad6ff457bdd0d040dc3d0036d7500081fdd99f4fea0928767374b867c7e627f3ee1b472e1a0e964c8f04cb64b486ab69e7733bbe31816f20cf4313c132e6be999309674ec115efb133d49a307a5d19143bd1a347a800403a6aafe029578d18712f6cc96abc90c9e67d5d28384be2adb2838453a0e784968580ccb897e42f3d06f9db359643912a292e80086978847d9a0bc40812a3f2e8ca0795a5a82d65a6405a64845904971f665dc280594e4ed73071ebdf4055edc7696c1d37ce91db37a9f2f0f9e76e9a50c73aeb8bcd4192c8817b5b730e5727af9916bd15cc5fdc7e8877b1c5ac28ca5458ad4bbb3fde187834589f80cac9d114a89591abddde7658596d3c35ca6498bcd62adbce04de33a02710f51c632055dff270ff0549a48e8a2003717c6aa2483c86f1b25cbf5845d58f9d0d206d43023924be950238409ae8161cc81e5da13d6175c60a1a0b49c1fc7ca409ec144a86d0e8b9728eb535bf4cc13f7ce17da3b5d1f3907b998a35bd07c269aedcc205689c3fff204e0cc0a1860828c25b49f2480c0fe772683a8ee9b18d1bebdde44b95eebe482e5a28ce4f153052ff388a967446ae27118f9fc2289f60a037eb6d0b87deae467b1bf189f3defca9885c20880d34ef4a0600f93ec306b1369341764891ee3d9e6eabd70060bbaa88fa5027adca3a06f9dc3bdcac1d16affbefbc8db7327067ba5bb649ea182310a08f65c6a6ab9cddbacc7251b97099582efc7cc5ef33f971fee69216430be3e270084f11d72bb77e4b5702201664ad8dfd65cf0ed53a08f760a7958d15573f3e29c800fcd949dac703274cf26a6b4ccfcf49ec262613174fb1a923b654b38d3ae46b79611dfc76f403a4cebb5e0a3d36e305d95a7f750da1e314a74f947ee26ce02269a7b5dd3f8dd6a9cd8d7c6915e52ea24bafc01b488abe0180c1980712cd5a2657e48661855b43ebca0d2df3417953ad2560d25a20c4ad344af4effd67e6ebf295ca435c15360b067019d0a392839c05ec765c29a882e06df086e059e6faf03c86c1369b626728c8eec3ea5b5685f68829968055246f99741c3cc2dccdcce7b153aee5b3795c0edfe09bba2b2c468dc97ccbdfda5eed83555c97ade7fb88b1d8f96a04b999e7b468dd3a091e51808a6166d8ef0df45d906a3627a2a2560db1379a212f159c55a033c9c721ec9f75e582eb08d4724071f6c2f65f23676f58096d899d193bc26db064eae62a2a4dc2b0d6dc1921f3c146254a4a7368352fa9f8ffecf3c4764b24620ea8de59eed5651fd3f7ed7e087b682f777ec31e5be6e55b173c2e9c32e6c30f3b042660654542a5ce640478b47a44495347169283c8b12cefde4ad03a3f86b595ec8b6d6fe00d7aef49006866d7291e6b28c7540b9ffd98989be2e36487350ecb9335a631866524242a16c37ac9e93c4f415aa7007db203eb310502840267e6e94e558a97ff66c00f01fd606e1d6ae369983e13145874fca8e09797a654d3587dcccabdc69a3df3c8da46d824a1b0812f44ecf2bcb3a3b4c1cd00cdbeea5f9a1c4f7e28690cc0ccc89d13af7404b38cfa15bf31988f206348284bffa3f7a3ed1f1dde0bf14730b733905ee98d6e5e848063fe94ab56b244beecdc916cd0ff3b6ca3f5c9b09cca9dfd0b43aadf3d695ad73006c115e49b9bb8b33e36b6a293462bd021f741813ffe1cd0cc801df515a9b1a8a46c7d67d28547c4015fc218c984d24dcb2e6a9a595793a2f3de07229738577c94cd0aff8e745c093089c0ba48870ded102c85369d4099aba458a86e44a91f5bfb14ff64583472ba95748a38fbfb65dbf438080d07ffef9f87d2d8e5d474cf270cac60b1b2fcf15ee067f7c8b19dc90fd57cf803a68281e9ce0daa646e42352fa0d95ce91fbff22f9115e68dc757325a26b107df32124436adb0043c52b29577e67b118df075ffacc4cff0994b1cadd84cdca3c6c1cfba66a7a49bab0147712091a36f9a4719b59730071f426dd200bd1a362d159e9800b96904908ca3eb2097a7ea5e3c20b7d81ba9642957b2fb1e3b11a15cd00b61b18385b6ac229e83ae531e5c1a40fbb8e5e294b04d82d6af13dd071cde6dc3e4999a733a64e895d142ad71b354c66da34488bb13d582cf32fec50883458818c3c8b60503bd016d24e5cbccdc02b1d0a4a9517617ce8d45dcde176657c76202e7e98b149e95b39feff175e09b521c717197a456d7809f282180383c2f8f90ddc72bf96b845adbd2d7497ed5eac8726da64c4b5b10bbe660b435d87007d9c18b8b99a1d15ce17e4ee043eea0c25259662904a212be59f973651736946e591c99d19e76bc78963eef161aaebdb13ca4d026a0d2842083abf5f75fafa9fd966620b79d7f360c65fefcc5e5b5274e860c2b40d6bcf2726775520c9e2af57f40adaf431d05617604527b7abf8211c68fe37e7f581141080f172e7aa608b0610a251ee7baf9a8ffd3d3a9758f03c9a19ff8abe8ab21d00d2ad72845fb81981da6322a0bbc704ea4175033798a136471ad3aba3b5f43a362bf23a1677910a673de597fcc737027e7bb865392e329a79b7e9e3a76b6cfb2818d54c01aad6419d6b7b62438175575d95e9f0d7ede8b7ae2609a27e56e097fe931bdb6d0921e8a7e8ee93d985fb3339d9204d9216d70c4dd1a2e4854723a3dd2adbbf881f1add0e54a02d852e27e046e58799fd2560e13faf84374e9b19a323a4d2bc6c5e750f929d83d946bcbe8c8f0e616e856ba619a45f78706167a39dd96c547a2acfadc912681845b2a6913453b70f82f5433e36a1a18d86901b274b6bde39b4777f15234a5e173a33758a19559da0f0a5e7d2c73bbc151f99649a513668399f1959c062f82d9e78530850ce5ce5d5b99cc594c75bbab56f91425f378721d369c2fd8c2bf66fdc0f179bdcb412dbc41a672859061545f1166a9e1c8690f4672d92e438d2095a2e6436974688d46c8fb61d2b6de575437c85cf9307efa6b9cc7e64bcd399e43a900ce130b9e63987d8584ece04ce78f7f4bdaf822a0d91b1cbc3ca5bedfd514d4f0cbecc3685eefb39884a72e1c1832b4f033ecceebcce2bdb4d60b6c0bf2f1c930b00cc6eba1cd06ca0c86464f7dd8c3d786afcd5fb50cdc4c71721dc5a59a0ee4692d49095b411ea19cd84989ddde80d4f9ee4b2da02907af65622172aeae0232a3a21bde9b36dbdc55899954850d57038933262d325c69f2cfb4c3b57c305597b2d473f67225e10c38a3ddca829bd2a805df9c137f2fdc3d06ee403872e03a7359ec3fbfb8faf405512bedbf78964adcbb4308671265e8b47e59c0716ca9203fdb06c5024346d1bc99c7569f5fac5f5a308fe873b12a128d9a83b6ada20ad537f3199f841b1df8cb91a63986eedea6819d95756ff77fa2f96f9a10c8115a152b701daf8e6b030c5c00e1ec9f8afcbd70423d5438bf847702b37dc6c95f786f591f68003b76c777ea145481a4b6f6d9bb8833309c7457fc7646caeb7be447f531671c805db154ef615c48557a9f773537e329305ce314186541906a959d59496bbf0167ae0b07e13f9afc7abc3fb96f03271d27c34eb98181f82644c3cc0479e397c377916cc3866183e7dfc91e2cfcc272804e872d3cf0a0743fdb9fc6b41a0526a82104a0ccc4</script> - <div class="hbe hbe-content"> - <div class="hbe hbe-input hbe-input-default"> - <input class="hbe hbe-input-field hbe-input-field-default" type="password" id="hbePass"> - <label class="hbe hbe-input-label hbe-input-label-default" for="hbePass"> - <span class="hbe hbe-input-label-content hbe-input-label-content-default">What is the system call number of Fork in this lab?(digital)</span> - </label> - </div> - </div> -</div> -<script data-pjax src="/lib/hbe.js"></script><link href="/css/hbe.style.css" rel="stylesheet" type="text/css"> - - - - - - - 2024-05-23T16:00:00.000Z - https://fuuzen.github.io/ie/de/memory-storage/ 存储器 diff --git a/categories/ACGN/index.html b/categories/ACGN/index.html index 150ebd9b..68e2edf3 100644 --- a/categories/ACGN/index.html +++ b/categories/ACGN/index.html @@ -1 +1 @@ -Category: ACGN | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: ACGN | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/cs/ai/index.html b/categories/cs/ai/index.html index 3125636f..d99c78cf 100644 --- a/categories/cs/ai/index.html +++ b/categories/cs/ai/index.html @@ -1 +1 @@ -Category: 人工智能 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 人工智能 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/cs/ctf/index.html b/categories/cs/ctf/index.html index 33cc6329..41a13022 100644 --- a/categories/cs/ctf/index.html +++ b/categories/cs/ctf/index.html @@ -1 +1 @@ -Category: CTF | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: CTF | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/cs/data-structure-and-algorithm/index.html b/categories/cs/data-structure-and-algorithm/index.html index 04778856..8dc8f364 100644 --- a/categories/cs/data-structure-and-algorithm/index.html +++ b/categories/cs/data-structure-and-algorithm/index.html @@ -1 +1 @@ -Category: 数据结构与算法 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 数据结构与算法 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/cs/data-structure-and-algorithm/page/2/index.html b/categories/cs/data-structure-and-algorithm/page/2/index.html index ec59ca70..e22a1913 100644 --- a/categories/cs/data-structure-and-algorithm/page/2/index.html +++ b/categories/cs/data-structure-and-algorithm/page/2/index.html @@ -1 +1 @@ -Category: 数据结构与算法 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 数据结构与算法 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/cs/devops/index.html b/categories/cs/devops/index.html index ef3ba6db..4aa53627 100644 --- a/categories/cs/devops/index.html +++ b/categories/cs/devops/index.html @@ -1 +1 @@ -Category: DevOps | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: DevOps | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/cs/front-end/index.html b/categories/cs/front-end/index.html index 4b87fbcc..d5f5006e 100644 --- a/categories/cs/front-end/index.html +++ b/categories/cs/front-end/index.html @@ -1 +1 @@ -Category: 前端 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 前端 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/cs/index.html b/categories/cs/index.html index b664ca26..d78490fb 100644 --- a/categories/cs/index.html +++ b/categories/cs/index.html @@ -1 +1 @@ -Category: 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/cs/languages/index.html b/categories/cs/languages/index.html index 22a56c56..f8bdefb6 100644 --- a/categories/cs/languages/index.html +++ b/categories/cs/languages/index.html @@ -1 +1 @@ -Category: 编程语言 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 编程语言 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/cs/os/index.html b/categories/cs/os/index.html index b65a18fb..c851de80 100644 --- a/categories/cs/os/index.html +++ b/categories/cs/os/index.html @@ -1 +1 @@ -Category: 操作系统 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 操作系统 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/cs/os/page/2/index.html b/categories/cs/os/page/2/index.html index d7f025d4..13daac98 100644 --- a/categories/cs/os/page/2/index.html +++ b/categories/cs/os/page/2/index.html @@ -1 +1 @@ -Category: 操作系统 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 操作系统 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/cs/os/ysosv2/index.html b/categories/cs/os/ysosv2/index.html index a541c0f8..35490804 100644 --- a/categories/cs/os/ysosv2/index.html +++ b/categories/cs/os/ysosv2/index.html @@ -1 +1 @@ -Category: YSOSv2 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: YSOSv2 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/cs/page/2/index.html b/categories/cs/page/2/index.html index 64e3a7ae..01ba7c11 100644 --- a/categories/cs/page/2/index.html +++ b/categories/cs/page/2/index.html @@ -1 +1 @@ -Category: 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/cs/page/3/index.html b/categories/cs/page/3/index.html index cb10edc9..566ef4b6 100644 --- a/categories/cs/page/3/index.html +++ b/categories/cs/page/3/index.html @@ -1 +1 @@ -Category: 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/cs/page/4/index.html b/categories/cs/page/4/index.html index 7ad19943..8f6131c8 100644 --- a/categories/cs/page/4/index.html +++ b/categories/cs/page/4/index.html @@ -1 +1 @@ -Category: 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/cs/page/5/index.html b/categories/cs/page/5/index.html index 1702a767..b9d1914c 100644 --- a/categories/cs/page/5/index.html +++ b/categories/cs/page/5/index.html @@ -1 +1 @@ -Category: 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/cs/page/6/index.html b/categories/cs/page/6/index.html index fa83834d..6f0cdc98 100644 --- a/categories/cs/page/6/index.html +++ b/categories/cs/page/6/index.html @@ -1 +1 @@ -Category: 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/cs/page/7/index.html b/categories/cs/page/7/index.html index 5d2c168c..6e44806f 100644 --- a/categories/cs/page/7/index.html +++ b/categories/cs/page/7/index.html @@ -1 +1 @@ -Category: 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/cs/page/8/index.html b/categories/cs/page/8/index.html index 9393c526..b92dd4d8 100644 --- a/categories/cs/page/8/index.html +++ b/categories/cs/page/8/index.html @@ -1 +1 @@ -Category: 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/cs/principles-of-computer-composition/index.html b/categories/cs/principles-of-computer-composition/index.html index 96243b12..af819bb9 100644 --- a/categories/cs/principles-of-computer-composition/index.html +++ b/categories/cs/principles-of-computer-composition/index.html @@ -1 +1 @@ -Category: 计算机组成原理 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 计算机组成原理 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/cs/principles-of-computer-composition/page/2/index.html b/categories/cs/principles-of-computer-composition/page/2/index.html index 4f90eed0..278172c4 100644 --- a/categories/cs/principles-of-computer-composition/page/2/index.html +++ b/categories/cs/principles-of-computer-composition/page/2/index.html @@ -1 +1 @@ -Category: 计算机组成原理 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 计算机组成原理 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/cs/principles-of-computer-composition/yatcpu/index.html b/categories/cs/principles-of-computer-composition/yatcpu/index.html index e8039d44..56aba32a 100644 --- a/categories/cs/principles-of-computer-composition/yatcpu/index.html +++ b/categories/cs/principles-of-computer-composition/yatcpu/index.html @@ -1 +1 @@ -Category: YatCPU | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: YatCPU | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/cs/raspberrypi/index.html b/categories/cs/raspberrypi/index.html index 92ae7e5d..0686c38c 100644 --- a/categories/cs/raspberrypi/index.html +++ b/categories/cs/raspberrypi/index.html @@ -1 +1 @@ -Category: 树莓派 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 树莓派 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/cs/tools/index.html b/categories/cs/tools/index.html index e2278a84..4bb3aa06 100644 --- a/categories/cs/tools/index.html +++ b/categories/cs/tools/index.html @@ -1 +1 @@ -Category: 工具 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 工具 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git "a/categories/cs/\345\271\266\350\241\214\347\250\213\345\272\217\350\256\276\350\256\241/index.html" "b/categories/cs/\345\271\266\350\241\214\347\250\213\345\272\217\350\256\276\350\256\241/index.html" index 54277322..a296744b 100644 --- "a/categories/cs/\345\271\266\350\241\214\347\250\213\345\272\217\350\256\276\350\256\241/index.html" +++ "b/categories/cs/\345\271\266\350\241\214\347\250\213\345\272\217\350\256\276\350\256\241/index.html" @@ -1 +1 @@ -Category: 并行程序设计 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 并行程序设计 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/ie/de/index.html b/categories/ie/de/index.html index 36077660..7b8b4a8d 100644 --- a/categories/ie/de/index.html +++ b/categories/ie/de/index.html @@ -1 +1 @@ -Category: 数字逻辑电路 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 数字逻辑电路 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/ie/index.html b/categories/ie/index.html index 0efa42eb..0290058b 100644 --- a/categories/ie/index.html +++ b/categories/ie/index.html @@ -1 +1 @@ -Category: 电子信息 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 电子信息 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/ie/page/2/index.html b/categories/ie/page/2/index.html index f7c8c716..7f8e752c 100644 --- a/categories/ie/page/2/index.html +++ b/categories/ie/page/2/index.html @@ -1 +1 @@ -Category: 电子信息 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 电子信息 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/ie/signals-and-systems/index.html b/categories/ie/signals-and-systems/index.html index cae73518..e1d4fc6e 100644 --- a/categories/ie/signals-and-systems/index.html +++ b/categories/ie/signals-and-systems/index.html @@ -1 +1 @@ -Category: 信号与系统 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 信号与系统 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/index.html b/categories/index.html index 6887dcf8..74f2ba6a 100644 --- a/categories/index.html +++ b/categories/index.html @@ -1 +1 @@ -AllCategory | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +AllCategory | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/math/cryptography/index.html b/categories/math/cryptography/index.html index 156b2461..d9ca03fa 100644 --- a/categories/math/cryptography/index.html +++ b/categories/math/cryptography/index.html @@ -1 +1 @@ -Category: 密码学 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 密码学 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/math/discrete-math/graph/index.html b/categories/math/discrete-math/graph/index.html index 71839767..85e72efb 100644 --- a/categories/math/discrete-math/graph/index.html +++ b/categories/math/discrete-math/graph/index.html @@ -1 +1 @@ -Category: 图论 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 图论 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/math/discrete-math/index.html b/categories/math/discrete-math/index.html index a3b98d66..71b53da0 100644 --- a/categories/math/discrete-math/index.html +++ b/categories/math/discrete-math/index.html @@ -1 +1 @@ -Category: 离散数学 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 离散数学 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/math/discrete-math/page/2/index.html b/categories/math/discrete-math/page/2/index.html index 11ba309a..648afa46 100644 --- a/categories/math/discrete-math/page/2/index.html +++ b/categories/math/discrete-math/page/2/index.html @@ -1 +1 @@ -Category: 离散数学 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 离散数学 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/math/discrete-math/page/3/index.html b/categories/math/discrete-math/page/3/index.html index 55aeb42a..78463264 100644 --- a/categories/math/discrete-math/page/3/index.html +++ b/categories/math/discrete-math/page/3/index.html @@ -1 +1 @@ -Category: 离散数学 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 离散数学 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git "a/categories/math/discrete-math/\344\273\243\346\225\260\347\273\223\346\236\204/index.html" "b/categories/math/discrete-math/\344\273\243\346\225\260\347\273\223\346\236\204/index.html" index de031e5e..fb64508a 100644 --- "a/categories/math/discrete-math/\344\273\243\346\225\260\347\273\223\346\236\204/index.html" +++ "b/categories/math/discrete-math/\344\273\243\346\225\260\347\273\223\346\236\204/index.html" @@ -1 +1 @@ -Category: 代数结构 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 代数结构 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git "a/categories/math/discrete-math/\347\246\273\346\225\243\346\225\260\345\255\246\345\237\272\347\241\200/index.html" "b/categories/math/discrete-math/\347\246\273\346\225\243\346\225\260\345\255\246\345\237\272\347\241\200/index.html" index 979b6132..d05e5c48 100644 --- "a/categories/math/discrete-math/\347\246\273\346\225\243\346\225\260\345\255\246\345\237\272\347\241\200/index.html" +++ "b/categories/math/discrete-math/\347\246\273\346\225\243\346\225\260\345\255\246\345\237\272\347\241\200/index.html" @@ -1 +1 @@ -Category: 离散数学基础 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 离散数学基础 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/math/index.html b/categories/math/index.html index 7d2ff6cb..9f530d83 100644 --- a/categories/math/index.html +++ b/categories/math/index.html @@ -1 +1 @@ -Category: 数学 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 数学 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/math/page/2/index.html b/categories/math/page/2/index.html index 6bc35d37..697311d9 100644 --- a/categories/math/page/2/index.html +++ b/categories/math/page/2/index.html @@ -1 +1 @@ -Category: 数学 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 数学 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/math/page/3/index.html b/categories/math/page/3/index.html index 2e5f9f28..3ccce6ac 100644 --- a/categories/math/page/3/index.html +++ b/categories/math/page/3/index.html @@ -1 +1 @@ -Category: 数学 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 数学 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/si-zheng/index.html b/categories/si-zheng/index.html index e28887e6..c14bddea 100644 --- a/categories/si-zheng/index.html +++ b/categories/si-zheng/index.html @@ -1 +1 @@ -Category: 思政 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 思政 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/categories/si-zheng/ma-yuan/index.html b/categories/si-zheng/ma-yuan/index.html index 9d2e8626..2d1d3197 100644 --- a/categories/si-zheng/ma-yuan/index.html +++ b/categories/si-zheng/ma-yuan/index.html @@ -1 +1 @@ -Category: 马克思主义基本原理 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file +Category: 马克思主义基本原理 | Balloon Party = 風船のパーティー = fuusen no party
\ No newline at end of file diff --git a/cs/ai/agent/index.html b/cs/ai/agent/index.html index 3fbb768a..0c3ee150 100644 --- a/cs/ai/agent/index.html +++ b/cs/ai/agent/index.html @@ -1 +1 @@ -Agent - 人工智能 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

# 智能体

# 定义

Agent 是指驻留在某一环境下能够自主(Autonomous)、灵活(Flexible)地执行动作以满足设计目标的行为实体。

上述定义具有如下二个特点。

  • (1) 定义方式 : Agent 概念定义是基于 Agent 的外部可观察行为特征,而不是其内部的结构
  • (2) 抽象层次 : Agent 概念更加贴近于人们对现实世界(而不是计算机世界)中行为实体的理解

# 特点

# 驻留性

# 自主性

智能体能在没有人类或其他智能体的干涉和指导的情况下运行,并能根据其内部状态和感知到的环境信息来决定该实施什么样的动作,进而控制自身的行为

# 灵活性

智能体行为的灵活性主要体现为以下三方面:

# 反应性

(Reactive)

反应性是指智能体能够感知所处的环境,并能对环境中发生的相关事件(比如智能体间的交互、用户的指令等)作出适时反应,以满足系统设计目标

# 自发性

(Pro-active)

自发性是指智能体不仅具有目标并根据目标行事,而且能够主动地产生目标,进而实施自发的行为

# 社会性

(Social)

社会性是指智能体所驻留的环境可能存在其他智能体,它拥有这些智能体的有关信息和知识(比如这些智能体的物理位置、所拥有的资源和能力等),并能与它们进行交互和通信,实现灵活多样和复杂的合作、协商和竞争,以满足系统的设计目标

# 其他特性

Agent 与分布式人工智能系统一样具有协作性、适应性等特性

  • (1) 行为自主性
  • (2) 作用交互性(也称反应性)
  • (3) 环境协调性
  • (4) 面向目标性
  • (5) 存在社会性
  • (6) 工作协作性
  • (7) 运行持续性
  • (8) 系统适应性
  • (9) 结构分布性
  • (10) 功能智能性

# BDI 模型

Agent 必须利用知识修改其内部状态 (心理状态),以适应环境变化和协作求解的需要。

人类心理状态的要素可分为以下三种

  • 认知 (信念、知识、学习等)
  • 情感 (愿望、兴趣、爱好等)
  • 意向 (意图、目标和承诺等)

受人类心理启发,传统 Agent 理论模型研究的主要方向为:

  • 信念 (Belief)
  • 愿望 (Desire)
  • 意图 (Intention)

的关系及其形式化描述,力图建立 Agent 的 BDI 模型

# 结构

计算机系统为 Agent 的开发和运行提供软件和硬件环境支持,使各个 Agent 依据全局状态协调地完成各项任务。具体地说:

(1) 在计算机系统中,Agent 相当于一个独立的功能模块、独立的计算机应用系统,它含有独立的外部设备、输入输出驱动装备、操作系统、数据结构和相应的输出。

(2) Agent 程序的核心部分叫做决策生成器问题求解器,起到主控作用。

(3) Agent 的运行是一个或多个进程,并接受总体调度。

(4) 各个 Agent 在多个计算机 CPU 上并行运行,其运行环境由体系结构支持。

# 知识型体系结构

将智能体视为一个定理证明器,智能体的行为决策是一个基于知识库的定理证明过程

智能体的内部状态对应于智能体所拥有的知识

各个知识条目用逻辑语言(比如经典的一阶谓词逻辑或者 Hoare 逻辑)来表示

知识型体系结构

基于知识型体系结构的智能体开发方法、技术和工具,代表性工作包括:

  • Lesperance 等人基于格局演算(Situation Calculus)理论的 ConGolog ,它是 Golog(Goal-Oriented Language)的扩展
  • Fisher 基于可执行时序逻辑(Temporal Logic)的多智能体系统编程语言 Concurrent MetateM

知识型体系结构的不足:

  • 表达能力问题
  • 计算复杂性问题
  • 易用性问题

# Concurrent MetateM

一个由 Concurrent MetateM 实现的系统包含一组可并发、异步执行的智能体:

  • 每个智能体由一组可直接执行的时序逻辑公式加以描述
  • 多个智能体间通过广播消息进行通信

对智能体进行编程的本质实际上就是用逻辑符号来定义智能体规约。一个智能体规约对应于一组时序逻辑公式,它描述了智能体行为。

由 Concurrent MetateM 所定义的智能体实际上是一个迭代地建立关于智能体规约逻辑模型的过程。因此,Concurrent MetateM 中的智能体类似于一个定理证明器

Concurrent MetateM 中的每一个智能体包含以下二个主要的部件:

  • 一个描述智能体行为的规约,每一个智能体规约均包含有一个接口(Interface)定义,用来描述智能体是如何与其所驻留的环境(即其他智能体)进行通信的;
  • 一个计算引擎,它负责执行智能体规约。

# 反应型体系结构

智能体对环境事件的响应是通过一组简单的反应式规则来实现的,这些规则描述了智能体的感知输入与响应动作之间的映射关系。

基于这些反应式规则,智能体无需通过复杂的推理和证明就可针对环境事件作出简单、快速地响应。

反应型体系结构

# 多智能体系统

多智能体系统是指由多个相对独立同时又相互作用的智能体所构成的系统

# 特点

# 独立性

每个智能体都是自主的行为实体,封装了行为以及行为控制机制,可以在无需外部指导的情况下实施行为;

# 协同性

这些智能体并不是孤立的,它们之间存在各种关系,需要相互交互和协同进而达成问题的求解

# 无全局视角

对于绝大多数多智能体系统而言,系统中的智能体要获得关于系统以及环境的完整、准确和及时的信息是非常困难的。这是由于智能体所驻留的环境可能会比较复杂,具有以下特点:

  • 开放性
  • 不确定性
  • 动态性
  • ......

# 不可预测和不确定性

在多智能体系统中,由于智能体行为的自主性、智能体行为对环境影响的有限性、系统中无全局控制智能体、智能体间交互行为的动态性,多智能体系统的运行具有不可预测性和不确定性的特点。

# Agent 研究分类

# 知识表示与推理 / 规划

核心假设:人工智能可以通过符号推理形式化刻画。

研究目的:将知识表达为计算机可处理(computer-tractable)的形式,以供智能体使用。

# 非合作博弈

Non-cooperative Game Theory

  • Actions are taken by individual agents
  • No binding agreements

# 拍卖理论

四种类型拍卖:

  • (1) 英式拍卖 (English Auction):又叫升价拍卖,竞价者在一起公开竞标,往上拾价,出价最高者获得拍品。这是我们最常见到的拍卖。
  • (2) 荷式拍卖 (Dutch Auction):又叫降价拍卖,价格由高往低降,第一个接受价格的人获得拍品。
  • (3) 第一价格拍卖 (Fist-price Auction):每个竞买人对拍品进行单独密封报价,但相互不知道其他竞买人的出价,标的装在信封里交上去,然后拍卖人拆开信封,拍卖人按各个标价的大小排序,最后在规定的时间、地点宜布标价,出价最高的竞买人获胜。
  • (4) 第二价格拍卖 (Second-price Auction):二级价格拍卖与一级价格拍卖类似,不同的是最后出价最高的竞买人获得拍品,但只需要按照排位第二高的价格进行支付,即第二价格,因此这种拍卖方式被称为二级密封价格拍卖或 “维克里拍卖”。

四种拍卖形式奠定了拍卖的基本类型,其他无论什么形式的拍卖都只是这四种形式的变型与组合:

拍卖

# 收益等价定理

在一些合理的约束条件保证下,四种拍卖形式最终的收益是相等的。

给定买方人数,假定所有买方都是风险中性的,各买方的价值独立且具有相同的连续分布,而且分布函数严格递增。

那么,任何满足下列两个条件的拍卖机制对于卖方都会产生相同的期望收益(对于买方则产生相同的期望支出):

  • 条件 1:标的物总是为最高信号的买方获得;
  • 条件 2:如果买方的信号位于信号区间的下限,那么他的期望剩余为零。

“收益等价定理” 是整个拍卖理论研究的起点。它表明:对于委托人来说,只要拍卖品不变,购买对象不变,无论采用什么拍卖方式,最终收益都是一样的。

当然 “收益等价定理” 说的是理想状态,这里我们千万不要忘记约束条件。在现实中,虽然拍卖品是共同的,但由于参与竞标的人不同、不同的拍卖流程和规则,结果产生拍卖结果的巨大差异。甚至同样一种拍卖办法,放到不同的国家,结果也会有天壤之别,原因在于现实的约束条件不同。

# 合作博弈

Cooperative / Colitional Game Theory

  • Actions are taken by groups of agents
  • Binding agreements are possible

合作博弈是一种侧重于玩家群体行为的模型。与非合作博弈不同,合作博弈中的战略互动没有被明确建模。

# 基本过程

  • Agent 们组成多个联盟(coalitions)
  • 每个联盟隐性地选择动作
  • 可转让效用(TU)博弈:联盟行动的选择(由所有联盟决定) 决定了每个联盟的回报 —— 联盟中的 Agents 随后需要决定如何分配这个总回报

# 人 - Agent 交互

仅仅拥有自主性(Autonomy)的 Agent 并不足以作为人 - Agent 交互的研究对象

当人们塑造他们与物体或技术的互动,好像它们有目的、动机或意图时,我们就有了人 - Agent 交互

通常,技术被设计成鼓励人们赋予它们 Agent 的性质,尽管这并非 Agent 的必要条件。人们会将 Agent 的性质赋予许多日常生活中的事物和技术,这是一个常见的研究领域

人机互补 —— 人类和 agent 各自互补缺点,利用优点

Edited on Views times
\ No newline at end of file +Agent - 人工智能 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

# 智能体

# 定义

Agent 是指驻留在某一环境下能够自主(Autonomous)、灵活(Flexible)地执行动作以满足设计目标的行为实体。

上述定义具有如下二个特点。

  • (1) 定义方式 : Agent 概念定义是基于 Agent 的外部可观察行为特征,而不是其内部的结构
  • (2) 抽象层次 : Agent 概念更加贴近于人们对现实世界(而不是计算机世界)中行为实体的理解

# 特点

# 驻留性

# 自主性

智能体能在没有人类或其他智能体的干涉和指导的情况下运行,并能根据其内部状态和感知到的环境信息来决定该实施什么样的动作,进而控制自身的行为

# 灵活性

智能体行为的灵活性主要体现为以下三方面:

# 反应性

(Reactive)

反应性是指智能体能够感知所处的环境,并能对环境中发生的相关事件(比如智能体间的交互、用户的指令等)作出适时反应,以满足系统设计目标

# 自发性

(Pro-active)

自发性是指智能体不仅具有目标并根据目标行事,而且能够主动地产生目标,进而实施自发的行为

# 社会性

(Social)

社会性是指智能体所驻留的环境可能存在其他智能体,它拥有这些智能体的有关信息和知识(比如这些智能体的物理位置、所拥有的资源和能力等),并能与它们进行交互和通信,实现灵活多样和复杂的合作、协商和竞争,以满足系统的设计目标

# 其他特性

Agent 与分布式人工智能系统一样具有协作性、适应性等特性

  • (1) 行为自主性
  • (2) 作用交互性(也称反应性)
  • (3) 环境协调性
  • (4) 面向目标性
  • (5) 存在社会性
  • (6) 工作协作性
  • (7) 运行持续性
  • (8) 系统适应性
  • (9) 结构分布性
  • (10) 功能智能性

# BDI 模型

Agent 必须利用知识修改其内部状态 (心理状态),以适应环境变化和协作求解的需要。

人类心理状态的要素可分为以下三种

  • 认知 (信念、知识、学习等)
  • 情感 (愿望、兴趣、爱好等)
  • 意向 (意图、目标和承诺等)

受人类心理启发,传统 Agent 理论模型研究的主要方向为:

  • 信念 (Belief)
  • 愿望 (Desire)
  • 意图 (Intention)

的关系及其形式化描述,力图建立 Agent 的 BDI 模型

# 结构

计算机系统为 Agent 的开发和运行提供软件和硬件环境支持,使各个 Agent 依据全局状态协调地完成各项任务。具体地说:

(1) 在计算机系统中,Agent 相当于一个独立的功能模块、独立的计算机应用系统,它含有独立的外部设备、输入输出驱动装备、操作系统、数据结构和相应的输出。

(2) Agent 程序的核心部分叫做决策生成器问题求解器,起到主控作用。

(3) Agent 的运行是一个或多个进程,并接受总体调度。

(4) 各个 Agent 在多个计算机 CPU 上并行运行,其运行环境由体系结构支持。

# 知识型体系结构

将智能体视为一个定理证明器,智能体的行为决策是一个基于知识库的定理证明过程

智能体的内部状态对应于智能体所拥有的知识

各个知识条目用逻辑语言(比如经典的一阶谓词逻辑或者 Hoare 逻辑)来表示

知识型体系结构

基于知识型体系结构的智能体开发方法、技术和工具,代表性工作包括:

  • Lesperance 等人基于格局演算(Situation Calculus)理论的 ConGolog ,它是 Golog(Goal-Oriented Language)的扩展
  • Fisher 基于可执行时序逻辑(Temporal Logic)的多智能体系统编程语言 Concurrent MetateM

知识型体系结构的不足:

  • 表达能力问题
  • 计算复杂性问题
  • 易用性问题

# Concurrent MetateM

一个由 Concurrent MetateM 实现的系统包含一组可并发、异步执行的智能体:

  • 每个智能体由一组可直接执行的时序逻辑公式加以描述
  • 多个智能体间通过广播消息进行通信

对智能体进行编程的本质实际上就是用逻辑符号来定义智能体规约。一个智能体规约对应于一组时序逻辑公式,它描述了智能体行为。

由 Concurrent MetateM 所定义的智能体实际上是一个迭代地建立关于智能体规约逻辑模型的过程。因此,Concurrent MetateM 中的智能体类似于一个定理证明器

Concurrent MetateM 中的每一个智能体包含以下二个主要的部件:

  • 一个描述智能体行为的规约,每一个智能体规约均包含有一个接口(Interface)定义,用来描述智能体是如何与其所驻留的环境(即其他智能体)进行通信的;
  • 一个计算引擎,它负责执行智能体规约。

# 反应型体系结构

智能体对环境事件的响应是通过一组简单的反应式规则来实现的,这些规则描述了智能体的感知输入与响应动作之间的映射关系。

基于这些反应式规则,智能体无需通过复杂的推理和证明就可针对环境事件作出简单、快速地响应。

反应型体系结构

# 多智能体系统

多智能体系统是指由多个相对独立同时又相互作用的智能体所构成的系统

# 特点

# 独立性

每个智能体都是自主的行为实体,封装了行为以及行为控制机制,可以在无需外部指导的情况下实施行为;

# 协同性

这些智能体并不是孤立的,它们之间存在各种关系,需要相互交互和协同进而达成问题的求解

# 无全局视角

对于绝大多数多智能体系统而言,系统中的智能体要获得关于系统以及环境的完整、准确和及时的信息是非常困难的。这是由于智能体所驻留的环境可能会比较复杂,具有以下特点:

  • 开放性
  • 不确定性
  • 动态性
  • ......

# 不可预测和不确定性

在多智能体系统中,由于智能体行为的自主性、智能体行为对环境影响的有限性、系统中无全局控制智能体、智能体间交互行为的动态性,多智能体系统的运行具有不可预测性和不确定性的特点。

# Agent 研究分类

# 知识表示与推理 / 规划

核心假设:人工智能可以通过符号推理形式化刻画。

研究目的:将知识表达为计算机可处理(computer-tractable)的形式,以供智能体使用。

# 非合作博弈

Non-cooperative Game Theory

  • Actions are taken by individual agents
  • No binding agreements

# 拍卖理论

四种类型拍卖:

  • (1) 英式拍卖 (English Auction):又叫升价拍卖,竞价者在一起公开竞标,往上拾价,出价最高者获得拍品。这是我们最常见到的拍卖。
  • (2) 荷式拍卖 (Dutch Auction):又叫降价拍卖,价格由高往低降,第一个接受价格的人获得拍品。
  • (3) 第一价格拍卖 (Fist-price Auction):每个竞买人对拍品进行单独密封报价,但相互不知道其他竞买人的出价,标的装在信封里交上去,然后拍卖人拆开信封,拍卖人按各个标价的大小排序,最后在规定的时间、地点宜布标价,出价最高的竞买人获胜。
  • (4) 第二价格拍卖 (Second-price Auction):二级价格拍卖与一级价格拍卖类似,不同的是最后出价最高的竞买人获得拍品,但只需要按照排位第二高的价格进行支付,即第二价格,因此这种拍卖方式被称为二级密封价格拍卖或 “维克里拍卖”。

四种拍卖形式奠定了拍卖的基本类型,其他无论什么形式的拍卖都只是这四种形式的变型与组合:

拍卖

# 收益等价定理

在一些合理的约束条件保证下,四种拍卖形式最终的收益是相等的。

给定买方人数,假定所有买方都是风险中性的,各买方的价值独立且具有相同的连续分布,而且分布函数严格递增。

那么,任何满足下列两个条件的拍卖机制对于卖方都会产生相同的期望收益(对于买方则产生相同的期望支出):

  • 条件 1:标的物总是为最高信号的买方获得;
  • 条件 2:如果买方的信号位于信号区间的下限,那么他的期望剩余为零。

“收益等价定理” 是整个拍卖理论研究的起点。它表明:对于委托人来说,只要拍卖品不变,购买对象不变,无论采用什么拍卖方式,最终收益都是一样的。

当然 “收益等价定理” 说的是理想状态,这里我们千万不要忘记约束条件。在现实中,虽然拍卖品是共同的,但由于参与竞标的人不同、不同的拍卖流程和规则,结果产生拍卖结果的巨大差异。甚至同样一种拍卖办法,放到不同的国家,结果也会有天壤之别,原因在于现实的约束条件不同。

# 合作博弈

Cooperative / Colitional Game Theory

  • Actions are taken by groups of agents
  • Binding agreements are possible

合作博弈是一种侧重于玩家群体行为的模型。与非合作博弈不同,合作博弈中的战略互动没有被明确建模。

# 基本过程

  • Agent 们组成多个联盟(coalitions)
  • 每个联盟隐性地选择动作
  • 可转让效用(TU)博弈:联盟行动的选择(由所有联盟决定) 决定了每个联盟的回报 —— 联盟中的 Agents 随后需要决定如何分配这个总回报

# 人 - Agent 交互

仅仅拥有自主性(Autonomy)的 Agent 并不足以作为人 - Agent 交互的研究对象

当人们塑造他们与物体或技术的互动,好像它们有目的、动机或意图时,我们就有了人 - Agent 交互

通常,技术被设计成鼓励人们赋予它们 Agent 的性质,尽管这并非 Agent 的必要条件。人们会将 Agent 的性质赋予许多日常生活中的事物和技术,这是一个常见的研究领域

人机互补 —— 人类和 agent 各自互补缺点,利用优点

Edited on Views times
\ No newline at end of file diff --git a/cs/ai/concepts/index.html b/cs/ai/concepts/index.html index a3bac90a..c1423386 100644 --- a/cs/ai/concepts/index.html +++ b/cs/ai/concepts/index.html @@ -1 +1 @@ -人工智能绪论 - 人工智能 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

# 人工智能的概念

人工智能(Artificial Intelligence - AI):研究如何通过计算机技术实现一个智能体,让智能体完成类似于人类智能的行为。

# 智能体 Agent

完成智能行为的主体,是物理或抽象实体,可以感知世界并对世界施加作用。

大模型智能体 LLM Agent

在近几年随 ChatGPT 兴起的新概念

# 智能体的两大目标

  • 能否使智能体像人一样思考、理性地思考?
  • 能否使智能体像人一样行动、理性地行动?

# 智能体可否像人一样思考

首先,需要测量人如何思考的方法:内省、心理实验、 脑成像。

# 认知科学

交叉学科,融合人工智能的计算机模型和心理学实验技术,试图构建一种精确可测试的人类思维理论。

# 认知神经科学

交叉学科,以认知科学的理论以及神经心理学、神经科学及计算机模型的实验证据为基础。

# 智能体如何像人一样行动

# 图林测试

1950 年,英国数学家图灵 (Alan Turing) 提出了一个测试方法来确定一个机器是否具有人类智能。

# 图林测试内容
  • 两个人对机器进行测试:一人扮演提问者,另外一人作为被测者。
  • 这两人与机器分别处在三个不同的房间,提问者通过打印问题和接受打印问题来与被测者和被测机器进行通迅。提问者可以向被测机器和被测者提问,但他只知道接受提问的是 A 或 B,而并不知道他是人还是机器。
  • 若超过 30% 的测试者不能确定出被测试者是人还是机器,那么这台机器就通过了测试,并被认为具有人类智能。
# 图灵提出了 AI 的几大元素
  • 知识
  • 推理
  • 语言
  • 理解
  • 学习
# 图灵测试存在的问题
  • 图灵测试不可重复
  • 无法数学分析

# ⭐️三大主流学派

# 逻辑学派

符号主义 / 功能主义方法,模拟大脑逻辑思维功能

原理主要为物理符号系统 (即符号操作系统) 假设和有限合理性原理。

# 仿生学派

联结主义 / 结构主义方法,模拟大脑结构

原理主要为神经网络及神经网络间的连接机制与学习算法

# 控制论学派

行为主义 / 进化主义方法,模拟智能系统行为

原理为控制论及感知 — 动作型控制系统

# 理论基础

# 物理符号系统的假设

一个完善的符号系统应具有下列 6 种功能

  • 输入 (input) 符号
  • 输出 (output) 符号
  • 存储 (store) 符号
  • 复制 (copy) 符号
  • 建立符号结构:通过找出各符号间的关系,在符号系统中形成符号结构
  • 条件性迁移 (conditional transfer) : 根据已有符号,继续完成活动过程

任何一个系统,如果它能够表现出智能,那么它就必定能够执行上述 6 种功能。反之,任何系统如果具有这 6 种功能,那么它就能够表现出智能;这种智能指的是人类所具有的那种智能。

物理符号系统的假设伴随 3 个推论,或称为附带条件

  • 既然人具有智能,那么他就 一定是 一个物理符号系统。人之所以能够表现出智能,就是基 于他的信息处理过程
  • 既 然 计 算 机 是 一个 物 理 符 号 系 统 ,它 就 一定 能 够 表 现 出 智 能 。这 是 人 工 智能的基本条件。
  • 既然人是 一个物理符号系统,计算机也是 一个物理符号系统,那么就能够 用计算机模拟人的活动

# 人工智能的内涵

  • 搜索
  • 知识
  • 推理
  • 归纳
  • 演绎
  • ...

# 强 / 弱 人工智能

# 强人工智能

一个机器和人的思考行为都完全一样

# 弱人工智能

只能表面上执行看上去的智能行为

# 泛 / 狭义 人工智能

# 泛人工智能

智能体拥有在任何问题上应用智能的能力

# 狭义人工智能

只能完成一项具体任务的人工智能

# 人工智能的发展历史

# 不同阶段

# 1950-1970

早期爆发阶段,巨大期望

  • 1952:Samuel 写了一个能学习的五子棋程序
  • 1955:Newell 和 Simon 写了一个推理的程序

# 1970-1990

知识就是力量,知识工程

  • 1980-1988:专家系统工业爆发
  • 1988-1993: 日本第五代计算机系统项目失败

# 1990-

机器学习兴起,人工智能的春天

# 2010-

深度学习获得工业投资和应用

# 三次大跃进

  1. 智能系统代替人完成部分逻辑推理工作,如机器 定理证明和专家系统
  2. 智能系统能够和环境交互,从运行的环境中获取信息,代替人完成包括不定性在内的部分思维 工作,通过自身的动作,对环境施加影响,并适应环境的变化,如智能机器人
  3. 智 能 系 统 具 有 人 类 的 认 知 和 思 维 能 力 ,能 够 发 现 新 的 知 识 ,完 成 面 临 的 任 务 ,如 基 于 数 据 挖 掘 的 系 统

# 人工智能研究课题

# 智 能 感 知

# 模式识别

# 计算机视觉

# 自然语言处理

# 智 能 推 理

# 搜索技术

# 问题求解

# 定理证明

# 专家系统和知识库

# 大数据知识工程

# 智 能 学 习

# 记忆与联想

# 神经网络

# 深度学习、迁移学习

# 计算智能与进化计算

# 遗传算法

# 数据挖掘与知识发现

# 智 能 行 动

# 智能检索

# 智能调度与指挥

# 智能控制

# 人机对话系统

# 智能机器人

# 分布式人工智能与 Agent

# 游戏

# 人机智能融合

# 人工智能实例

  • 专家系统:在专门的领域(医疗、探矿、财务等领域)内的咨询服务系统
  • 自然语言处理:在有限范围内的问题回答系统。
  • 程序验证系统:通过定理证明途径验证程序的正确性。
  • 智能机器人:通过计算机视觉和智能控制完成具体行动任务
Edited on Views times
\ No newline at end of file +人工智能绪论 - 人工智能 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

# 人工智能的概念

人工智能(Artificial Intelligence - AI):研究如何通过计算机技术实现一个智能体,让智能体完成类似于人类智能的行为。

# 智能体 Agent

完成智能行为的主体,是物理或抽象实体,可以感知世界并对世界施加作用。

大模型智能体 LLM Agent

在近几年随 ChatGPT 兴起的新概念

# 智能体的两大目标

  • 能否使智能体像人一样思考、理性地思考?
  • 能否使智能体像人一样行动、理性地行动?

# 智能体可否像人一样思考

首先,需要测量人如何思考的方法:内省、心理实验、 脑成像。

# 认知科学

交叉学科,融合人工智能的计算机模型和心理学实验技术,试图构建一种精确可测试的人类思维理论。

# 认知神经科学

交叉学科,以认知科学的理论以及神经心理学、神经科学及计算机模型的实验证据为基础。

# 智能体如何像人一样行动

# 图林测试

1950 年,英国数学家图灵 (Alan Turing) 提出了一个测试方法来确定一个机器是否具有人类智能。

# 图林测试内容
  • 两个人对机器进行测试:一人扮演提问者,另外一人作为被测者。
  • 这两人与机器分别处在三个不同的房间,提问者通过打印问题和接受打印问题来与被测者和被测机器进行通迅。提问者可以向被测机器和被测者提问,但他只知道接受提问的是 A 或 B,而并不知道他是人还是机器。
  • 若超过 30% 的测试者不能确定出被测试者是人还是机器,那么这台机器就通过了测试,并被认为具有人类智能。
# 图灵提出了 AI 的几大元素
  • 知识
  • 推理
  • 语言
  • 理解
  • 学习
# 图灵测试存在的问题
  • 图灵测试不可重复
  • 无法数学分析

# ⭐️三大主流学派

# 逻辑学派

符号主义 / 功能主义方法,模拟大脑逻辑思维功能

原理主要为物理符号系统 (即符号操作系统) 假设和有限合理性原理。

# 仿生学派

联结主义 / 结构主义方法,模拟大脑结构

原理主要为神经网络及神经网络间的连接机制与学习算法

# 控制论学派

行为主义 / 进化主义方法,模拟智能系统行为

原理为控制论及感知 — 动作型控制系统

# 理论基础

# 物理符号系统的假设

一个完善的符号系统应具有下列 6 种功能

  • 输入 (input) 符号
  • 输出 (output) 符号
  • 存储 (store) 符号
  • 复制 (copy) 符号
  • 建立符号结构:通过找出各符号间的关系,在符号系统中形成符号结构
  • 条件性迁移 (conditional transfer) : 根据已有符号,继续完成活动过程

任何一个系统,如果它能够表现出智能,那么它就必定能够执行上述 6 种功能。反之,任何系统如果具有这 6 种功能,那么它就能够表现出智能;这种智能指的是人类所具有的那种智能。

物理符号系统的假设伴随 3 个推论,或称为附带条件

  • 既然人具有智能,那么他就 一定是 一个物理符号系统。人之所以能够表现出智能,就是基 于他的信息处理过程
  • 既 然 计 算 机 是 一个 物 理 符 号 系 统 ,它 就 一定 能 够 表 现 出 智 能 。这 是 人 工 智能的基本条件。
  • 既然人是 一个物理符号系统,计算机也是 一个物理符号系统,那么就能够 用计算机模拟人的活动

# 人工智能的内涵

  • 搜索
  • 知识
  • 推理
  • 归纳
  • 演绎
  • ...

# 强 / 弱 人工智能

# 强人工智能

一个机器和人的思考行为都完全一样

# 弱人工智能

只能表面上执行看上去的智能行为

# 泛 / 狭义 人工智能

# 泛人工智能

智能体拥有在任何问题上应用智能的能力

# 狭义人工智能

只能完成一项具体任务的人工智能

# 人工智能的发展历史

# 不同阶段

# 1950-1970

早期爆发阶段,巨大期望

  • 1952:Samuel 写了一个能学习的五子棋程序
  • 1955:Newell 和 Simon 写了一个推理的程序

# 1970-1990

知识就是力量,知识工程

  • 1980-1988:专家系统工业爆发
  • 1988-1993: 日本第五代计算机系统项目失败

# 1990-

机器学习兴起,人工智能的春天

# 2010-

深度学习获得工业投资和应用

# 三次大跃进

  1. 智能系统代替人完成部分逻辑推理工作,如机器 定理证明和专家系统
  2. 智能系统能够和环境交互,从运行的环境中获取信息,代替人完成包括不定性在内的部分思维 工作,通过自身的动作,对环境施加影响,并适应环境的变化,如智能机器人
  3. 智 能 系 统 具 有 人 类 的 认 知 和 思 维 能 力 ,能 够 发 现 新 的 知 识 ,完 成 面 临 的 任 务 ,如 基 于 数 据 挖 掘 的 系 统

# 人工智能研究课题

# 智 能 感 知

# 模式识别

# 计算机视觉

# 自然语言处理

# 智 能 推 理

# 搜索技术

# 问题求解

# 定理证明

# 专家系统和知识库

# 大数据知识工程

# 智 能 学 习

# 记忆与联想

# 神经网络

# 深度学习、迁移学习

# 计算智能与进化计算

# 遗传算法

# 数据挖掘与知识发现

# 智 能 行 动

# 智能检索

# 智能调度与指挥

# 智能控制

# 人机对话系统

# 智能机器人

# 分布式人工智能与 Agent

# 游戏

# 人机智能融合

# 人工智能实例

  • 专家系统:在专门的领域(医疗、探矿、财务等领域)内的咨询服务系统
  • 自然语言处理:在有限范围内的问题回答系统。
  • 程序验证系统:通过定理证明途径验证程序的正确性。
  • 智能机器人:通过计算机视觉和智能控制完成具体行动任务
Edited on Views times
\ No newline at end of file diff --git a/cs/ai/dl/index.html b/cs/ai/dl/index.html index 202f16af..0288905e 100644 --- a/cs/ai/dl/index.html +++ b/cs/ai/dl/index.html @@ -1 +1 @@ -| Balloon Party = 風船のパーティー = fuusen no party

# MP

正向传播

反向传播

https://www.cnblogs.com/charlotte77/p/5629865.html

感知机训练

BP 算法

# CNN

最大池化

max pooling

可以是取最大值、平均值。保留区域的特征,降维降低计算复杂度

# RNN

在 CNN 的基础上,增加了记忆功能

# LSTM

长短期记忆网络 (Long Short-term Memory)

# BPTT

(Backpropagation through time)

# Transformer

通过最小化交叉熵 (minimize cross entropy) 训练

# Encoder

# Decoder

Edited on Views times
\ No newline at end of file +| Balloon Party = 風船のパーティー = fuusen no party

# MP

正向传播

反向传播

https://www.cnblogs.com/charlotte77/p/5629865.html

感知机训练

BP 算法

# CNN

最大池化

max pooling

可以是取最大值、平均值。保留区域的特征,降维降低计算复杂度

# RNN

在 CNN 的基础上,增加了记忆功能

# LSTM

长短期记忆网络 (Long Short-term Memory)

# BPTT

(Backpropagation through time)

# Transformer

通过最小化交叉熵 (minimize cross entropy) 训练

# Encoder

# Decoder

Edited on Views times
\ No newline at end of file diff --git a/cs/ai/knowledge-presentation-and-resolution/index.html b/cs/ai/knowledge-presentation-and-resolution/index.html index 440a4a72..36e44204 100644 --- a/cs/ai/knowledge-presentation-and-resolution/index.html +++ b/cs/ai/knowledge-presentation-and-resolution/index.html @@ -1 +1 @@ -知识表示与推理 - 人工智能 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

# 知识和知识表示

# 知识

  • 数据一般指单独的事实,是信息的载体
  • 信息由符号组成,如文字和数字,但是对符号赋予了一定的意义,因此有一定的用途或价值
  • 知识是由经验总结升华出来的,因此知识是经验的结晶。知识在信息的基础上增加了上下文信息,提供了更多的意义,因此也就更加有用和有价值。知识是随着时间的变化而动态变化的,新的知识可以根据规则和已有的知识推导出来。

知识是经过加工的信息,它包括:

  • 事实:是关于对象和物体的知识。
  • 规则:是有关问题中与事物的行动、动作相联系的因果关系的知识。
  • 元知识:是有关知识的知识,是知识库中的高层知识。
  • 常识性知识:泛指普遍存在而且被普遍认识了的客观事实一类知识。

# 策略

策略 —— 关于如何解决问题的政策方略,包括在什么时间、什么地点、由什么主体采取什么行动、达到什么目标、注意什么事项等等一整套完整而具体的行动计划规划、行动步骤、工作方式和工作方法

# 智能

在给定的 问题 - 问题环境 - 主体目的 的条件下,智能就是有针对性地获取 问题 - 环境 的信息,恰当地对这些信息进行处理以提炼知识达到认知,在此基础上,把已有的知识与主体的目的信息相结合,合理地产生解决问题的策略信息,并利用所得到的策略信息在给定的环境下成功地解决问题达到主体的目的 (行为)。

省流:信息经加工提炼而成知识,知识被目的激活而成智能。

智能具有 4 个能力:

  • 获取有用信息的能力
  • 由信息生成 == 知识 (认知)== 的能力
  • 由知识和目的生成 == 策略 (决策)== 的能力
  • 实施策略取得效果 (施效) 的能力

# 知识表示语言问题

  • 语法 —— 语言的语法描述了组成语句的可能的搭配关系。
  • 语义 —— 语义定义了语句所指的世界中的事实。

从语法和语义,可以给出使用该语言的 Agent 的必要的推理机制。

基于该推理机制,Agent 可以从已知的语句推导出结论,或判断某条信息是不是已蕴涵在现有的知识当中。

知识表示语言需要:

  • 语法规则
  • 语义解释
  • 用于演绎和推导的规则

知识表示语言应该支持知识不完全的情况。不能表达这种不完全性的语言是表达能力不够的语言

程序设计语言比较善于描述算法和具体的数据结构

# 现代逻辑学的基本研究方法

逻辑学 (logic) 是研究人类思维规律的科学,而现代逻辑学则是用数学 (符号化、公理化、形式化) 的方法来研究这些规律。

# 符号化

所谓符号化即是用 “一种只作整体认读的记号 (signs)”—— 符号 (symbols) 表示量、数及数量关系。也有用字母、符号表示思维的物理对象、概念对象、判断对象等。

语言化是符号化的初级阶段。

现代逻辑学对思维的研究,需要更加彻底的符号化过程。

# 公理化

欧氏几何公理系统中的所有概念都有鲜明的直观背景,其公理、定理也都有强烈的客观意义。像欧氏几何这样的公理系统,常被称为具体公理系统。

始于 Aristotle 的逻辑学被符号化、公理化,逐步演化为现代逻辑学。

# 形式化

所谓形式化,就是彻头彻尾的 “符号化 + 抽象公理化

现代逻辑学形式系统组成:

  • 用于将概念符号化的符号语言,通常为一形式语言,包括一符号表∑及语言的文法,可生成表示对象的语言成分项,表示概念、判断的公式;
  • 表示思维规律的逻辑学公理模式和推理规则模式 (抽象公理系统),及其依据它们推演可得到的全部定理组成的理论体系。

# 命题逻辑

以下讨论实际上均为二值逻辑

# 基本概念

# 命题

命题是具有真假意义的陈述句。

在特殊的情况下都具有 “真 (True)” 和 “假(False)” 的意义句子,都是命题。真值 —— 用 T 和 F 表示。

命题有两种类型:

  • 原子命题
  • 复合命题 —— 由以下内容复合构成的命题:
    • 联结词
    • 标点符号
    • 原子命题

# 命题逻辑

命题逻辑就是研究命题和命题之间关系的符号逻辑系统。

# 命题逻辑的符号

命题逻辑的符号:

  • 命题常元:True (T) 和 False (F)
  • 命题标识符:P、Q、R 等;
  • 联结词:
    • ¬(否定)
    • \land(合取)
    • \lor (析取)
    • →(蕴含)
    • ↔︎(等价)
  • 括号:( )

# 命题标识符

用 P、Q、R、S 等来表示命题。可分为:

  • 命题常量(表示确定的命题的命题标识符)
  • 命题变元(只表示任意命题的位置标志的命题标识符)
    • 当命题变元表示原子命题时,该变元称为原子变元

因为命题变元可以表示任意命题,所以它不能确定真值,故命题变元不是命题。

# 联结词

复合命题的意义是命题组成成份的函数。

联结词的语义可以定义如下:

  • ¬P 为真,当且仅当 P 为假
  • PQP \land Q 为真,当且仅当 P 和 Q 都为真
  • PQP \lor Q 为真,当且仅当 P 为真,或者 Q 为真
  • P → Q 为真,当且仅当 P 为假,或者 Q 为真
  • P ↔︎ Q 为真,当且仅当 P→Q 为真,并且 Q→P 为真

# 指派

当命题变元 P 用一个特定的命题取代时,P 才能确定真值,这时也称为对命题变元 P 进行指派

设 G 是公式,A1, ... , An, 为 G 中出现的所有原子命题。公式 G 的一种指派是对 A1, ... , An 赋予的一组真值,其中每个 Ai (i=1 , ... , n) 或者为 T 或者为 F。

公式 G 称为在一种指派下为真,当且仅当 G 按该指派算出的真值为 T,否则称为在该指派下为假

若在公式中有 n 个不同的原子 A1 , ... , An,那么该公式就有 2n2^n 个不同的指派。

# 永真 / 可满足 / 永假

公式 AA 称为永真式重言式 (tautology),如果对任意指派 α,α 均弄真 A,即 α(A)=T 。

公式 AA 称为可满足的 (satisfiable), 如果存在指派 α 使 α(A)=T,否则称 AA不可满足的 (unsatisfiable),或永假式

永真式是可满足的

AA 为永真式 (永假式) 时, ¬A\neg A 为永假式 (永真式)

# 蕴含 / 等价

称公式 AA 逻辑蕴涵公式 B,记为 ABA\to B ,如果所有弄真 AA 的指派亦必弄真公式 B

称公式集 Γ\Gamma 逻辑蕴涵公式 B,记为 ΓB\Gamma \to B ,如果弄真 Γ\Gamma 中所有公式的指派亦必弄真公式 B

称公式 AA 逻辑等价公式 B,记为 ABA\lrarr B ,如果 ABA\to BBAB\to A

  • 逻辑蕴涵关系具有
    • 自反性
    • 反对称性
    • 传递性
  • 逻辑等价关系具有
    • 自反性
    • 对称性
    • 传递性

# 代换

设 A 为含有命题变元 p 的永真式,那么将 A 中 p 的所有出现均代换为命题公式 B,所得公式 (称为 A 的代入实例) 仍为永真式。

# 替换

设命题公式 AA 含有子公式 C (C 为 A 中的符号串,且 C 为命题公式),如果 CDC\lrarr D,那么将 AA 中子公式 C 的某些出现 (未必全部) 用 D 替换后所得公式 B 满足 ABA\lrarr B

# 合 / 析 取范式

命题公式 B 称为命题公式 AA合取(或析取范式,如果 BAB\lrarr A,且 B 呈如下形式:

C1C2Cm(C1C2Cm),Ci(i=1,2,...,m)C_1\land C_2\land …\land C_m(或 C_1\lor C_2\lor …\lor C_m)\ \ ,\ \ C_i (i=1, 2,...,m)

其中 CiC_i 形如:

L1L2...Ln(L1L2Ln),Lj(j=l,2,...,n)L_1\lor L_2\lor ...\lor L_n (或L_1\land L_2\land …\land L_n)\ \ ,\ \ L_j(j=l,2,...,n)

其中 LiL_i 为原子公式或原子公式的否定,并称 LjL_j文字

【定理】

任一命题公式 φ 有其对应的合取 (析取) 范式。

# 主 合 / 析 取范式

命题公式 B 称为公式 A 的主合取 (或主析取) 范式,如果:

  • B 是 AA 的合取 (或析取) 范式
  • B 中每一子句均有 AA 中命题变元的全部出现,且仅出现一次。

n 元命题公式的全体可以划分为 22n2^{2^n} 个等价类,每一类中的公式彼此逻辑等价,并等价于它们共同的主合取范式 (或主析取范式)。

# 命题演算形式系统 PC

# 合式公式

pl,p2,p3,...p_l,p_2,p_3,... 为命题逻辑的合式公式

如果 A,B 是公式,那么这些也是命题逻辑的合式公式:

  • (¬A)(\neg A)
  • (AB)(A→B)

# 形式系统 PC 的组成

命题逻辑的形式系统 PC 包括 3 条公理模式 (A1A3)(A_1-A_3) 和 1 条推理规则(即分离规则rmpr_{mp}

  • A1:A(BA)A_1 : A\to(B\to A)
  • A2:A(BC)(AB)(AC)A_2 : A\to(B\to C)\to (A\to B)\to(A\to C)
  • A3:(¬A¬B)(BA)A_3 : (\neg A\to\neg B)\to(B\to A)
  • rmp:A,ABBr_{mp}:\frac{A,A\to B}{B}

# 证明

称下列公式序列为公式 AA 在 PC 中的一个证明 (proof):

A1,A2,...,Am(=A)A_1,A_2,...,A_m(=A)

其中 Ai(i=1,2,...,m)A_i (i=1,2,...,m)

  • 或者是 PC 的公理
  • 或者是 Aj(j<i)A_j (j\lt i)
  • 或者是由 Aj,Ak(j,k<i)A_j,A_k (j,k\lt i) 使用分离规则所导出
  • $A_m =A $

# 定理

AA 为 PC 中的定理,记为 PCA\vdash_{PC} A,如果公式 AA 在 PC 中有一个证明。

\vdash ( \vdash ) 断定符,表示可被推导出,可被证明

A\vDash A ( \vDash ) 表示 AA 永真

# 演绎

Γ\Gamma 为一公式集,称以下公式序列为公式 AA 的,以 Γ\Gamma 为前提的演绎

A1,A2,...,Am=A,Ai(i=1,2,...,m)A_1,A_2,...,A_m=A\ \ ,\ \ A_i (i=1,2,...,m)

其中 AiA_i 满足:

  • 或者是 PC 的公理
  • 或者是 Γ\Gamma 的成员
  • 或者是 Aj(j<i)A_j (j<i)
  • 或者是由 Aj,Ak(j,k<i)A_j, A_k (j,k\lt i) 使用分离规则所导出
  • Am=AA_m=A

# 带前提的演绎

AA 为前提 Γ\Gamma 的演绎结果,如果公式 AA 有以 Γ\Gamma 为前提的演绎。记为:

ΓPCA\Gamma\vdash_{PC}A

Γ={B}\Gamma=\{B\}ΓPCA\Gamma\vdash_{PC}A 可以表示为:

BPCAB\vdash_{PC}A

BPCAB\vdash_{PC}AAPCBA\vdash_{PC}B 则记为:

ABA\vdash\dashv B

# 演绎定理

对应离散数学课程中的 “附加前提证明法”

对 PC 中任意公式集 Γ\Gamma 和公式 A,BA,B

Γ{A}PCBΓPC(AB)\Gamma\cup\{A\}\vdash_{PC}B\lrArr\Gamma\vdash_{PC}(A\to B)

# 可靠性定理

PC 是可靠的,即对任意公式集 Γ\Gamma 及公式 AA

ΓAΓA\Gamma\vdash A\to \Gamma\vDash A

特别地,若 AA 为 PC 的定理 (A)(\vdash A) ,则 AA 永真 (A)(\vDash A) :

AA\vdash A\to \vDash A

# 一致性定理

PC 是一致的 (consistent),即不存在公式 A,使得 AA¬A\neg A 均为 PC 之定理。

# 完全性定理

PC 是完全的,即对任意公式集 Γ\Gamma 和公式 AA

ΓAΓA\Gamma\vDash A\to \Gamma\vdash A

特别地,若 AA 永真 (A)(\vDash A) ,A 必为 PC 的定理 (A)(\vdash A)

AA\vDash A\to \vdash A

# 谓词逻辑

谓词逻辑的语法元素表示如下。

  • 常量符号: A、B、张三、李四 等,通常是对象名称
  • 变量符号:通常用小写字母表示,如 x、 y、z 等
  • 函数符号:通常用小写英文字母或小写英文字母串表示,如 plus、f、g
    • 一元函数、二元函数、...... n 元函数
  • 谓词符号:通常用大写英文字母或 (首字母) 大写英文字母串表示
    • 一元谓词、二元谓词、...... n 元谓词
  • 联结词: ......
  • 量词:全称量词 \forall、存在量詞 \existx\forall x 表示 “対个体域中所有 xx”、x\exist x 表示 “在个 体域中存在个体 xx ”。\forall\exist 后面跟的 xx 叫作量词的指导变元

一般一元谓词表达了个体的性质,而多元谓词表达了个体之间的关系

# 谓词的阶

如果谓词 P 中的所有个体都是:

  • 个体常量
  • 变元
  • 函数

则该谓词为一阶谓词,如果某个个体本身又是一个一阶谓词,则称 PP二阶谓词,以此类推递归定义 nn 阶谓词

#

项可递归定义如下:

  1. 单独一个个体是项 (包括常量和变量)
  2. ffnn 元函数符号,而 t1,t2,...,tnt_1,t_2,...,t_n 是项,则 f(t1,t2,...,tn)f(t_1,t_2,...,t_n) 是项
  3. 任何项仅由规则 (1)、(2) 生成。

# 原子公式

PPnn 元谓词符号,t1,t2,...,tnt_1,t_2,...,t_n 都是项,则称 P(t1,t2,...,tn)P(t_1,t_2,...,t_n) 为原子公式。在原子公式中,若 t1,t2,...,tnt_1,t_2,...,t_n 都不含变量,则 P(t1,t2,...,tn)P(t_1,t_2,...,t_n) 是命题。

# 归结推理

# 最一般合一算法

给定公式集 S={S1,S2}S=\{S_1,S_2\} 进行合一,就是对 S1,S2S_1,S_2 进行合一。

DEMO:S={P(a,x,f(g(y))),P(z,h(z,u),f(u))}S=\{P(a,x,f(g(y))),P(z,h(z,u),f(u))\}

    • δ0=ϵ\delta_0=\epsilon
    • W0=S={P(a,x,f(g(y))),P(z,h(z,u),f(u))}W_0=S=\{P(a,x,f(g(y))),P(z,h(z,u),f(u))\}
    • D0={a,z}D_0=\{a,z\}
    • δ1=a/z\delta_1=a/z
    • W1={P(a,x,f(g(y)),P(a,h(a,u),f(u))}W_1=\{P(a,x,f(g(y)),P(a,h(a,u),f(u))\}
    • W1W_1 未合一,令 D1={x,h(a,u)}D_1=\{x,h(a,u)\}
    • δ2={a/z,h(a,u)/x}\delta_2=\{a/z,h(a,u)/x\}
    • W2={P(a,h(a,u),f(g(y))),P(a,h(a,u),f(u))}W_2=\{P(a,h(a,u),f(g(y))),P(a,h(a,u),f(u))\}
    • W2W_2 未合一,令 D2={g(y),u}D_2=\{g(y),u\}
    • δ3={a/z,h(a,u)/x,u/g(y)}\delta_3=\{a/z,h(a,u)/x,u/g(y)\}
    • W3={P(a,h(a,u)f(u)),P(a,h(a,u),f(u))}W_3=\{P(a,h(a,u)f(u)),P(a,h(a,u),f(u))\}
    • W3W_3 合一。所以 δ3={a/z,h(a,u)/x,u/g(y)}\delta_3=\{a/z,h(a,u)/x,u/g(y)\} 为最一般合一。

# 归结推理解题步骤

求证一个目标 GG 时,利用归结反演:若子句集 SS 是不可满足的,则可以使用归结规则由 SS 产生空子句 NIL ,具体解题步骤:

  1. 谓词公式表示问题

    1. (optional,定义常量)

    2. 定义谓词

    3. 将前提表示成谓词公式

      • 规则:

        R1:R_1:\cdots

        R2:R_2:\cdots

        ......

      • 事实

    4. 将要求证的问题表示成谓词公式

      • G=G=\cdots (target)
  2. 问题化为子句集

    1. R1:S1=R_1:\cdots \rArr S_1=\dots

      R1:S1=R_1:\cdots \rArr S_1=\cdots

      ......

    2. 事实:Si=S_i=\cdots

    3. 目标:SG=S_{\sim G}=\cdots(⚠️注意是 G\sim G⚠️)

    4. 得到 S=S1S2SG={,}S=S_1\cup S_2\cup\cdots\cup S_{\sim G}=\{\cdots\lor\cdots,\cdots\}

  3. 利用归结原理进行归结

    (1)S1(2)S2(k)(1)(2)归结,σ={a/x,b/y}(k+1)(3)(k)归结,σ={c/u,d/v}NIL\begin{aligned} &(1) S_1\\ &(2) S_2\\ &\cdots\\ &(k)\cdots &(1)(2)\text{归结}&,\sigma=\{a/x,b/y\}\\ &(k+1)\cdots &(3)(k)\text{归结}&,\sigma=\{c/u,d/v\}\\ &\cdots\\ &\text{NIL} \end{aligned}

# 搜索策略

对子句集进行归结时,一个关键问题是决定选取哪两个子句作归结,为此需要研究有效的归结控制策略。

# 排序策略

假设原始子句 (包括待证明合式公式的否定的子句) 称为 00 层归结式。(i+1)(i+1) 层的归结式是一个 ii 层归结式和一个 j(ji)j(j≤i) 层归结式进行归结所得到的归结式。

# 宽度优先

宽度优先就是先生成第 11 层所有的归结式,然后是第 22 层所有的归结式,以此类推,直到产生空子句结束,或不能再进行归结为止。

# 深度优先

深度优先是产生一个第 11 层的归结式,然后用第 11 层的归结式和第 00 层的归结式进行归结,得到第 22 层的归结式,直到产生空子句结束,否则,用第 22 层及其以下各层进行归结,产生第 33 层,以此类推。

# 单元优先

单元优先 (unit preference) 策略,即在归结过程中优先考虑仅由一个文字构成的子句,这样的子句称为单元子句。

# 精确策略

精确策略不涉及被归结子句的排序,它们只允许某些归结发生。

# 支持集

支持集 (set of support) 策略,实际是一种目标制导的反向推理

每次归结时,参与归结的子句中至少应有一个是:

  • 由目标公式的否定得到的子句
  • 或者是它们的后裔

所谓后裔是说,如果 SiS_iSi+1S_{i+1} 与另外某子句的归结式,或者 Si+1S_{i+1}SiS_i 的后裔与其他子句的归结式,则称 Si+1S_{i+1}SiS_i 的后裔, SiS_iSi+1S_{i+1} 的祖先。

支持集策略是完备的,即假如对一个不可满足的子句集合运用支持集策略进行归结,那么最终会导出空子句。

# 线性输入

线性输人 (linear input) 策略中参与归结的两个子句中至少有一个是原始子句集中的子句 (包括那些待证明的合式公式的否定)。

线性输人策略是高效的,但是不完备的,如子句集合 {PQ,PQ,PQ,PQ}\{P\lor Q,\ P\lor \sim Q,\ \sim P\lor Q,\ \sim P\lor \sim Q\} 不可满足,但是无法用线性输入归结推出。

# 线性归结

在归结过程中,除第一次归结都用给定的子句集中的子句外,其后每次归结则至少要有一个亲本子句是上次归结的结果。线性归结策略是完备的高效的

# 祖先过滤

由于线性输入策略是不完备的,改进该策略得到祖先过滤 (ancestry filtering) 策略,该策略是完备的而且是高效的。参与归结的两个子句中:

  • 至少有一个是初始子句集中的句子
  • 或者一个子句是另一个子句的祖先

# 单元归结

不完备但是高效,与单元优先的区别在于每次只考虑单元子句进行归结。

# 例题

实际上是我的作业

# 不是兄弟

已知:

  • 规则 1: 任何人的兄弟不是女性
  • 规则 2: 任何人的姐妹必是女性
  • 事实:Mary 是 Bill 的姐妹

用归结推理方法证明 Mary 不是 Tom 的兄弟


定义谓词:

  • F(x)F(x) : xx 是女性;(female)
  • M(x)M(x) : xx 是男性;(male)
  • B(x,y)B(x,y) : xxyy 的兄弟;(brother)
  • S(x,y)S(x,y) : xxyy 的姐妹;(sister)

问题用谓词公式表示:

R1:(x)(y)(B(x,y)F(x))R2:(x)(y)(S(x,y)F(x))Fact:(S(Marry,Bill))G:B(Marry,Tom)\begin{aligned} &R_1:(\forall x)(\forall y)(B(x,y)\to\sim F(x))\\ &R_2:(\forall x)(\forall y)(S(x,y)\to F(x))\\ &Fact:(S(Marry,Bill))\\ &\rArr\\ &G:\sim B(Marry,Tom) \end{aligned}

再将问题化为子句集:

R1:S1={B(x,y)F(x)}R2:S2={S(x,y)F(x)}Fact:S3={S(Marry,Bill)}G:S4=SG={B(Marry,Tom)}S=S1S2S3S4\begin{aligned} &R_1:S_1=\{\sim B(x,y)\lor\sim F(x)\}\\ &R_2:S_2=\{\sim S(x,y)\lor F(x)\}\\ &Fact:S_3=\{S(Marry,Bill)\}\\ &\sim G:S_4=S_{\sim G}=\{B(Marry,Tom)\}\\ &S=S_1\cup S_2\cup S_3\cup S_4 \end{aligned}

归结过程:

(1)B(x,y)F(x)(2)S(x,y)F(x)(3)S(Marry,Bill)(4)B(Marry,Tom)(5)F(x)(1)(4)归结,σ={Marry/x,Tom/y}(6)S(x,y)(2)(5)归结(7)NIL(3)(6)归结,σ={Marry/x,Bill/y}\begin{aligned} (1)& \sim B(x,y)\lor\sim F(x)\\ (2)& \sim S(x,y)\lor F(x)\\ (3)& S(Marry,Bill)\\ (4)& B(Marry,Tom)\\ (5)& \sim F(x) &(1)(4)\text{归结}&,\sigma=\{Marry/x,Tom/y\}\\ (6)& \sim S(x,y) &(2)(5)\text{归结}&\\ (7)&\text{NIL}&(3)(6)\text{归结}&,\sigma=\{Marry/x,Bill/y\}\\ \end{aligned}

程序验证:

ResolutionFOLeg1

# John 偷东西

用谓词逻辑的子句集表示下述刑侦知识,并用反演归结的支持集策略证明结论。

(1) 用子句集表示下述知识。

  1. John 是贼;
  2. Paul 喜欢酒 (wine);
  3. Paul (也) 喜欢奶酪 (cheese);
  4. 如果 Paul 喜欢某物,则 John 也喜欢;
  5. 如果某人是贼,而且喜欢某物,则他就可能会偷窃该物。

(2) 求:John 可能会偷窃什么?

# 用子句集表示知识

定义常量:

  • JohnJohn
  • PaulPaul
  • winewine:酒
  • cheesecheese:奶酪

定义谓词:

  • T(x)T(x) : xx 是贼;(thief)
  • L(x,y)L(x,y) : xx 喜欢 yy ;(like)
  • S(x,y)S(x,y) : xx 可能会偷窃 yy ; (stole)

用谓词逻辑的子句集表示:

S1={T(John)}S2={L(Paul,wine)}S3={L(Paul,cheese)}S4={L(Paul,x)L(John,x)}S5={T(x)L(x,y)S(x,y)}\begin{aligned} &S_1=\{T(John)\}\\ &S_2=\{L(Paul,wine)\}\\ &S_3=\{L(Paul,cheese)\}\\ &S_4=\{\sim L(Paul,x)\lor L(John,x)\}\\ &S_5=\{\sim T(x)\lor\sim L(x,y)\lor S(x,y)\} \end{aligned}

# 求解问题

这个 ANSWER(z)ANSWER(z) 是我按照网上答案做的,实际上我觉得不需要这么麻烦,用 S(John,z)S(John,z) 就可以了...

定义谓词 ANSWER(z)ANSWER(z) : zz 是所求问题的一个答案;

则问题 "John 可能会偷窃什么?" 可以表示为 S(John,z)ANSWER(z)S(John,z)\to ANSWER(z) 也就是析取式: S(John,z)ANSWER(z)\sim S(John,z)\lor ANSWER(z)

所有可能的答案包括:

  • ANSWER(wine)ANSWER(wine)
  • ANSWER(cheese)ANSWER(cheese)

证明 G=ANSWER(wine)ANSWER(cheese)G=ANSWER(wine)\land ANSWER(cheese) 问题,化为子句集:

S1={T(John)}S2={L(Paul,wine)}S3={L(Paul,cheese)}S4={L(Paul,x)L(John,x)}S5={T(x)L(x,y)S(x,y)}S6={S(John,z)ANSWER(z)}S7=SG=ANSWER(wine)ANSWER(cheese)S=S1S7\begin{aligned} &S_1=\{T(John)\}\\ &S_2=\{L(Paul,wine)\}\\ &S_3=\{L(Paul,cheese)\}\\ &S_4=\{\sim L(Paul,x)\lor L(John,x)\}\\ &S_5=\{\sim T(x)\lor\sim L(x,y)\lor S(x,y)\}\\ &S_6=\{\sim S(John,z)\lor ANSWER(z)\}\\ &S_7=S_{\sim G}=\sim ANSWER(wine)\lor\sim ANSWER(cheese)\\ &S=S_1\cup\cdots\cup S_7 \end{aligned}

支持集策略归结过程:

(1)T(John)(2)L(Paul,wine)(3)L(Paul,cheese)(4)L(Paul,x)L(John,x)(5)T(x)L(x,y)S(x,y)(6)S(John,z)ANSWER(z)(7)ANSWER(wine)ANSWER(cheese)(8)T(John)L(John,z)ANSWER(z)(5)(6)归结,σ={John/x,z/y}(9)T(John)L(Paul,z)ANSWER(z)(4)(8)归结,σ={z/x}(10)L(Paul,z)ANSWER(z)(1)(9)归结(11)ANSWER(wine)(2)(10)归结,σ={wine/z}(12)ANSWER(cheese)(7)(11)归结(13)ANSWER(cheese)(3)(10)归结,σ={cheese/z}(14)NIL(12)(13)归结\begin{aligned} (1)& T(John)\\ (2)& L(Paul,wine)\\ (3)& L(Paul,cheese)\\ (4)& \sim L(Paul,x)\lor L(John,x)\\ (5)& \sim T(x)\lor\sim L(x,y)\lor S(x,y)\\ (6)& \sim S(John,z)\lor ANSWER(z)\\ (7)& \sim ANSWER(wine)\lor\sim ANSWER(cheese)\\ (8)& \sim T(John)\lor\sim L(John,z)\lor ANSWER(z) &(5)(6)\text{归结}&,\sigma=\{John/x,z/y\}\\ (9)& \sim T(John)\lor\sim L(Paul,z)\lor ANSWER(z) &(4)(8)\text{归结}&,\sigma=\{z/x\}\\ (10)& \sim L(Paul,z)\lor ANSWER(z) &(1)(9)\text{归结}\\ (11)& ANSWER(wine) &(2)(10)\text{归结}&,\sigma=\{wine/z\}\\ (12)& \sim ANSWER(cheese) &(7)(11)\text{归结}\\ (13)& ANSWER(cheese) &(3)(10)\text{归结}&,\sigma=\{cheese/z\}\\ (14)& \text{NIL} &(12)(13)\text{归结} \end{aligned}

可证得:John 可能会偷酒(wine)和奶酪(cheese)

程序验证:

ResolutionFOLeg2

# 小张是快乐的

任何通过了历史考试并中了彩票的人都是快乐的。任何肯学习或幸运的人都可以通过所有考试,小张不学习,但很幸运,任何人只要是幸运的,就能中彩。

求证:小张是快乐的。


定义常量;

  • aa 表示小张
  • hh 表示历史考试(history)

定义谓词:

  • P(x,y)P(x,y) : xx 可以通过考试 yy(pass)
  • F(x)F(x) : xx 是幸运的(fortune)
  • S(x)S(x) : xx 肯学习(study)
  • H(x)H(x) : xx 是快乐的(happy)
  • L(x)L(x) : xx 中了彩票(lottery)

用谓词公式表示问题:

R1:(x)(P(x,h)L(x)H(x))R2:(x)(S(x)(y)(P(x,y)))R3:(x)(F(x)(y)(P(x,y)))R4:(x)(F(x)L(x))F1:S(a)F2:F(a)\begin{aligned} & R_1:(\forall x)(P(x,h)\land L(x)\to H(x))\\ & R_2:(\forall x)(S(x)\to(\forall y)(P(x,y)))\\ & R_3:(\forall x)(F(x)\to(\forall y)(P(x,y)))\\ & R_4:(\forall x)(F(x)\to L(x))\\ & F_1:\sim S(a)\\ & F_2:F(a) \end{aligned}

求证目标为:

G:H(a)G:H(a)

反演归结,将问题表示为子句集:

S1={P(x,h)L(x)H(x)}S2={S(x)P(x,y)}S3={F(x)P(x,y)}S4={F(x)L(x)}S5={S(a)}S6={F(a)}S7=SG={H(a)}S=S1S7\begin{aligned} & S_1=\{\sim P(x,h)\lor\sim L(x)\lor H(x)\}\\ & S_2=\{\sim S(x)\lor P(x,y)\}\\ & S_3=\{\sim F(x)\lor P(x,y)\}\\ & S_4=\{\sim F(x)\lor L(x)\}\\ & S_5=\{\sim S(a)\}\\ & S_6=\{F(a)\}\\ & S_7=S_{\sim G}=\{\sim H(a)\}\\ & S=S_1\cup\cdots\cup S_7 \end{aligned}

归结推理过程;

(1)P(x,h)L(x)H(x)(2)S(x)P(x,y)(3)F(x)P(x,y)(4)F(x)L(x)(5)S(a)(6)F(a)(7)H(a)(8)P(a,y)(3)(6)归结,σ={a/x}(9)L(a)(4)(6)归结,σ={a/x}(10)L(a)H(a)(1)(8)归结,σ={a/x,h/y}(11)H(a)(9)(10)归结(12)NIL(7)(11)归结\begin{aligned} (1)& \sim P(x,h)\lor\sim L(x)\lor H(x)\\ (2)& \sim S(x)\lor P(x,y)\\ (3)& \sim F(x)\lor P(x,y)\\ (4)& \sim F(x)\lor L(x)\\ (5)& \sim S(a)\\ (6)& F(a)\\ (7)& \sim H(a)\\ (8)& P(a,y) &(3)(6)\text{归结}&,\sigma=\{a/x\}\\ (9)& L(a) &(4)(6)\text{归结}&,\sigma=\{a/x\}\\ (10)& \sim L(a)\lor H(a) &(1)(8)\text{归结}&,\sigma=\{a/x,h/y\}\\ (11)& H(a) &(9)(10)\text{归结}\\ (12)& \text{NIL} &(7)(11)\text{归结} \end{aligned}

由此证得了:小张是快乐的。

程序验证:

ResolutionFOLeg3

# 程序实现

实际上是我的作业

python
import queue
def parse(KB: str):
    KB = KB.replace("{(", "").replace(")}", "")
    items = KB.split("),(")
    resolution_steps = map(lambda x: f'{items.index(x) + 1} ({x})', items)
    clauses = []
    for item in items:
        if item.endswith(')'):
            item = item[:-1]
        clause = []
        elements = item.split("),")
        if elements[-1] == '':
            elements.pop()
        for i in range(len(elements)):
            key, value = elements[i].split("(")
            clause.append([key, value.split(',')])
        clauses.append(clause)
    return clauses, list(resolution_steps)
def MGU(l1, l2):
    variants = ('x', 'y', 'z', 'u', 'v', 'w', 'xx', 'yy', 'zz', 'uu', 'vv', 'ww')
    dimension = len(l1)
    mapdict = {}
    for i in range(dimension):
        if l1[i] in variants and l2[i] not in variants:
            if mapdict.get(l1[i]) is None or mapdict[l1[i]] == l2[i]:
                mapdict[l1[i]] = l2[i]
            else:
                return None
        elif l1[i] not in variants and l2[i] in variants:
            if mapdict.get(l2[i]) is None or mapdict[l2[i]] == l1[i]:
                mapdict[l2[i]] = l1[i]
            else:
                return None
        elif l1[i] not in variants and l2[i] not in variants and l1[i] != l2[i]:
            return None
        else:
            if l1[i] != l2[i]:
                mapdict[l1[i]] = l2[i]
    return mapdict
# 以下是广度优先搜索策略
def resolve(c1, c2):
    list_c1keys = [sublist[0] for sublist in c1]
    list_c2keys = [sublist[0] for sublist in c2]
    for i in range(len(list_c1keys)):
        c1key: str = list_c1keys[i]
        idx_c1key = i
        flag = c1key.startswith('~')
        if flag:
            complement = c1key[1:]
        else:
            complement = '~' + c1key
        if complement in list_c2keys:
            idx_complement = list_c2keys.index(complement)
            mapdict = None
            if c1[idx_c1key][1] != c2[idx_complement][1]:
                result = MGU(c1[idx_c1key][1], c2[idx_complement][1])
                if result is None:
                    return None
                else:
                    mapdict = result
            if len(c1) > 1:
                char1 = chr(ord('a') + idx_c1key)
            else:
                char1 = '0'
            if len(c2) > 1:
                char2 = chr(ord('a') + idx_complement)
            else:
                char2 = '0'
            c1_to_del = list(c1)
            c2_to_del = list(c2)
            del c1_to_del[idx_c1key]
            del c2_to_del[idx_complement]
            c1_to_del.extend(c2_to_del)
            return c1_to_del, char1, char2, mapdict
    return None
def resolution(KB: str):
    clauses, resolution_steps = parse(KB)
    steps_idx = list(range(1, len(clauses) + 1))
    min_len = len(clauses)
    while True:
        l = len(clauses)
        for i in range(len(clauses)):
            for j in range(i + 1, len(clauses)):
                if i == j:
                    continue
                result = resolve(clauses[i], clauses[j])
                if result is None:
                    continue
                resolvent = result[0]
                if resolvent is None or resolvent in clauses:
                    continue
                clauses.append(resolvent)
                steps_idx.append([i, j])
                char1 = result[1]
                char2 = result[2]
                mapdict = result[3]
                if char1 == '0' and char2 == '0':
                    pre_string = f"R[{i + 1},{j + 1}]"
                elif char1 != '0' and char2 == '0':
                    pre_string = f"R[{i + 1}{char1},{j + 1}]"
                elif char1 == '0' and char2 != '0':
                    pre_string = f"R[{i + 1},{j + 1}{char2}]"
                else:
                    pre_string = f"R[{i + 1}{char1},{j + 1}{char2}]"
                map_string = ''
                if mapdict is not None:
                    pairs = []
                    for key, value in mapdict.items():
                        pairs.append(f"{key}={value}")
                    map_string = "{" + ",".join(pairs) + "}"
                result_string = ''
                for [key, value] in resolvent:
                    result_string += f'{key}({",".join(value)}),'
                if len(resolvent) > 1:
                    result_string = result_string[:-1]
                result_string = f'({result_string})'
                resolution_steps.append(f"{len(resolution_steps) + 1} {pre_string}{map_string} = {result_string}")
                if len(resolvent) == 0:
                    return resolution_steps, steps_idx, min_len
        if l == len(clauses):
            return None
def simplify_steps(steps, steps_idx, min_len):
    q = queue.Queue()
    not_del = [len(steps) - 1]
    q.put(steps_idx[len(steps) - 1][0])
    q.put(steps_idx[len(steps) - 1][1])
    while not q.empty():
        i = q.get()
        if i < min_len:
            continue
        not_del.append(i)
        q.put(steps_idx[i][0])
        q.put(steps_idx[i][1])
    for i in range(len(steps) - 1, min_len - 1, -1):
        if i not in not_del:
            del steps[i]
    num_map = dict()
    for i in range(len(steps)):
        num = (steps[i].split(' '))[0]
        num_map[num] = f'{i+1}'
        for t in list(num_map.keys()):
            steps[i] = steps[i].replace(t, num_map[t])
    return steps
def ResolutionFOL(KB):
    result = resolution(KB)
    if result is None:
        print("Can't be resolved!")
    else:
        (steps, steps_idx, min_len) = result
        steps = simplify_steps(steps, steps_idx, min_len)
        for step in steps:
            print(step)
if __name__ == '__main__':
    KB0 = "{(GradStudent(sue),),(~GradStudent(x),Student(x)),(~Student(x),HardWorker(x)),(~HardWorker(sue),)}"
    KB1 = "{(A(tony),),(A(mike),),(A(john),),(L(tony,rain),),(L(tony,snow),),(~A(x),S(x),C(x)),(~C(y),~L(y,rain)),(L(z,snow),~S(z)),(~L(tony,u),~L(mike,u)),(L(tony,v),L(mike,v)),(~A(w),~C(w),S(w))}"
    KB2 = "{(On(tony,mike),),(On(mike,john),),(Green(tony),),(~Green(john),),(~On(xx,yy),~Green(xx),Green(yy))}"
    KB3 = "{(~B(x,y),~F(x)),(~S(x,y),F(x)),(S(Marry,Bill)),(B(Marry,Tom))}"
    KB4 = "{(T(John),),(L(Paul,wine),),(L(Paul,cheese),),(~L(Paul,x),L(John,x)),(~T(x),~L(x,y),S(x,y)),(~S(John,z),ANSWER(z)),(~ANSWER(wine),~ANSWER(cheese))}"
    KB5 = "{(~P(x,h),~L(x),H(x)),(~S(x),P(x,y)),(~F(x),P(x,y)),(~F(x),L(x)),(~S(a)),(F(a)),(~H(a))}"
    ResolutionFOL(KB0)
    print("\x1B[31m--------------------\x1B[m")
    ResolutionFOL(KB1)
    print("\x1B[31m--------------------\x1B[m")
    ResolutionFOL(KB2)
    print("\x1B[31m--------------------\x1B[m")
    ResolutionFOL(KB3) # 不是兄弟
    print("\x1B[31m--------------------\x1B[m")
    ResolutionFOL(KB4) # John 偷东西
    print("\x1B[31m--------------------\x1B[m")
    ResolutionFOL(KB5) # 小张是快乐的
# 以下为单元归结策略
def resolve_2(c1, c2):
    list_c1keys = [sublist[0] for sublist in c1]
    list_c2keys = [sublist[0] for sublist in c2]
    c1key: str = list_c1keys[0]
    idx_c1key = list_c1keys.index(c1key)
    flag = c1key.startswith('~')
    if flag:
        complement = c1key[1:]
    else:
        complement = '~' + c1key
    if complement in list_c2keys:
        idx_complement = list_c2keys.index(complement)
        mapdict = None
        if c1[idx_c1key][1] != c2[idx_complement][1]:
            result = MGU(c1[idx_c1key][1], c2[idx_complement][1])
            if result is None:
                return None
            else:
                mapdict = result
        if len(c1) > 1:
            char1 = chr(ord('a') + idx_c1key)
        else:
            char1 = '0'
        if len(c2) > 1:
            char2 = chr(ord('a') + idx_complement)
        else:
            char2 = '0'
        c2_to_del = list(c2)
        del c2_to_del[idx_complement]
        return c2_to_del, char1, char2, mapdict
    return None
def resolution_2(KB: str):
    clauses, resolution_steps = parse(KB)
    singles = set()
    for i in range(len(clauses)):
        if len(clauses[i]) == 1:
            singles.add(i)
    while True:
        l = len(clauses)
        for i in range(len(clauses)):
            if i not in singles:
                continue
            for j in range(len(clauses)):
                if i == j:
                    continue
                result = resolve_2(clauses[i], clauses[j])
                if result is None:
                    continue
                resolvent = result[0]
                if resolvent is None or resolvent in clauses:
                    continue
                clauses.append(resolvent)
                char1 = result[1]
                char2 = result[2]
                mapdict = result[3]
                if char1 == '0' and char2 == '0':
                    pre_string = f"R[{i + 1},{j + 1}]"
                elif char1 != '0' and char2 == '0':
                    pre_string = f"R[{i + 1}{char1},{j + 1}]"
                elif char1 == '0' and char2 != '0':
                    pre_string = f"R[{i + 1},{j + 1}{char2}]"
                else:
                    pre_string = f"R[{i + 1}{char1},{j + 1}{char2}]"
                map_string = ''
                if mapdict is not None:
                    pairs = []
                    for key, value in mapdict.items():
                        pairs.append(f"{key}={value}")
                    map_string = "{" + ",".join(pairs) + "}"
                result_string = ''
                for [key, value] in resolvent:
                    result_string += f'{key}({",".join(value)}),'
                if len(resolvent) > 1:
                    result_string = result_string[:-1]
                result_string = f'({result_string})'
                resolution_steps.append(f"{len(resolution_steps) + 1} {pre_string}{map_string} = {result_string}")
                if len(resolvent) == 0:
                    return resolution_steps
                elif len(resolvent) == 1:
                    singles.add(len(clauses) - 1)
        if l == len(clauses):
            return None
def ResolutionFOL_2(KB):
    result = resolution_2(KB)
    if result is None:
        print("Can't be resolved!")
    else:
        for step in result:
            print(step)
\ No newline at end of file +知识表示与推理 - 人工智能 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

# 知识和知识表示

# 知识

  • 数据一般指单独的事实,是信息的载体
  • 信息由符号组成,如文字和数字,但是对符号赋予了一定的意义,因此有一定的用途或价值
  • 知识是由经验总结升华出来的,因此知识是经验的结晶。知识在信息的基础上增加了上下文信息,提供了更多的意义,因此也就更加有用和有价值。知识是随着时间的变化而动态变化的,新的知识可以根据规则和已有的知识推导出来。

知识是经过加工的信息,它包括:

  • 事实:是关于对象和物体的知识。
  • 规则:是有关问题中与事物的行动、动作相联系的因果关系的知识。
  • 元知识:是有关知识的知识,是知识库中的高层知识。
  • 常识性知识:泛指普遍存在而且被普遍认识了的客观事实一类知识。

# 策略

策略 —— 关于如何解决问题的政策方略,包括在什么时间、什么地点、由什么主体采取什么行动、达到什么目标、注意什么事项等等一整套完整而具体的行动计划规划、行动步骤、工作方式和工作方法

# 智能

在给定的 问题 - 问题环境 - 主体目的 的条件下,智能就是有针对性地获取 问题 - 环境 的信息,恰当地对这些信息进行处理以提炼知识达到认知,在此基础上,把已有的知识与主体的目的信息相结合,合理地产生解决问题的策略信息,并利用所得到的策略信息在给定的环境下成功地解决问题达到主体的目的 (行为)。

省流:信息经加工提炼而成知识,知识被目的激活而成智能。

智能具有 4 个能力:

  • 获取有用信息的能力
  • 由信息生成 == 知识 (认知)== 的能力
  • 由知识和目的生成 == 策略 (决策)== 的能力
  • 实施策略取得效果 (施效) 的能力

# 知识表示语言问题

  • 语法 —— 语言的语法描述了组成语句的可能的搭配关系。
  • 语义 —— 语义定义了语句所指的世界中的事实。

从语法和语义,可以给出使用该语言的 Agent 的必要的推理机制。

基于该推理机制,Agent 可以从已知的语句推导出结论,或判断某条信息是不是已蕴涵在现有的知识当中。

知识表示语言需要:

  • 语法规则
  • 语义解释
  • 用于演绎和推导的规则

知识表示语言应该支持知识不完全的情况。不能表达这种不完全性的语言是表达能力不够的语言

程序设计语言比较善于描述算法和具体的数据结构

# 现代逻辑学的基本研究方法

逻辑学 (logic) 是研究人类思维规律的科学,而现代逻辑学则是用数学 (符号化、公理化、形式化) 的方法来研究这些规律。

# 符号化

所谓符号化即是用 “一种只作整体认读的记号 (signs)”—— 符号 (symbols) 表示量、数及数量关系。也有用字母、符号表示思维的物理对象、概念对象、判断对象等。

语言化是符号化的初级阶段。

现代逻辑学对思维的研究,需要更加彻底的符号化过程。

# 公理化

欧氏几何公理系统中的所有概念都有鲜明的直观背景,其公理、定理也都有强烈的客观意义。像欧氏几何这样的公理系统,常被称为具体公理系统。

始于 Aristotle 的逻辑学被符号化、公理化,逐步演化为现代逻辑学。

# 形式化

所谓形式化,就是彻头彻尾的 “符号化 + 抽象公理化

现代逻辑学形式系统组成:

  • 用于将概念符号化的符号语言,通常为一形式语言,包括一符号表∑及语言的文法,可生成表示对象的语言成分项,表示概念、判断的公式;
  • 表示思维规律的逻辑学公理模式和推理规则模式 (抽象公理系统),及其依据它们推演可得到的全部定理组成的理论体系。

# 命题逻辑

以下讨论实际上均为二值逻辑

# 基本概念

# 命题

命题是具有真假意义的陈述句。

在特殊的情况下都具有 “真 (True)” 和 “假(False)” 的意义句子,都是命题。真值 —— 用 T 和 F 表示。

命题有两种类型:

  • 原子命题
  • 复合命题 —— 由以下内容复合构成的命题:
    • 联结词
    • 标点符号
    • 原子命题

# 命题逻辑

命题逻辑就是研究命题和命题之间关系的符号逻辑系统。

# 命题逻辑的符号

命题逻辑的符号:

  • 命题常元:True (T) 和 False (F)
  • 命题标识符:P、Q、R 等;
  • 联结词:
    • ¬(否定)
    • \land(合取)
    • \lor (析取)
    • →(蕴含)
    • ↔︎(等价)
  • 括号:( )

# 命题标识符

用 P、Q、R、S 等来表示命题。可分为:

  • 命题常量(表示确定的命题的命题标识符)
  • 命题变元(只表示任意命题的位置标志的命题标识符)
    • 当命题变元表示原子命题时,该变元称为原子变元

因为命题变元可以表示任意命题,所以它不能确定真值,故命题变元不是命题。

# 联结词

复合命题的意义是命题组成成份的函数。

联结词的语义可以定义如下:

  • ¬P 为真,当且仅当 P 为假
  • PQP \land Q 为真,当且仅当 P 和 Q 都为真
  • PQP \lor Q 为真,当且仅当 P 为真,或者 Q 为真
  • P → Q 为真,当且仅当 P 为假,或者 Q 为真
  • P ↔︎ Q 为真,当且仅当 P→Q 为真,并且 Q→P 为真

# 指派

当命题变元 P 用一个特定的命题取代时,P 才能确定真值,这时也称为对命题变元 P 进行指派

设 G 是公式,A1, ... , An, 为 G 中出现的所有原子命题。公式 G 的一种指派是对 A1, ... , An 赋予的一组真值,其中每个 Ai (i=1 , ... , n) 或者为 T 或者为 F。

公式 G 称为在一种指派下为真,当且仅当 G 按该指派算出的真值为 T,否则称为在该指派下为假

若在公式中有 n 个不同的原子 A1 , ... , An,那么该公式就有 2n2^n 个不同的指派。

# 永真 / 可满足 / 永假

公式 AA 称为永真式重言式 (tautology),如果对任意指派 α,α 均弄真 A,即 α(A)=T 。

公式 AA 称为可满足的 (satisfiable), 如果存在指派 α 使 α(A)=T,否则称 AA不可满足的 (unsatisfiable),或永假式

永真式是可满足的

AA 为永真式 (永假式) 时, ¬A\neg A 为永假式 (永真式)

# 蕴含 / 等价

称公式 AA 逻辑蕴涵公式 B,记为 ABA\to B ,如果所有弄真 AA 的指派亦必弄真公式 B

称公式集 Γ\Gamma 逻辑蕴涵公式 B,记为 ΓB\Gamma \to B ,如果弄真 Γ\Gamma 中所有公式的指派亦必弄真公式 B

称公式 AA 逻辑等价公式 B,记为 ABA\lrarr B ,如果 ABA\to BBAB\to A

  • 逻辑蕴涵关系具有
    • 自反性
    • 反对称性
    • 传递性
  • 逻辑等价关系具有
    • 自反性
    • 对称性
    • 传递性

# 代换

设 A 为含有命题变元 p 的永真式,那么将 A 中 p 的所有出现均代换为命题公式 B,所得公式 (称为 A 的代入实例) 仍为永真式。

# 替换

设命题公式 AA 含有子公式 C (C 为 A 中的符号串,且 C 为命题公式),如果 CDC\lrarr D,那么将 AA 中子公式 C 的某些出现 (未必全部) 用 D 替换后所得公式 B 满足 ABA\lrarr B

# 合 / 析 取范式

命题公式 B 称为命题公式 AA合取(或析取范式,如果 BAB\lrarr A,且 B 呈如下形式:

C1C2Cm(C1C2Cm),Ci(i=1,2,...,m)C_1\land C_2\land …\land C_m(或 C_1\lor C_2\lor …\lor C_m)\ \ ,\ \ C_i (i=1, 2,...,m)

其中 CiC_i 形如:

L1L2...Ln(L1L2Ln),Lj(j=l,2,...,n)L_1\lor L_2\lor ...\lor L_n (或L_1\land L_2\land …\land L_n)\ \ ,\ \ L_j(j=l,2,...,n)

其中 LiL_i 为原子公式或原子公式的否定,并称 LjL_j文字

【定理】

任一命题公式 φ 有其对应的合取 (析取) 范式。

# 主 合 / 析 取范式

命题公式 B 称为公式 A 的主合取 (或主析取) 范式,如果:

  • B 是 AA 的合取 (或析取) 范式
  • B 中每一子句均有 AA 中命题变元的全部出现,且仅出现一次。

n 元命题公式的全体可以划分为 22n2^{2^n} 个等价类,每一类中的公式彼此逻辑等价,并等价于它们共同的主合取范式 (或主析取范式)。

# 命题演算形式系统 PC

# 合式公式

pl,p2,p3,...p_l,p_2,p_3,... 为命题逻辑的合式公式

如果 A,B 是公式,那么这些也是命题逻辑的合式公式:

  • (¬A)(\neg A)
  • (AB)(A→B)

# 形式系统 PC 的组成

命题逻辑的形式系统 PC 包括 3 条公理模式 (A1A3)(A_1-A_3) 和 1 条推理规则(即分离规则rmpr_{mp}

  • A1:A(BA)A_1 : A\to(B\to A)
  • A2:A(BC)(AB)(AC)A_2 : A\to(B\to C)\to (A\to B)\to(A\to C)
  • A3:(¬A¬B)(BA)A_3 : (\neg A\to\neg B)\to(B\to A)
  • rmp:A,ABBr_{mp}:\frac{A,A\to B}{B}

# 证明

称下列公式序列为公式 AA 在 PC 中的一个证明 (proof):

A1,A2,...,Am(=A)A_1,A_2,...,A_m(=A)

其中 Ai(i=1,2,...,m)A_i (i=1,2,...,m)

  • 或者是 PC 的公理
  • 或者是 Aj(j<i)A_j (j\lt i)
  • 或者是由 Aj,Ak(j,k<i)A_j,A_k (j,k\lt i) 使用分离规则所导出
  • $A_m =A $

# 定理

AA 为 PC 中的定理,记为 PCA\vdash_{PC} A,如果公式 AA 在 PC 中有一个证明。

\vdash ( \vdash ) 断定符,表示可被推导出,可被证明

A\vDash A ( \vDash ) 表示 AA 永真

# 演绎

Γ\Gamma 为一公式集,称以下公式序列为公式 AA 的,以 Γ\Gamma 为前提的演绎

A1,A2,...,Am=A,Ai(i=1,2,...,m)A_1,A_2,...,A_m=A\ \ ,\ \ A_i (i=1,2,...,m)

其中 AiA_i 满足:

  • 或者是 PC 的公理
  • 或者是 Γ\Gamma 的成员
  • 或者是 Aj(j<i)A_j (j<i)
  • 或者是由 Aj,Ak(j,k<i)A_j, A_k (j,k\lt i) 使用分离规则所导出
  • Am=AA_m=A

# 带前提的演绎

AA 为前提 Γ\Gamma 的演绎结果,如果公式 AA 有以 Γ\Gamma 为前提的演绎。记为:

ΓPCA\Gamma\vdash_{PC}A

Γ={B}\Gamma=\{B\}ΓPCA\Gamma\vdash_{PC}A 可以表示为:

BPCAB\vdash_{PC}A

BPCAB\vdash_{PC}AAPCBA\vdash_{PC}B 则记为:

ABA\vdash\dashv B

# 演绎定理

对应离散数学课程中的 “附加前提证明法”

对 PC 中任意公式集 Γ\Gamma 和公式 A,BA,B

Γ{A}PCBΓPC(AB)\Gamma\cup\{A\}\vdash_{PC}B\lrArr\Gamma\vdash_{PC}(A\to B)

# 可靠性定理

PC 是可靠的,即对任意公式集 Γ\Gamma 及公式 AA

ΓAΓA\Gamma\vdash A\to \Gamma\vDash A

特别地,若 AA 为 PC 的定理 (A)(\vdash A) ,则 AA 永真 (A)(\vDash A) :

AA\vdash A\to \vDash A

# 一致性定理

PC 是一致的 (consistent),即不存在公式 A,使得 AA¬A\neg A 均为 PC 之定理。

# 完全性定理

PC 是完全的,即对任意公式集 Γ\Gamma 和公式 AA

ΓAΓA\Gamma\vDash A\to \Gamma\vdash A

特别地,若 AA 永真 (A)(\vDash A) ,A 必为 PC 的定理 (A)(\vdash A)

AA\vDash A\to \vdash A

# 谓词逻辑

谓词逻辑的语法元素表示如下。

  • 常量符号: A、B、张三、李四 等,通常是对象名称
  • 变量符号:通常用小写字母表示,如 x、 y、z 等
  • 函数符号:通常用小写英文字母或小写英文字母串表示,如 plus、f、g
    • 一元函数、二元函数、...... n 元函数
  • 谓词符号:通常用大写英文字母或 (首字母) 大写英文字母串表示
    • 一元谓词、二元谓词、...... n 元谓词
  • 联结词: ......
  • 量词:全称量词 \forall、存在量詞 \existx\forall x 表示 “対个体域中所有 xx”、x\exist x 表示 “在个 体域中存在个体 xx ”。\forall\exist 后面跟的 xx 叫作量词的指导变元

一般一元谓词表达了个体的性质,而多元谓词表达了个体之间的关系

# 谓词的阶

如果谓词 P 中的所有个体都是:

  • 个体常量
  • 变元
  • 函数

则该谓词为一阶谓词,如果某个个体本身又是一个一阶谓词,则称 PP二阶谓词,以此类推递归定义 nn 阶谓词

#

项可递归定义如下:

  1. 单独一个个体是项 (包括常量和变量)
  2. ffnn 元函数符号,而 t1,t2,...,tnt_1,t_2,...,t_n 是项,则 f(t1,t2,...,tn)f(t_1,t_2,...,t_n) 是项
  3. 任何项仅由规则 (1)、(2) 生成。

# 原子公式

PPnn 元谓词符号,t1,t2,...,tnt_1,t_2,...,t_n 都是项,则称 P(t1,t2,...,tn)P(t_1,t_2,...,t_n) 为原子公式。在原子公式中,若 t1,t2,...,tnt_1,t_2,...,t_n 都不含变量,则 P(t1,t2,...,tn)P(t_1,t_2,...,t_n) 是命题。

# 归结推理

# 最一般合一算法

给定公式集 S={S1,S2}S=\{S_1,S_2\} 进行合一,就是对 S1,S2S_1,S_2 进行合一。

DEMO:S={P(a,x,f(g(y))),P(z,h(z,u),f(u))}S=\{P(a,x,f(g(y))),P(z,h(z,u),f(u))\}

    • δ0=ϵ\delta_0=\epsilon
    • W0=S={P(a,x,f(g(y))),P(z,h(z,u),f(u))}W_0=S=\{P(a,x,f(g(y))),P(z,h(z,u),f(u))\}
    • D0={a,z}D_0=\{a,z\}
    • δ1=a/z\delta_1=a/z
    • W1={P(a,x,f(g(y)),P(a,h(a,u),f(u))}W_1=\{P(a,x,f(g(y)),P(a,h(a,u),f(u))\}
    • W1W_1 未合一,令 D1={x,h(a,u)}D_1=\{x,h(a,u)\}
    • δ2={a/z,h(a,u)/x}\delta_2=\{a/z,h(a,u)/x\}
    • W2={P(a,h(a,u),f(g(y))),P(a,h(a,u),f(u))}W_2=\{P(a,h(a,u),f(g(y))),P(a,h(a,u),f(u))\}
    • W2W_2 未合一,令 D2={g(y),u}D_2=\{g(y),u\}
    • δ3={a/z,h(a,u)/x,u/g(y)}\delta_3=\{a/z,h(a,u)/x,u/g(y)\}
    • W3={P(a,h(a,u)f(u)),P(a,h(a,u),f(u))}W_3=\{P(a,h(a,u)f(u)),P(a,h(a,u),f(u))\}
    • W3W_3 合一。所以 δ3={a/z,h(a,u)/x,u/g(y)}\delta_3=\{a/z,h(a,u)/x,u/g(y)\} 为最一般合一。

# 归结推理解题步骤

求证一个目标 GG 时,利用归结反演:若子句集 SS 是不可满足的,则可以使用归结规则由 SS 产生空子句 NIL ,具体解题步骤:

  1. 谓词公式表示问题

    1. (optional,定义常量)

    2. 定义谓词

    3. 将前提表示成谓词公式

      • 规则:

        R1:R_1:\cdots

        R2:R_2:\cdots

        ......

      • 事实

    4. 将要求证的问题表示成谓词公式

      • G=G=\cdots (target)
  2. 问题化为子句集

    1. R1:S1=R_1:\cdots \rArr S_1=\dots

      R1:S1=R_1:\cdots \rArr S_1=\cdots

      ......

    2. 事实:Si=S_i=\cdots

    3. 目标:SG=S_{\sim G}=\cdots(⚠️注意是 G\sim G⚠️)

    4. 得到 S=S1S2SG={,}S=S_1\cup S_2\cup\cdots\cup S_{\sim G}=\{\cdots\lor\cdots,\cdots\}

  3. 利用归结原理进行归结

    (1)S1(2)S2(k)(1)(2)归结,σ={a/x,b/y}(k+1)(3)(k)归结,σ={c/u,d/v}NIL\begin{aligned} &(1) S_1\\ &(2) S_2\\ &\cdots\\ &(k)\cdots &(1)(2)\text{归结}&,\sigma=\{a/x,b/y\}\\ &(k+1)\cdots &(3)(k)\text{归结}&,\sigma=\{c/u,d/v\}\\ &\cdots\\ &\text{NIL} \end{aligned}

# 搜索策略

对子句集进行归结时,一个关键问题是决定选取哪两个子句作归结,为此需要研究有效的归结控制策略。

# 排序策略

假设原始子句 (包括待证明合式公式的否定的子句) 称为 00 层归结式。(i+1)(i+1) 层的归结式是一个 ii 层归结式和一个 j(ji)j(j≤i) 层归结式进行归结所得到的归结式。

# 宽度优先

宽度优先就是先生成第 11 层所有的归结式,然后是第 22 层所有的归结式,以此类推,直到产生空子句结束,或不能再进行归结为止。

# 深度优先

深度优先是产生一个第 11 层的归结式,然后用第 11 层的归结式和第 00 层的归结式进行归结,得到第 22 层的归结式,直到产生空子句结束,否则,用第 22 层及其以下各层进行归结,产生第 33 层,以此类推。

# 单元优先

单元优先 (unit preference) 策略,即在归结过程中优先考虑仅由一个文字构成的子句,这样的子句称为单元子句。

# 精确策略

精确策略不涉及被归结子句的排序,它们只允许某些归结发生。

# 支持集

支持集 (set of support) 策略,实际是一种目标制导的反向推理

每次归结时,参与归结的子句中至少应有一个是:

  • 由目标公式的否定得到的子句
  • 或者是它们的后裔

所谓后裔是说,如果 SiS_iSi+1S_{i+1} 与另外某子句的归结式,或者 Si+1S_{i+1}SiS_i 的后裔与其他子句的归结式,则称 Si+1S_{i+1}SiS_i 的后裔, SiS_iSi+1S_{i+1} 的祖先。

支持集策略是完备的,即假如对一个不可满足的子句集合运用支持集策略进行归结,那么最终会导出空子句。

# 线性输入

线性输人 (linear input) 策略中参与归结的两个子句中至少有一个是原始子句集中的子句 (包括那些待证明的合式公式的否定)。

线性输人策略是高效的,但是不完备的,如子句集合 {PQ,PQ,PQ,PQ}\{P\lor Q,\ P\lor \sim Q,\ \sim P\lor Q,\ \sim P\lor \sim Q\} 不可满足,但是无法用线性输入归结推出。

# 线性归结

在归结过程中,除第一次归结都用给定的子句集中的子句外,其后每次归结则至少要有一个亲本子句是上次归结的结果。线性归结策略是完备的高效的

# 祖先过滤

由于线性输入策略是不完备的,改进该策略得到祖先过滤 (ancestry filtering) 策略,该策略是完备的而且是高效的。参与归结的两个子句中:

  • 至少有一个是初始子句集中的句子
  • 或者一个子句是另一个子句的祖先

# 单元归结

不完备但是高效,与单元优先的区别在于每次只考虑单元子句进行归结。

# 例题

实际上是我的作业

# 不是兄弟

已知:

  • 规则 1: 任何人的兄弟不是女性
  • 规则 2: 任何人的姐妹必是女性
  • 事实:Mary 是 Bill 的姐妹

用归结推理方法证明 Mary 不是 Tom 的兄弟


定义谓词:

  • F(x)F(x) : xx 是女性;(female)
  • M(x)M(x) : xx 是男性;(male)
  • B(x,y)B(x,y) : xxyy 的兄弟;(brother)
  • S(x,y)S(x,y) : xxyy 的姐妹;(sister)

问题用谓词公式表示:

R1:(x)(y)(B(x,y)F(x))R2:(x)(y)(S(x,y)F(x))Fact:(S(Marry,Bill))G:B(Marry,Tom)\begin{aligned} &R_1:(\forall x)(\forall y)(B(x,y)\to\sim F(x))\\ &R_2:(\forall x)(\forall y)(S(x,y)\to F(x))\\ &Fact:(S(Marry,Bill))\\ &\rArr\\ &G:\sim B(Marry,Tom) \end{aligned}

再将问题化为子句集:

R1:S1={B(x,y)F(x)}R2:S2={S(x,y)F(x)}Fact:S3={S(Marry,Bill)}G:S4=SG={B(Marry,Tom)}S=S1S2S3S4\begin{aligned} &R_1:S_1=\{\sim B(x,y)\lor\sim F(x)\}\\ &R_2:S_2=\{\sim S(x,y)\lor F(x)\}\\ &Fact:S_3=\{S(Marry,Bill)\}\\ &\sim G:S_4=S_{\sim G}=\{B(Marry,Tom)\}\\ &S=S_1\cup S_2\cup S_3\cup S_4 \end{aligned}

归结过程:

(1)B(x,y)F(x)(2)S(x,y)F(x)(3)S(Marry,Bill)(4)B(Marry,Tom)(5)F(x)(1)(4)归结,σ={Marry/x,Tom/y}(6)S(x,y)(2)(5)归结(7)NIL(3)(6)归结,σ={Marry/x,Bill/y}\begin{aligned} (1)& \sim B(x,y)\lor\sim F(x)\\ (2)& \sim S(x,y)\lor F(x)\\ (3)& S(Marry,Bill)\\ (4)& B(Marry,Tom)\\ (5)& \sim F(x) &(1)(4)\text{归结}&,\sigma=\{Marry/x,Tom/y\}\\ (6)& \sim S(x,y) &(2)(5)\text{归结}&\\ (7)&\text{NIL}&(3)(6)\text{归结}&,\sigma=\{Marry/x,Bill/y\}\\ \end{aligned}

程序验证:

ResolutionFOLeg1

# John 偷东西

用谓词逻辑的子句集表示下述刑侦知识,并用反演归结的支持集策略证明结论。

(1) 用子句集表示下述知识。

  1. John 是贼;
  2. Paul 喜欢酒 (wine);
  3. Paul (也) 喜欢奶酪 (cheese);
  4. 如果 Paul 喜欢某物,则 John 也喜欢;
  5. 如果某人是贼,而且喜欢某物,则他就可能会偷窃该物。

(2) 求:John 可能会偷窃什么?

# 用子句集表示知识

定义常量:

  • JohnJohn
  • PaulPaul
  • winewine:酒
  • cheesecheese:奶酪

定义谓词:

  • T(x)T(x) : xx 是贼;(thief)
  • L(x,y)L(x,y) : xx 喜欢 yy ;(like)
  • S(x,y)S(x,y) : xx 可能会偷窃 yy ; (stole)

用谓词逻辑的子句集表示:

S1={T(John)}S2={L(Paul,wine)}S3={L(Paul,cheese)}S4={L(Paul,x)L(John,x)}S5={T(x)L(x,y)S(x,y)}\begin{aligned} &S_1=\{T(John)\}\\ &S_2=\{L(Paul,wine)\}\\ &S_3=\{L(Paul,cheese)\}\\ &S_4=\{\sim L(Paul,x)\lor L(John,x)\}\\ &S_5=\{\sim T(x)\lor\sim L(x,y)\lor S(x,y)\} \end{aligned}

# 求解问题

这个 ANSWER(z)ANSWER(z) 是我按照网上答案做的,实际上我觉得不需要这么麻烦,用 S(John,z)S(John,z) 就可以了...

定义谓词 ANSWER(z)ANSWER(z) : zz 是所求问题的一个答案;

则问题 "John 可能会偷窃什么?" 可以表示为 S(John,z)ANSWER(z)S(John,z)\to ANSWER(z) 也就是析取式: S(John,z)ANSWER(z)\sim S(John,z)\lor ANSWER(z)

所有可能的答案包括:

  • ANSWER(wine)ANSWER(wine)
  • ANSWER(cheese)ANSWER(cheese)

证明 G=ANSWER(wine)ANSWER(cheese)G=ANSWER(wine)\land ANSWER(cheese) 问题,化为子句集:

S1={T(John)}S2={L(Paul,wine)}S3={L(Paul,cheese)}S4={L(Paul,x)L(John,x)}S5={T(x)L(x,y)S(x,y)}S6={S(John,z)ANSWER(z)}S7=SG=ANSWER(wine)ANSWER(cheese)S=S1S7\begin{aligned} &S_1=\{T(John)\}\\ &S_2=\{L(Paul,wine)\}\\ &S_3=\{L(Paul,cheese)\}\\ &S_4=\{\sim L(Paul,x)\lor L(John,x)\}\\ &S_5=\{\sim T(x)\lor\sim L(x,y)\lor S(x,y)\}\\ &S_6=\{\sim S(John,z)\lor ANSWER(z)\}\\ &S_7=S_{\sim G}=\sim ANSWER(wine)\lor\sim ANSWER(cheese)\\ &S=S_1\cup\cdots\cup S_7 \end{aligned}

支持集策略归结过程:

(1)T(John)(2)L(Paul,wine)(3)L(Paul,cheese)(4)L(Paul,x)L(John,x)(5)T(x)L(x,y)S(x,y)(6)S(John,z)ANSWER(z)(7)ANSWER(wine)ANSWER(cheese)(8)T(John)L(John,z)ANSWER(z)(5)(6)归结,σ={John/x,z/y}(9)T(John)L(Paul,z)ANSWER(z)(4)(8)归结,σ={z/x}(10)L(Paul,z)ANSWER(z)(1)(9)归结(11)ANSWER(wine)(2)(10)归结,σ={wine/z}(12)ANSWER(cheese)(7)(11)归结(13)ANSWER(cheese)(3)(10)归结,σ={cheese/z}(14)NIL(12)(13)归结\begin{aligned} (1)& T(John)\\ (2)& L(Paul,wine)\\ (3)& L(Paul,cheese)\\ (4)& \sim L(Paul,x)\lor L(John,x)\\ (5)& \sim T(x)\lor\sim L(x,y)\lor S(x,y)\\ (6)& \sim S(John,z)\lor ANSWER(z)\\ (7)& \sim ANSWER(wine)\lor\sim ANSWER(cheese)\\ (8)& \sim T(John)\lor\sim L(John,z)\lor ANSWER(z) &(5)(6)\text{归结}&,\sigma=\{John/x,z/y\}\\ (9)& \sim T(John)\lor\sim L(Paul,z)\lor ANSWER(z) &(4)(8)\text{归结}&,\sigma=\{z/x\}\\ (10)& \sim L(Paul,z)\lor ANSWER(z) &(1)(9)\text{归结}\\ (11)& ANSWER(wine) &(2)(10)\text{归结}&,\sigma=\{wine/z\}\\ (12)& \sim ANSWER(cheese) &(7)(11)\text{归结}\\ (13)& ANSWER(cheese) &(3)(10)\text{归结}&,\sigma=\{cheese/z\}\\ (14)& \text{NIL} &(12)(13)\text{归结} \end{aligned}

可证得:John 可能会偷酒(wine)和奶酪(cheese)

程序验证:

ResolutionFOLeg2

# 小张是快乐的

任何通过了历史考试并中了彩票的人都是快乐的。任何肯学习或幸运的人都可以通过所有考试,小张不学习,但很幸运,任何人只要是幸运的,就能中彩。

求证:小张是快乐的。


定义常量;

  • aa 表示小张
  • hh 表示历史考试(history)

定义谓词:

  • P(x,y)P(x,y) : xx 可以通过考试 yy(pass)
  • F(x)F(x) : xx 是幸运的(fortune)
  • S(x)S(x) : xx 肯学习(study)
  • H(x)H(x) : xx 是快乐的(happy)
  • L(x)L(x) : xx 中了彩票(lottery)

用谓词公式表示问题:

R1:(x)(P(x,h)L(x)H(x))R2:(x)(S(x)(y)(P(x,y)))R3:(x)(F(x)(y)(P(x,y)))R4:(x)(F(x)L(x))F1:S(a)F2:F(a)\begin{aligned} & R_1:(\forall x)(P(x,h)\land L(x)\to H(x))\\ & R_2:(\forall x)(S(x)\to(\forall y)(P(x,y)))\\ & R_3:(\forall x)(F(x)\to(\forall y)(P(x,y)))\\ & R_4:(\forall x)(F(x)\to L(x))\\ & F_1:\sim S(a)\\ & F_2:F(a) \end{aligned}

求证目标为:

G:H(a)G:H(a)

反演归结,将问题表示为子句集:

S1={P(x,h)L(x)H(x)}S2={S(x)P(x,y)}S3={F(x)P(x,y)}S4={F(x)L(x)}S5={S(a)}S6={F(a)}S7=SG={H(a)}S=S1S7\begin{aligned} & S_1=\{\sim P(x,h)\lor\sim L(x)\lor H(x)\}\\ & S_2=\{\sim S(x)\lor P(x,y)\}\\ & S_3=\{\sim F(x)\lor P(x,y)\}\\ & S_4=\{\sim F(x)\lor L(x)\}\\ & S_5=\{\sim S(a)\}\\ & S_6=\{F(a)\}\\ & S_7=S_{\sim G}=\{\sim H(a)\}\\ & S=S_1\cup\cdots\cup S_7 \end{aligned}

归结推理过程;

(1)P(x,h)L(x)H(x)(2)S(x)P(x,y)(3)F(x)P(x,y)(4)F(x)L(x)(5)S(a)(6)F(a)(7)H(a)(8)P(a,y)(3)(6)归结,σ={a/x}(9)L(a)(4)(6)归结,σ={a/x}(10)L(a)H(a)(1)(8)归结,σ={a/x,h/y}(11)H(a)(9)(10)归结(12)NIL(7)(11)归结\begin{aligned} (1)& \sim P(x,h)\lor\sim L(x)\lor H(x)\\ (2)& \sim S(x)\lor P(x,y)\\ (3)& \sim F(x)\lor P(x,y)\\ (4)& \sim F(x)\lor L(x)\\ (5)& \sim S(a)\\ (6)& F(a)\\ (7)& \sim H(a)\\ (8)& P(a,y) &(3)(6)\text{归结}&,\sigma=\{a/x\}\\ (9)& L(a) &(4)(6)\text{归结}&,\sigma=\{a/x\}\\ (10)& \sim L(a)\lor H(a) &(1)(8)\text{归结}&,\sigma=\{a/x,h/y\}\\ (11)& H(a) &(9)(10)\text{归结}\\ (12)& \text{NIL} &(7)(11)\text{归结} \end{aligned}

由此证得了:小张是快乐的。

程序验证:

ResolutionFOLeg3

# 程序实现

实际上是我的作业

python
import queue
def parse(KB: str):
    KB = KB.replace("{(", "").replace(")}", "")
    items = KB.split("),(")
    resolution_steps = map(lambda x: f'{items.index(x) + 1} ({x})', items)
    clauses = []
    for item in items:
        if item.endswith(')'):
            item = item[:-1]
        clause = []
        elements = item.split("),")
        if elements[-1] == '':
            elements.pop()
        for i in range(len(elements)):
            key, value = elements[i].split("(")
            clause.append([key, value.split(',')])
        clauses.append(clause)
    return clauses, list(resolution_steps)
def MGU(l1, l2):
    variants = ('x', 'y', 'z', 'u', 'v', 'w', 'xx', 'yy', 'zz', 'uu', 'vv', 'ww')
    dimension = len(l1)
    mapdict = {}
    for i in range(dimension):
        if l1[i] in variants and l2[i] not in variants:
            if mapdict.get(l1[i]) is None or mapdict[l1[i]] == l2[i]:
                mapdict[l1[i]] = l2[i]
            else:
                return None
        elif l1[i] not in variants and l2[i] in variants:
            if mapdict.get(l2[i]) is None or mapdict[l2[i]] == l1[i]:
                mapdict[l2[i]] = l1[i]
            else:
                return None
        elif l1[i] not in variants and l2[i] not in variants and l1[i] != l2[i]:
            return None
        else:
            if l1[i] != l2[i]:
                mapdict[l1[i]] = l2[i]
    return mapdict
# 以下是广度优先搜索策略
def resolve(c1, c2):
    list_c1keys = [sublist[0] for sublist in c1]
    list_c2keys = [sublist[0] for sublist in c2]
    for i in range(len(list_c1keys)):
        c1key: str = list_c1keys[i]
        idx_c1key = i
        flag = c1key.startswith('~')
        if flag:
            complement = c1key[1:]
        else:
            complement = '~' + c1key
        if complement in list_c2keys:
            idx_complement = list_c2keys.index(complement)
            mapdict = None
            if c1[idx_c1key][1] != c2[idx_complement][1]:
                result = MGU(c1[idx_c1key][1], c2[idx_complement][1])
                if result is None:
                    return None
                else:
                    mapdict = result
            if len(c1) > 1:
                char1 = chr(ord('a') + idx_c1key)
            else:
                char1 = '0'
            if len(c2) > 1:
                char2 = chr(ord('a') + idx_complement)
            else:
                char2 = '0'
            c1_to_del = list(c1)
            c2_to_del = list(c2)
            del c1_to_del[idx_c1key]
            del c2_to_del[idx_complement]
            c1_to_del.extend(c2_to_del)
            return c1_to_del, char1, char2, mapdict
    return None
def resolution(KB: str):
    clauses, resolution_steps = parse(KB)
    steps_idx = list(range(1, len(clauses) + 1))
    min_len = len(clauses)
    while True:
        l = len(clauses)
        for i in range(len(clauses)):
            for j in range(i + 1, len(clauses)):
                if i == j:
                    continue
                result = resolve(clauses[i], clauses[j])
                if result is None:
                    continue
                resolvent = result[0]
                if resolvent is None or resolvent in clauses:
                    continue
                clauses.append(resolvent)
                steps_idx.append([i, j])
                char1 = result[1]
                char2 = result[2]
                mapdict = result[3]
                if char1 == '0' and char2 == '0':
                    pre_string = f"R[{i + 1},{j + 1}]"
                elif char1 != '0' and char2 == '0':
                    pre_string = f"R[{i + 1}{char1},{j + 1}]"
                elif char1 == '0' and char2 != '0':
                    pre_string = f"R[{i + 1},{j + 1}{char2}]"
                else:
                    pre_string = f"R[{i + 1}{char1},{j + 1}{char2}]"
                map_string = ''
                if mapdict is not None:
                    pairs = []
                    for key, value in mapdict.items():
                        pairs.append(f"{key}={value}")
                    map_string = "{" + ",".join(pairs) + "}"
                result_string = ''
                for [key, value] in resolvent:
                    result_string += f'{key}({",".join(value)}),'
                if len(resolvent) > 1:
                    result_string = result_string[:-1]
                result_string = f'({result_string})'
                resolution_steps.append(f"{len(resolution_steps) + 1} {pre_string}{map_string} = {result_string}")
                if len(resolvent) == 0:
                    return resolution_steps, steps_idx, min_len
        if l == len(clauses):
            return None
def simplify_steps(steps, steps_idx, min_len):
    q = queue.Queue()
    not_del = [len(steps) - 1]
    q.put(steps_idx[len(steps) - 1][0])
    q.put(steps_idx[len(steps) - 1][1])
    while not q.empty():
        i = q.get()
        if i < min_len:
            continue
        not_del.append(i)
        q.put(steps_idx[i][0])
        q.put(steps_idx[i][1])
    for i in range(len(steps) - 1, min_len - 1, -1):
        if i not in not_del:
            del steps[i]
    num_map = dict()
    for i in range(len(steps)):
        num = (steps[i].split(' '))[0]
        num_map[num] = f'{i+1}'
        for t in list(num_map.keys()):
            steps[i] = steps[i].replace(t, num_map[t])
    return steps
def ResolutionFOL(KB):
    result = resolution(KB)
    if result is None:
        print("Can't be resolved!")
    else:
        (steps, steps_idx, min_len) = result
        steps = simplify_steps(steps, steps_idx, min_len)
        for step in steps:
            print(step)
if __name__ == '__main__':
    KB0 = "{(GradStudent(sue),),(~GradStudent(x),Student(x)),(~Student(x),HardWorker(x)),(~HardWorker(sue),)}"
    KB1 = "{(A(tony),),(A(mike),),(A(john),),(L(tony,rain),),(L(tony,snow),),(~A(x),S(x),C(x)),(~C(y),~L(y,rain)),(L(z,snow),~S(z)),(~L(tony,u),~L(mike,u)),(L(tony,v),L(mike,v)),(~A(w),~C(w),S(w))}"
    KB2 = "{(On(tony,mike),),(On(mike,john),),(Green(tony),),(~Green(john),),(~On(xx,yy),~Green(xx),Green(yy))}"
    KB3 = "{(~B(x,y),~F(x)),(~S(x,y),F(x)),(S(Marry,Bill)),(B(Marry,Tom))}"
    KB4 = "{(T(John),),(L(Paul,wine),),(L(Paul,cheese),),(~L(Paul,x),L(John,x)),(~T(x),~L(x,y),S(x,y)),(~S(John,z),ANSWER(z)),(~ANSWER(wine),~ANSWER(cheese))}"
    KB5 = "{(~P(x,h),~L(x),H(x)),(~S(x),P(x,y)),(~F(x),P(x,y)),(~F(x),L(x)),(~S(a)),(F(a)),(~H(a))}"
    ResolutionFOL(KB0)
    print("\x1B[31m--------------------\x1B[m")
    ResolutionFOL(KB1)
    print("\x1B[31m--------------------\x1B[m")
    ResolutionFOL(KB2)
    print("\x1B[31m--------------------\x1B[m")
    ResolutionFOL(KB3) # 不是兄弟
    print("\x1B[31m--------------------\x1B[m")
    ResolutionFOL(KB4) # John 偷东西
    print("\x1B[31m--------------------\x1B[m")
    ResolutionFOL(KB5) # 小张是快乐的
# 以下为单元归结策略
def resolve_2(c1, c2):
    list_c1keys = [sublist[0] for sublist in c1]
    list_c2keys = [sublist[0] for sublist in c2]
    c1key: str = list_c1keys[0]
    idx_c1key = list_c1keys.index(c1key)
    flag = c1key.startswith('~')
    if flag:
        complement = c1key[1:]
    else:
        complement = '~' + c1key
    if complement in list_c2keys:
        idx_complement = list_c2keys.index(complement)
        mapdict = None
        if c1[idx_c1key][1] != c2[idx_complement][1]:
            result = MGU(c1[idx_c1key][1], c2[idx_complement][1])
            if result is None:
                return None
            else:
                mapdict = result
        if len(c1) > 1:
            char1 = chr(ord('a') + idx_c1key)
        else:
            char1 = '0'
        if len(c2) > 1:
            char2 = chr(ord('a') + idx_complement)
        else:
            char2 = '0'
        c2_to_del = list(c2)
        del c2_to_del[idx_complement]
        return c2_to_del, char1, char2, mapdict
    return None
def resolution_2(KB: str):
    clauses, resolution_steps = parse(KB)
    singles = set()
    for i in range(len(clauses)):
        if len(clauses[i]) == 1:
            singles.add(i)
    while True:
        l = len(clauses)
        for i in range(len(clauses)):
            if i not in singles:
                continue
            for j in range(len(clauses)):
                if i == j:
                    continue
                result = resolve_2(clauses[i], clauses[j])
                if result is None:
                    continue
                resolvent = result[0]
                if resolvent is None or resolvent in clauses:
                    continue
                clauses.append(resolvent)
                char1 = result[1]
                char2 = result[2]
                mapdict = result[3]
                if char1 == '0' and char2 == '0':
                    pre_string = f"R[{i + 1},{j + 1}]"
                elif char1 != '0' and char2 == '0':
                    pre_string = f"R[{i + 1}{char1},{j + 1}]"
                elif char1 == '0' and char2 != '0':
                    pre_string = f"R[{i + 1},{j + 1}{char2}]"
                else:
                    pre_string = f"R[{i + 1}{char1},{j + 1}{char2}]"
                map_string = ''
                if mapdict is not None:
                    pairs = []
                    for key, value in mapdict.items():
                        pairs.append(f"{key}={value}")
                    map_string = "{" + ",".join(pairs) + "}"
                result_string = ''
                for [key, value] in resolvent:
                    result_string += f'{key}({",".join(value)}),'
                if len(resolvent) > 1:
                    result_string = result_string[:-1]
                result_string = f'({result_string})'
                resolution_steps.append(f"{len(resolution_steps) + 1} {pre_string}{map_string} = {result_string}")
                if len(resolvent) == 0:
                    return resolution_steps
                elif len(resolvent) == 1:
                    singles.add(len(clauses) - 1)
        if l == len(clauses):
            return None
def ResolutionFOL_2(KB):
    result = resolution_2(KB)
    if result is None:
        print("Can't be resolved!")
    else:
        for step in result:
            print(step)
\ No newline at end of file diff --git a/cs/ai/ml/index.html b/cs/ai/ml/index.html index 557725ae..3744a6dd 100644 --- a/cs/ai/ml/index.html +++ b/cs/ai/ml/index.html @@ -1,5 +1,5 @@ -机器学习 - 人工智能 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

# 概述

# 机器学习范式

  • 监督学习(归纳学习)
    • 给定训练数据 training data
    • 和期望的输出(标签)desired outputs (with labels)
  • 半监督学习
    • 给定训练数据 training data
    • 和部分期望的输出 some labels
  • 无监督学习
    • 仅给定训练数据 training data
    • without labels
  • 强化学习 Reinforce Learning (RL)
    • 从序列决策中获取奖励 observations and periodic rewards as the agent takes sequential actions;

# 监督学习

# 回归问题

给定 (x1,y1),(x2,y2),,(xn,yn)(x_1,y_1),(x_2, y_2),\cdots,(x_n,y_n) ,学习一个函数 f(x)f(x) 能够根据输入 xx 预测 yy ,其中 yy数值,即回归

# 分类问题

给定 (x1,y1),(x2,y2),,(xn,yn)(x_1,y_1),(x_2, y_2),\cdots,(x_n,y_n) ,学习一个函数 f(x)f(x) 能够根据输入 xx 预测 yy ,其中 yy类别,即分类

# 聚类问题

给定 x1,x2,,xnx_1,x_2,\cdots,x_n(无标签),xx 是多维的,每一维对应一个属性,输出 xx 的隐变量

# 无监督学习

独立分量分析(ICA):一种用于分离混合信号中不同独立源的方法

e g :

  • 构建计算集群
  • 社交网络分析
  • 天文数据分析
  • 市场分割

# 强化学习

输入一组包括状态、动作、延迟奖励的序列,输出一个策略

策略是一个从状态到行动的映射函数,表明每个状态下所采取的行动或行动的分布

e g :

  • 分配问题
  • 游戏 AI
  • 机器人走迷宫
  • 控制平衡问题

# 机器学习归纳

现如今已有数以万计的机器学习算法,每年仍有数百种新方法。万变不离其宗,每种机器学习都由以下三部分组成:

  • 表征(模型)
  • 优化 / 搜索
  • 评估

# 表征 (模型)

# 数值函数

线性回归

神经网络

支持向量机

# 符号函数

决策树

命题逻辑中的规则

一阶谓语逻辑中的规则

# 基于实例的函数

最近邻

基于案例求解

# 概率图解模型

朴素贝叶斯

贝叶斯网络

隐性马尔可夫模型(HMMs)

概率上下文自由语法(PCFGs)

马尔可夫网络

# 优化 / 搜索

# 梯度下降

感知机

反向传播算法

# 动态编程

HMM 学习

PCFG 学习

# 分而治之

决策树归纳法

规则学习

# 进化计算

遗传算法(GA)

遗传编程(GP)

神经进化算法

# 评估方法

准确率

精度和召回率

均方误差

先验概率

后验概率

成本 / 效率

边际效用

KL 散度

......

# 一些经验

机器学习可以被看作是利用直接或间接的经验来逼近一个选定的目标函数。

函数近似可以看作是在假定的空间(函数的表征)中寻找最适合一组参数。

不同的学习方法设定不同的假定空间(表征语言)和 / 或 采用不同的搜索技术

# 基本方法

机器学习的开发流程:

  1. 数据搜集
  2. 数据清洗
  3. 特征工程
  4. 数据建模

# 模型

机器学习首先要考虑使用什么样的模型,直观理解就是使用什么分布函数拟合数据。

模型的类别,大致有两种:

  • 概率模型 (Probabilistic Model)
    • 决策树、朴素贝叶斯、隐马尔科夫模型、高斯混合模型
  • 非概率模型 (Non-Probabilistic Model)
    • 感知机、支持向量机、KNN、AdaBoost、K-means 以及神经网络

# 损失函数

  • 0-1 损失函数 (0-1 Loss Function)

L(Y,f(X))={1,Yf(x)0,Y=f(X)L(Y,f(X))=\begin{cases} 1,Y\ne f(x)\\ 0,Y=f(X) \end{cases}

  • 平方损失函数 (Quadratic Loss Function)

L(Y,f(X))=(Yf(X))2L(Y,f(X))=(Y-f(X))^2

  • 绝对损失函数 (Absolute Loss Function)

L(Y,f(X))=Yf(X)L(Y,f(X))=|Y-f(X)|

  • 对数损失函数 (Logarithmic Loss Function)

L(Y,P(YX))=logP(YX)L(Y,P(Y|X))=-\log P(Y|X)

根据上述损失函数模型,我们可知,损失函数值越小,模型性能越好。给定一个数据集,我们将训练数据集的平均损失称为经验风险

基于经验风险最小化原则,可构建全局损失函数求解最优化问题:

minf1Nn=1NL(yn,f(xn))\min_{f}\frac{1}{N}\sum_{n=1}^{N}L(y_n,f(x_n))

当样本数量足够大时,根据大数定理,经验风险会近似于模型的期望风险。此时,经验风险最小化能确保有好的学习性能。

然而,当样本数量不足时,单单利用经验风险最小化可能会导致 “过拟合” 的问题。为此,我们在原有基础上加上用于控制模型复杂度的正则项 (Regularizer),得到结构最小化准则。

# 优化算法

定义了损失函数计算 loss 之后,问题就转化为了求 loss 最小化的问题;如何使得 loss 最小化?就需要使用优化算法。

算法指的是模型学习中的具体计算方法。

一般来说,基于参数模型构建的统计学习问题都为最优化问题,它们都具有显式的解析解。

现有的优化方法主要有:

  • 梯度下降法⭐️
    • 一阶导数为 0 的点,不断下降寻找极小值
  • 牛顿法
  • 拟牛顿法
  • ADAM
  • ......

# 回归

# 损失函数

y(i)y^{(i)}y^(i)\hat{y}^{(i)} 分别表示第 ii 个样本的真实值和预测值, mm 为样本个数

# MSE

均方误差 (Mean Square Error, MSE)

MSE(y,y^)=1mi=1m(y(i)y^(i))2MSE(y,\hat y)=\frac 1m \sum_{i=1}^m (y^{(i)}-\hat{y}^{(i)})^2

# RMSE

均方根误差 (Root Mean Square Error, RMSE)

RMSE(y,y^)=1mi=1m(y(i)y^(i))2RMSE(y,\hat y)=\sqrt{\frac 1m \sum_{i=1}^m (y^{(i)}-\hat{y}^{(i)})^2}

# 概述

# 机器学习范式

  • 监督学习(归纳学习)
    • 给定训练数据 training data
    • 和期望的输出(标签)desired outputs (with labels)
  • 半监督学习
    • 给定训练数据 training data
    • 和部分期望的输出 some labels
  • 无监督学习
    • 仅给定训练数据 training data
    • without labels
  • 强化学习 Reinforce Learning (RL)
    • 从序列决策中获取奖励 observations and periodic rewards as the agent takes sequential actions;

# 监督学习

# 回归问题

给定 (x1,y1),(x2,y2),,(xn,yn)(x_1,y_1),(x_2, y_2),\cdots,(x_n,y_n) ,学习一个函数 f(x)f(x) 能够根据输入 xx 预测 yy ,其中 yy数值,即回归

# 分类问题

给定 (x1,y1),(x2,y2),,(xn,yn)(x_1,y_1),(x_2, y_2),\cdots,(x_n,y_n) ,学习一个函数 f(x)f(x) 能够根据输入 xx 预测 yy ,其中 yy类别,即分类

# 聚类问题

给定 x1,x2,,xnx_1,x_2,\cdots,x_n(无标签),xx 是多维的,每一维对应一个属性,输出 xx 的隐变量

# 无监督学习

独立分量分析(ICA):一种用于分离混合信号中不同独立源的方法

e g :

  • 构建计算集群
  • 社交网络分析
  • 天文数据分析
  • 市场分割

# 强化学习

输入一组包括状态、动作、延迟奖励的序列,输出一个策略

策略是一个从状态到行动的映射函数,表明每个状态下所采取的行动或行动的分布

e g :

  • 分配问题
  • 游戏 AI
  • 机器人走迷宫
  • 控制平衡问题

# 机器学习归纳

现如今已有数以万计的机器学习算法,每年仍有数百种新方法。万变不离其宗,每种机器学习都由以下三部分组成:

  • 表征(模型)
  • 优化 / 搜索
  • 评估

# 表征 (模型)

# 数值函数

线性回归

神经网络

支持向量机

# 符号函数

决策树

命题逻辑中的规则

一阶谓语逻辑中的规则

# 基于实例的函数

最近邻

基于案例求解

# 概率图解模型

朴素贝叶斯

贝叶斯网络

隐性马尔可夫模型(HMMs)

概率上下文自由语法(PCFGs)

马尔可夫网络

# 优化 / 搜索

# 梯度下降

感知机

反向传播算法

# 动态编程

HMM 学习

PCFG 学习

# 分而治之

决策树归纳法

规则学习

# 进化计算

遗传算法(GA)

遗传编程(GP)

神经进化算法

# 评估方法

准确率

精度和召回率

均方误差

先验概率

后验概率

成本 / 效率

边际效用

KL 散度

......

# 一些经验

机器学习可以被看作是利用直接或间接的经验来逼近一个选定的目标函数。

函数近似可以看作是在假定的空间(函数的表征)中寻找最适合一组参数。

不同的学习方法设定不同的假定空间(表征语言)和 / 或 采用不同的搜索技术

# 基本方法

机器学习的开发流程:

  1. 数据搜集
  2. 数据清洗
  3. 特征工程
  4. 数据建模

# 模型

机器学习首先要考虑使用什么样的模型,直观理解就是使用什么分布函数拟合数据。

模型的类别,大致有两种:

  • 概率模型 (Probabilistic Model)
    • 决策树、朴素贝叶斯、隐马尔科夫模型、高斯混合模型
  • 非概率模型 (Non-Probabilistic Model)
    • 感知机、支持向量机、KNN、AdaBoost、K-means 以及神经网络

# 损失函数

  • 0-1 损失函数 (0-1 Loss Function)

L(Y,f(X))={1,Yf(x)0,Y=f(X)L(Y,f(X))=\begin{cases} 1,Y\ne f(x)\\ 0,Y=f(X) \end{cases}

  • 平方损失函数 (Quadratic Loss Function)

L(Y,f(X))=(Yf(X))2L(Y,f(X))=(Y-f(X))^2

  • 绝对损失函数 (Absolute Loss Function)

L(Y,f(X))=Yf(X)L(Y,f(X))=|Y-f(X)|

  • 对数损失函数 (Logarithmic Loss Function)

L(Y,P(YX))=logP(YX)L(Y,P(Y|X))=-\log P(Y|X)

根据上述损失函数模型,我们可知,损失函数值越小,模型性能越好。给定一个数据集,我们将训练数据集的平均损失称为经验风险

基于经验风险最小化原则,可构建全局损失函数求解最优化问题:

minf1Nn=1NL(yn,f(xn))\min_{f}\frac{1}{N}\sum_{n=1}^{N}L(y_n,f(x_n))

当样本数量足够大时,根据大数定理,经验风险会近似于模型的期望风险。此时,经验风险最小化能确保有好的学习性能。

然而,当样本数量不足时,单单利用经验风险最小化可能会导致 “过拟合” 的问题。为此,我们在原有基础上加上用于控制模型复杂度的正则项 (Regularizer),得到结构最小化准则。

# 优化算法

定义了损失函数计算 loss 之后,问题就转化为了求 loss 最小化的问题;如何使得 loss 最小化?就需要使用优化算法。

算法指的是模型学习中的具体计算方法。

一般来说,基于参数模型构建的统计学习问题都为最优化问题,它们都具有显式的解析解。

现有的优化方法主要有:

  • 梯度下降法⭐️
    • 一阶导数为 0 的点,不断下降寻找极小值
  • 牛顿法
  • 拟牛顿法
  • ADAM
  • ......

# 回归

# 损失函数

y(i)y^{(i)}y^(i)\hat{y}^{(i)} 分别表示第 ii 个样本的真实值和预测值, mm 为样本个数

# MSE

均方误差 (Mean Square Error, MSE)

MSE(y,y^)=1mi=1m(y(i)y^(i))2MSE(y,\hat y)=\frac 1m \sum_{i=1}^m (y^{(i)}-\hat{y}^{(i)})^2

# RMSE

均方根误差 (Root Mean Square Error, RMSE)

RMSE(y,y^)=1mi=1m(y(i)y^(i))2RMSE(y,\hat y)=\sqrt{\frac 1m \sum_{i=1}^m (y^{(i)}-\hat{y}^{(i)})^2}

# MAE

平均绝对误差 (Mean Absolute Error, MAE)

MAE(y,y^)=1mi=1my(i)y^(i)MAE(y,\hat y)=\frac 1m \sum_{i=1}^m |y^{(i)}-\hat{y}^{(i)}|

# 线性回归

线性回归(Linear Regression)

是一种通过属性的线性组合来进行预测的线性模型,其目的是找到一条直线或者一个平面或者更高维的超平面,使得预测值与真实值之间的误差最小化。

  • 损失函数(Loss Function)度量单样本预测的错误程度,损失函数值越小,模型就越好。
  • 代价函数(Cost Function)度量全部样本集的平均误差。常用的代价函数包括均方误差、均方根误差、平均绝对误差等。
  • 目标函数(Object Function)代价函数和正则化函数,最终要优化的函数。
  • 学习率 α\alpha ,可以理解为步长

learning-rate

# 梯度下降

  • 批量梯度下降 BGD(Batch Gradient Descent)
    • 梯度下降的每一步中,都用到了所有的训练样本
  • 随机梯度下降 SGD(Stochastic Gradient Descent)
    • 梯度下降的每一步中,用到一个样本,在每一次计算之后便更新参数 ,而不需要首先将所有的训练集求和
  • 小批量梯度下降 MBGD(Mini-Batch Gradient Descent)
    • 梯度下降的每一步中,用到了一定批量的训练样本

wj:=wjα1bk=ii+b1(h(x(k))y(k))xj(k)w_{j}:=w_{j}-\alpha\frac{1}{b}\sum_{k=i}^{i+b-1}\big(h\big(x^{(k)}\big)-y^{(k)}\big)x_{j}^{(k)}

b=1b=1 为随机梯度下降 (SGD, Stochastic Gradient Descent)

b=mb=m 为批量梯度下降 (BGD, Batch Gradient Descent)

b=batch_sizeb=batch\_ size 为小批量梯度下降 (MBGD, Mini-Batch Gradient Descent)

归一化

数据归一化的目的是使得各特征对目标变量的影响一致,会将特征数据进行伸缩变化,所以数据归一化是会改变特征数据分布的

将数据映射到 [0,1][0,1] 区间

Z-Score

数据标准化为了不同特征之间具备可比性,经过标准化变换之后的特征数据分布没有发生改变。

就是当数据特征取值范围或单位差异较大时,最好是做一下标准化处理

处理后的数据均值为 0,方差为 1

欠拟合过拟合

# 过拟合的处理

# 交叉验证

交叉验证 Cross Validation

最常见的缓解过拟合问题的方法,将数据集分为训练用的和测试用的两份数据集(大多是 训练:测试 = 9:1 )

多重交叉验证 N-fold Cross Validation

迁移学习

# 获得更多的训练数据

使用更多的训练数据是解决过拟合问题最有效的手段,因为更多的样本能够让模型学习到更多更有效的特征,减小噪声的影响

# 降维

即丢弃一些不能帮助我们正确预测的特征。可以是手工选择保留哪些特征,或者使用一些模型选择的算法来帮忙(例如 PCA)。

# 正则化

正则化 (regularization) 的技术,保留所有的特征,但是减少参数的大小(magnitude),它可以改善或者减少过拟合问题。

# 集成学习方法

集成学习是把多个模型集成在一起,来降低单一模型的过拟合风险。

# 欠拟合的处理

# 添加新特征

当特征不足或者现有特征与样本标签的相关性不强时,模型容易出现欠拟合。通过挖掘组合特征等新的特征,往往能够取得更好的效果。

# 增加模型复杂度

简单模型的学习能力较差,通过增加模型的复杂度可以使模型拥有更强的拟合能力。例如,在线性模型中添加高次项,在神经网络模型中增加网络层数或神经元个数等。

# 减小正则化系数

正则化是用来防止过拟合的,但当模型出现欠拟合现象时,则需要有针对性地减小正则化系数。

# 分类问题

分类问题和回归问题有一定的相似性,都是通过对数据集的学习来对未知结果进行预测,区别在于输出值不同。

  • 分类问题的输出值是离散值(如垃圾邮件和正常邮件)。
  • 回归问题的输出值是连续值(例如房子的价格)。

在回归问题的基础上,对预测结果值进行量化,即将连续值量化为离散值,就能解决分类问题。

# 逻辑回归

逻辑回归是线性回归的一种扩展,用来处理分类问题。

常用的逻辑函数 Sigmoid 及其导函数:

σ(x)=11+exσ(x)=σ(x)[1σ(x)]\sigma(x)=\frac{1}{1+e^{-x}}\\ \sigma'(x)=\sigma(x)\cdot[1-\sigma(x)]

损失函数:

J(w)=1ml(w)=1mi=1m[y(i)ln(y^(i))+(1y(i))ln(1y^(i))]J(w)=-\frac{1}{m}l(w)=-\frac{1}{m}\sum_{i=1}^{m}\big[y^{(i)}\ln(\hat y^{(i)})+(1-y^{(i)})\ln(1-\hat y^{(i)})\big]

梯度为:

1mi=1m(y^(i)y(i))xj(i)\frac{1}{m}\sum_{i=1}^{m}(\hat y^{(i)}-y^{(i)})x^{(i)}_j

梯度下降更新参数 wjw_j

wj:=wjαJ(w)wjw_{j}:=w_{j}-\alpha\frac{\partial J(w)}{\partial w_j}

# SVM

支持向量机,Support Vector Machines,SVM

  • 支持向量:支撑平面上把两类类别划分开来的超平面的向量点
  • 机:一个算法

# 聚类问题

只考 K-means

# K-means

K-means 算法是一种无监督学习方法,是最普及的聚类算法,算法使用一个没有标签的数据集,然后将数据聚类成不同的组。

# 算法流程

  1. 初试化簇质心为的任意点(KK 个)。初试化时,必须注意簇的质心必须小于训练数据点的数目。

  2. 遍历所有数据点,计算所有质心与数据点的距离。这些簇将根据质心的最小距离而形成。

  3. 移动质心,因为上面步骤中形成的簇没有优化,所以需要形成优化的簇。为此需要迭代地将质心移动到一个新位置。取一个簇的数据点,计算平均值,然后将簇的质心移动到这个新位置。所有簇重复相同的步骤。

  4. 重复上述步骤,直至收敛。

# 优点缺点

  • 优点

    • 原理简单,实现容易,收敛速度快

    • 聚类效果较优

    • 算法的可解释度比较强

    • 主要需要调参的参数仅仅时簇数 K

  • 缺点

    • 需要预先指定簇的数量

    • 如果有两个高度重叠的数据,那么它就无法区分,也不能判断有两个簇

    • 欧几里得距离限制了能处理的数据变量类型

    • 随机选择质心并不能带来理想的结果

    • 无法处理异常值和噪声数据

    • 不适用于非线性数据

    • 对特征尺度敏感

    • 如果遇到非常大的数据集,那么计算机可能回崩溃

# 决策树模型

决策树基于 “树” 结构进行决策

  • 每个 “内部结点” 对应于某个属性上的 “测试”(test)
  • 每个分支对应于该测试的一种可能结果 (即该属性的某个取值)
  • 每个 “叶结点” 对应于一个 “预测结果”

学习过程:通过对训练样本的分析来确定 “划分属性”(即内部结点所对应的属性 )

预测过程:将测试示例从根结点开始沿着划分属性所构成的 “判定测试序列” 下行,直到叶结点

策略:“分而治之”( divide-and-conquer)

自根至叶的递归过程在每个中间结点寻找一个 “划分”( split or test) 属性。

三种停止条件

  • 当前结点包含的样本全属于同一类别,无需划分
  • 当前属性集为空,或是所有样本在所有属性上取值相同,无法划分
  • 当前结点包含的样本集合为空,不能划分

# 伪代码

  • 输入:

    • 训练集 D={(x1,y1),(x2,y2),,(xm,ym)}D=\{(x_1,y_1),(x_2,y_2),\cdots,(x_m,y_m)\}
    • 属性集 A={a1,a2,,ad}A=\{a_1,a_2,\cdots,a_d\}
  • 过程:

    • 函数 TreeGenerate(D,A)
  • 输出:

    • 以 node 为根结点的一棵决策树
伪代码
生成结点 node;
if D 中样本全属于同一类别 C then:
		将 node 标记为 C 类叶结点;
  	return;
end if;
if A is not None or D 中样本在 A 上取值相同 then:
		将 node 标记为叶结点,其类别标记为 D 中样本数最多的类;  # 利用当前结点的后验分布
  	return;
end if;
从 A 中选择最优划分属性 a ;
for a  的每一个值 a(i) do
		为 node 生成一个分支;
  	令 D(i) 表示 D 中在 a 上取值为 a(i) 的样本子集;
    if D(i) is None then:
    		将分支结点标记为叶结点,其类别标记为 D 中样本最多的类;  # 将父结点的样本分布作为当前结点的先验分布
      	return;
    else:
				以 TreeGenerate(D(i), A\{a}) 为分支结点
    end if;
end for;

# 信息增益 (ID3)

(ID3 算法)

一个离散属性 aa 的取值:{a1,a2,,aV}\{a^1,a^2,\cdots,a^V\}

DD 的信息熵定义为

Ent(D)=k=1ypklog2(pk)\text{Ent}(D)=-\sum_{k=1}^{|y|}p_k\log_2(p_k)

Ent(D)\text{Ent}(D) 的值越小,则 DD 的纯度越高

以属性 aa 对数据集 DD 进行划分,DvD^vDDaa 上取值为 ava_v 的样本集合,获得属性 aa信息增益为:

Gain(D,a)=Ent(D)v=1VDvDEnt(Dv)\text{Gain}(D,a)=\text{Ent}(D)-\sum_{v=1}^{V}\frac{|D^v|}{|D|}\text{Ent}(D^v)

其中:

  • Ent(D)\text{Ent}(D) 是划分前的信息熵;
  • v=1VDvD\sum_{v=1}^{V}\frac{|D^v|}{|D|} 是第 vv 个分支的权重,样本越多权重越大,该分支也就越重要;
  • Ent(Dv)\text{Ent}(D^v) 是划分后的信息熵;

Ent(D)\text{Ent}(D) 满足以下性质

Ent(D)=k=1ypklog2(pk)=k=1ypklog2(1pk)log2(k=1ypk1pk)=log2(y)\begin{aligned} \text{Ent}(D)&=-\sum_{k=1}^{|y|}p_k\log_2(p_k)=\sum_{k=1}^{|y|}p_k\log_2(\frac 1{p_k})\\ &\le \log_2\bigg(\sum_{k=1}^{|y|}p_k\frac 1{p_k}\bigg)=\log_2(|y|) \end{aligned}

# 增益率 (C4.5)

(C4.5 算法)

信息增益对可取值数目较多的属性有所偏好,有明显弱点,例如考虑将 “编号” 作为一个属性。所以引入增益率

GainRatio(D,a)=Gain(D,a)IV(a)\text{GainRatio}(D,a)=\frac{\text{Gain}(D,a)}{\text{IV}(a)}

其中:

IV(a)=v=1VDvDlog2DvD\text{IV}(a)=-\sum_{v=1}^{V}\frac{|D^v|}{|D|}\log_2\frac{|D^v|}{|D|}

属性 a 的可能取值数目越多 (即 V 越大),则 IV (a) 的值通常就越大

启发式:先从候选划分属性中找出信息增益高于平均水平的,再从中选取增益率最高的。

# 基尼系数 (CART)

(CART 算法)

样本集(数据集)DD 的基尼系数定义如下:

Gini(D)=k=1ykkpkpk=1k=1ypk2\text{Gini}(D)=\sum_{k=1}^{|y|}\sum_{k'\neq k}p_kp_{k'}=1-\sum_{k=1}^{|y|}p_k^2

反映了从 D 中随机抽取两个样例,其类别标记不一致的概率。

样本集(数据集)DD 的属性 aa 的基尼系数定义如下:

Gini(D,a)=v=1VDvDGini(D)\text{Gini}(D,a)=\sum_{v=1}^{V}\frac{|D^v|}{|D|}\text{Gini}(D)

建立决策树时,在候选属性集合中,选取那个使划分后基尼指数最小的属性。

# 剪枝

为了尽可能正确分类训练样本,有可能造成分支过多,也就是决策树的过拟合问题,可通过主动去掉一些分支来降低过拟合的风险

基本策略 :

  • 预剪枝 (pre-pruning) : 提前终止某些分支的生长

    在构造决策树的过程中,先对每个结点在划分前进行估计,如果当前结点的划分不能带来决策树模型泛化性能的提升,则不对当前结点进行划分并且将当前结点标记为叶结点。(泛化性能可根据在测试集上的准确率衡量)

    • 训练时间开销降低,测试时间开销降低
    • : 过拟合风险降低,欠拟合风险增加
  • 后剪枝 (post-pruning) : 生成一棵完全树,再 “回头” 剪枝

    先把整颗决策树构造完毕,然后自底向上的对非叶结点进行考察,若将该结点对应的子树换为叶结点能够带来泛化性能的提升,则把该子树替换为叶结点。

    • 训练时间开销增加,测试时间开销降低
    • 过拟合风险降低,欠拟合风险基本不变

泛化性能:后剪枝通常优于预剪枝

剪枝过程中需评估剪枝前后决策树的优劣

  • 现在我们假定使用 “留出法”

推荐阅读 https://blog.csdn.net/u012328159/article/details/79285214

Edited on Views times
\ No newline at end of file +219 661 l218 661zM702 80H400000v40H742z"/>

# MAE

平均绝对误差 (Mean Absolute Error, MAE)

MAE(y,y^)=1mi=1my(i)y^(i)MAE(y,\hat y)=\frac 1m \sum_{i=1}^m |y^{(i)}-\hat{y}^{(i)}|

# 线性回归

线性回归(Linear Regression)

是一种通过属性的线性组合来进行预测的线性模型,其目的是找到一条直线或者一个平面或者更高维的超平面,使得预测值与真实值之间的误差最小化。

  • 损失函数(Loss Function)度量单样本预测的错误程度,损失函数值越小,模型就越好。
  • 代价函数(Cost Function)度量全部样本集的平均误差。常用的代价函数包括均方误差、均方根误差、平均绝对误差等。
  • 目标函数(Object Function)代价函数和正则化函数,最终要优化的函数。
  • 学习率 α\alpha ,可以理解为步长

learning-rate

# 梯度下降

  • 批量梯度下降 BGD(Batch Gradient Descent)
    • 梯度下降的每一步中,都用到了所有的训练样本
  • 随机梯度下降 SGD(Stochastic Gradient Descent)
    • 梯度下降的每一步中,用到一个样本,在每一次计算之后便更新参数 ,而不需要首先将所有的训练集求和
  • 小批量梯度下降 MBGD(Mini-Batch Gradient Descent)
    • 梯度下降的每一步中,用到了一定批量的训练样本

wj:=wjα1bk=ii+b1(h(x(k))y(k))xj(k)w_{j}:=w_{j}-\alpha\frac{1}{b}\sum_{k=i}^{i+b-1}\big(h\big(x^{(k)}\big)-y^{(k)}\big)x_{j}^{(k)}

b=1b=1 为随机梯度下降 (SGD, Stochastic Gradient Descent)

b=mb=m 为批量梯度下降 (BGD, Batch Gradient Descent)

b=batch_sizeb=batch\_ size 为小批量梯度下降 (MBGD, Mini-Batch Gradient Descent)

归一化

数据归一化的目的是使得各特征对目标变量的影响一致,会将特征数据进行伸缩变化,所以数据归一化是会改变特征数据分布的

将数据映射到 [0,1][0,1] 区间

Z-Score

数据标准化为了不同特征之间具备可比性,经过标准化变换之后的特征数据分布没有发生改变。

就是当数据特征取值范围或单位差异较大时,最好是做一下标准化处理

处理后的数据均值为 0,方差为 1

欠拟合过拟合

# 过拟合的处理

# 交叉验证

交叉验证 Cross Validation

最常见的缓解过拟合问题的方法,将数据集分为训练用的和测试用的两份数据集(大多是 训练:测试 = 9:1 )

多重交叉验证 N-fold Cross Validation

迁移学习

# 获得更多的训练数据

使用更多的训练数据是解决过拟合问题最有效的手段,因为更多的样本能够让模型学习到更多更有效的特征,减小噪声的影响

# 降维

即丢弃一些不能帮助我们正确预测的特征。可以是手工选择保留哪些特征,或者使用一些模型选择的算法来帮忙(例如 PCA)。

# 正则化

正则化 (regularization) 的技术,保留所有的特征,但是减少参数的大小(magnitude),它可以改善或者减少过拟合问题。

# 集成学习方法

集成学习是把多个模型集成在一起,来降低单一模型的过拟合风险。

# 欠拟合的处理

# 添加新特征

当特征不足或者现有特征与样本标签的相关性不强时,模型容易出现欠拟合。通过挖掘组合特征等新的特征,往往能够取得更好的效果。

# 增加模型复杂度

简单模型的学习能力较差,通过增加模型的复杂度可以使模型拥有更强的拟合能力。例如,在线性模型中添加高次项,在神经网络模型中增加网络层数或神经元个数等。

# 减小正则化系数

正则化是用来防止过拟合的,但当模型出现欠拟合现象时,则需要有针对性地减小正则化系数。

# 分类问题

分类问题和回归问题有一定的相似性,都是通过对数据集的学习来对未知结果进行预测,区别在于输出值不同。

  • 分类问题的输出值是离散值(如垃圾邮件和正常邮件)。
  • 回归问题的输出值是连续值(例如房子的价格)。

在回归问题的基础上,对预测结果值进行量化,即将连续值量化为离散值,就能解决分类问题。

# 逻辑回归

逻辑回归是线性回归的一种扩展,用来处理分类问题。

常用的逻辑函数 Sigmoid 及其导函数:

σ(x)=11+exσ(x)=σ(x)[1σ(x)]\sigma(x)=\frac{1}{1+e^{-x}}\\ \sigma'(x)=\sigma(x)\cdot[1-\sigma(x)]

损失函数:

J(w)=1ml(w)=1mi=1m[y(i)ln(y^(i))+(1y(i))ln(1y^(i))]J(w)=-\frac{1}{m}l(w)=-\frac{1}{m}\sum_{i=1}^{m}\big[y^{(i)}\ln(\hat y^{(i)})+(1-y^{(i)})\ln(1-\hat y^{(i)})\big]

梯度为:

1mi=1m(y^(i)y(i))xj(i)\frac{1}{m}\sum_{i=1}^{m}(\hat y^{(i)}-y^{(i)})x^{(i)}_j

梯度下降更新参数 wjw_j

wj:=wjαJ(w)wjw_{j}:=w_{j}-\alpha\frac{\partial J(w)}{\partial w_j}

# SVM

支持向量机,Support Vector Machines,SVM

  • 支持向量:支撑平面上把两类类别划分开来的超平面的向量点
  • 机:一个算法

# 聚类问题

只考 K-means

# K-means

K-means 算法是一种无监督学习方法,是最普及的聚类算法,算法使用一个没有标签的数据集,然后将数据聚类成不同的组。

# 算法流程

  1. 初试化簇质心为的任意点(KK 个)。初试化时,必须注意簇的质心必须小于训练数据点的数目。

  2. 遍历所有数据点,计算所有质心与数据点的距离。这些簇将根据质心的最小距离而形成。

  3. 移动质心,因为上面步骤中形成的簇没有优化,所以需要形成优化的簇。为此需要迭代地将质心移动到一个新位置。取一个簇的数据点,计算平均值,然后将簇的质心移动到这个新位置。所有簇重复相同的步骤。

  4. 重复上述步骤,直至收敛。

# 优点缺点

  • 优点

    • 原理简单,实现容易,收敛速度快

    • 聚类效果较优

    • 算法的可解释度比较强

    • 主要需要调参的参数仅仅时簇数 K

  • 缺点

    • 需要预先指定簇的数量

    • 如果有两个高度重叠的数据,那么它就无法区分,也不能判断有两个簇

    • 欧几里得距离限制了能处理的数据变量类型

    • 随机选择质心并不能带来理想的结果

    • 无法处理异常值和噪声数据

    • 不适用于非线性数据

    • 对特征尺度敏感

    • 如果遇到非常大的数据集,那么计算机可能回崩溃

# 决策树模型

决策树基于 “树” 结构进行决策

  • 每个 “内部结点” 对应于某个属性上的 “测试”(test)
  • 每个分支对应于该测试的一种可能结果 (即该属性的某个取值)
  • 每个 “叶结点” 对应于一个 “预测结果”

学习过程:通过对训练样本的分析来确定 “划分属性”(即内部结点所对应的属性 )

预测过程:将测试示例从根结点开始沿着划分属性所构成的 “判定测试序列” 下行,直到叶结点

策略:“分而治之”( divide-and-conquer)

自根至叶的递归过程在每个中间结点寻找一个 “划分”( split or test) 属性。

三种停止条件

  • 当前结点包含的样本全属于同一类别,无需划分
  • 当前属性集为空,或是所有样本在所有属性上取值相同,无法划分
  • 当前结点包含的样本集合为空,不能划分

# 伪代码

  • 输入:

    • 训练集 D={(x1,y1),(x2,y2),,(xm,ym)}D=\{(x_1,y_1),(x_2,y_2),\cdots,(x_m,y_m)\}
    • 属性集 A={a1,a2,,ad}A=\{a_1,a_2,\cdots,a_d\}
  • 过程:

    • 函数 TreeGenerate(D,A)
  • 输出:

    • 以 node 为根结点的一棵决策树
伪代码
生成结点 node;
if D 中样本全属于同一类别 C then:
		将 node 标记为 C 类叶结点;
  	return;
end if;
if A is not None or D 中样本在 A 上取值相同 then:
		将 node 标记为叶结点,其类别标记为 D 中样本数最多的类;  # 利用当前结点的后验分布
  	return;
end if;
从 A 中选择最优划分属性 a ;
for a  的每一个值 a(i) do
		为 node 生成一个分支;
  	令 D(i) 表示 D 中在 a 上取值为 a(i) 的样本子集;
    if D(i) is None then:
    		将分支结点标记为叶结点,其类别标记为 D 中样本最多的类;  # 将父结点的样本分布作为当前结点的先验分布
      	return;
    else:
				以 TreeGenerate(D(i), A\{a}) 为分支结点
    end if;
end for;

# 信息增益 (ID3)

(ID3 算法)

一个离散属性 aa 的取值:{a1,a2,,aV}\{a^1,a^2,\cdots,a^V\}

DD 的信息熵定义为

Ent(D)=k=1ypklog2(pk)\text{Ent}(D)=-\sum_{k=1}^{|y|}p_k\log_2(p_k)

Ent(D)\text{Ent}(D) 的值越小,则 DD 的纯度越高

以属性 aa 对数据集 DD 进行划分,DvD^vDDaa 上取值为 ava_v 的样本集合,获得属性 aa信息增益为:

Gain(D,a)=Ent(D)v=1VDvDEnt(Dv)\text{Gain}(D,a)=\text{Ent}(D)-\sum_{v=1}^{V}\frac{|D^v|}{|D|}\text{Ent}(D^v)

其中:

  • Ent(D)\text{Ent}(D) 是划分前的信息熵;
  • v=1VDvD\sum_{v=1}^{V}\frac{|D^v|}{|D|} 是第 vv 个分支的权重,样本越多权重越大,该分支也就越重要;
  • Ent(Dv)\text{Ent}(D^v) 是划分后的信息熵;

Ent(D)\text{Ent}(D) 满足以下性质

Ent(D)=k=1ypklog2(pk)=k=1ypklog2(1pk)log2(k=1ypk1pk)=log2(y)\begin{aligned} \text{Ent}(D)&=-\sum_{k=1}^{|y|}p_k\log_2(p_k)=\sum_{k=1}^{|y|}p_k\log_2(\frac 1{p_k})\\ &\le \log_2\bigg(\sum_{k=1}^{|y|}p_k\frac 1{p_k}\bigg)=\log_2(|y|) \end{aligned}

# 增益率 (C4.5)

(C4.5 算法)

信息增益对可取值数目较多的属性有所偏好,有明显弱点,例如考虑将 “编号” 作为一个属性。所以引入增益率

GainRatio(D,a)=Gain(D,a)IV(a)\text{GainRatio}(D,a)=\frac{\text{Gain}(D,a)}{\text{IV}(a)}

其中:

IV(a)=v=1VDvDlog2DvD\text{IV}(a)=-\sum_{v=1}^{V}\frac{|D^v|}{|D|}\log_2\frac{|D^v|}{|D|}

属性 a 的可能取值数目越多 (即 V 越大),则 IV (a) 的值通常就越大

启发式:先从候选划分属性中找出信息增益高于平均水平的,再从中选取增益率最高的。

# 基尼系数 (CART)

(CART 算法)

样本集(数据集)DD 的基尼系数定义如下:

Gini(D)=k=1ykkpkpk=1k=1ypk2\text{Gini}(D)=\sum_{k=1}^{|y|}\sum_{k'\neq k}p_kp_{k'}=1-\sum_{k=1}^{|y|}p_k^2

反映了从 D 中随机抽取两个样例,其类别标记不一致的概率。

样本集(数据集)DD 的属性 aa 的基尼系数定义如下:

Gini(D,a)=v=1VDvDGini(D)\text{Gini}(D,a)=\sum_{v=1}^{V}\frac{|D^v|}{|D|}\text{Gini}(D)

建立决策树时,在候选属性集合中,选取那个使划分后基尼指数最小的属性。

# 剪枝

为了尽可能正确分类训练样本,有可能造成分支过多,也就是决策树的过拟合问题,可通过主动去掉一些分支来降低过拟合的风险

基本策略 :

  • 预剪枝 (pre-pruning) : 提前终止某些分支的生长

    在构造决策树的过程中,先对每个结点在划分前进行估计,如果当前结点的划分不能带来决策树模型泛化性能的提升,则不对当前结点进行划分并且将当前结点标记为叶结点。(泛化性能可根据在测试集上的准确率衡量)

    • 训练时间开销降低,测试时间开销降低
    • : 过拟合风险降低,欠拟合风险增加
  • 后剪枝 (post-pruning) : 生成一棵完全树,再 “回头” 剪枝

    先把整颗决策树构造完毕,然后自底向上的对非叶结点进行考察,若将该结点对应的子树换为叶结点能够带来泛化性能的提升,则把该子树替换为叶结点。

    • 训练时间开销增加,测试时间开销降低
    • 过拟合风险降低,欠拟合风险基本不变

泛化性能:后剪枝通常优于预剪枝

剪枝过程中需评估剪枝前后决策树的优劣

  • 现在我们假定使用 “留出法”

推荐阅读 https://blog.csdn.net/u012328159/article/details/79285214

Edited on Views times
\ No newline at end of file diff --git a/cs/ai/rl/index.html b/cs/ai/rl/index.html index 67e696cb..2f493ed2 100644 --- a/cs/ai/rl/index.html +++ b/cs/ai/rl/index.html @@ -1 +1 @@ -强化学习基础 - 人工智能 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

强烈推荐:https://hrl.boyuai.com/

# 基本概念

RL, in a nutshell, is to “learn to make good sequences of decisions through trail-and-errors”

# 机器学习范式

  • 监督

    • 给定训练数据 training data
    • 和期望的输出(标签)desired outputs (with labels)
  • 半监督

    • 给定训练数据 training data
    • 和部分期望的输出 some labels
  • 无监督

    • 仅给定训练数据 training data
    • without labels
  • 强化学习 Reinforce Learning (RL)

    • 从序列决策中获取奖励 observations and periodic rewards as the agent takes sequential actions;

强化学习是一种自动化目标导向的计算方法,与其他机器学习范式最大的区别在于:

  • 交互:与环境之间存在交互,从环境中学习
  • 决策序列:以前的决策会影响以后的决策,也就是学习的过程
  • 探索:不断试错、探索的学习过程

# 强化学习算法构成

# 给定

# 三大集合

  • 状态集合
  • 动作集合
  • 奖励值集合

# 两大模型 (两大主角)

两大模型

# 环境模型(optional)

述了环境的具体表现。在给定状态和动作情况下,模型可能会预测结果的下一个状态和下一个奖励

  • 有环境模型:定义了环境模型
  • 无环境模型:不定义环境模型,靠智能体感知实时状态来理解环境
# 主角模型(智能体)

概念来源于分布式人工智能思想。通常意义上,智能体是一个能够感知并施加某种作用到自身和环境、有生命周期的一个物理的或抽象的计算实体。智能体接受一个任务指令后,经过学习,可自主完成该任务

# 目标

智能体获取从任务开始到任务结束过程中每一步的最优动作

# 三大特征

  • 探索试错
  • 延迟回报
  • 长期目标

强化学习的关键特征是长期目标优化,智能体关注整体学习问题,优化整个过程

大多强化学习算法的关键特征是值函数的使用

# 特有挑战

探索和开发 (exploration and exploitation):短期和长期目标的权衡

# 强化学习的问题建模

# 问题模型

# 状态空间

智能体所处的实际环境所有状态集合

# 当前状态

智能体感知到的环境当前所处的状态

# 动作策略

智能体在某时间的行为

# 奖励信号

智能体收到的关于动作效果的反馈

# 价值函数

奖励信号 只能说明 动作的即时效果,而 价值函数 则指明什么策略的 长期效果 是好还是差。

# 马尔科夫决策问题模型

# 马尔可夫奖励过程 (MRP)

马尔可夫奖励过程(Markov reward process)

# 动作序列集

动作序列集(Action Sequence Set)是指一组按特定顺序排列的动作序列。它通常用于描述某个系统、机器人或智能体在特定环境中的行为。动作序列集可以包含一系列离散的动作步骤,也可以包含连续的动作序列。

动作序列集可以被分为两类:

  • 有终止态的离散动作(动作幕)
  • 无终止态的连续动作

# 动作幕 (Episode)

一个完整的有起始终止状态动作序列集(Episode,简称动作幕

  • 从标准起始态或从标准分布的初始态集合中抽取初始态开始
  • 以终止态结束
  • 不同动作序列集都可以认为是在同一个终止态结束,只是序列具体元素不同

区分奖励和收益

# 简单收益

Return (aka cumulative future reward) ,未来的累积奖励,也就是简单收益

# 折扣回报

Discounted Return ,折扣回报,也就是引入了折扣因子的简单收益。

# 马尔科夫决策过程 (MDP)

马尔可夫决策过程(Markov decision process,MDP)

MDP 与 MRP 非常相像,主要区别为 MDP 中的状态转移函数和奖励函数都比 MRP 多了动作 aa 作为自变量。

# 问题模型

MDP(马尔科夫决策过程)是一个描述智能体和环境交互的四元组 (S, A, 𝑅, 𝑝)

  • S :Status 状态的集合

  • A :Action 动作的集合

  • R :Reward 奖励的集合

  • p :当前状态执行了一个动作跳转到下一个状态的概率

  • 对象

    • 智能体
    • 环境
  • 交互

    • 动作
    • 状态
    • 奖励

MDP模型

# 状态 - 奖励表

扫地机器人状态 - 奖励表

扫地机器人状态-奖励表

# 状态迁移图

扫地机器人状态迁移图

扫地机器人状态迁移图

# 游戏轨迹

(state, action, reward) trajectory 游戏轨迹

s1,a1,r1,s2,a2,r2,sT,aT,rTs_1,a_1,r_1,s_2,a_2,r_2,\cdots s_T,a_T,r_T

即时奖励和长期目标的关系

某一决策的即时奖励较高不一定带来较高的长期目标

追求长期目标 ≠ 追求即时奖励

# 强化学习的值函数

# 基础概统知识

以下定义针对离散型随机变量:

# 概率

随机变量 A = a 发生的概率

p(A=a)p(A=a)

# 条件概率

在 C = c 条件下,A = a 的概率

p(A=aC=c)p(A=a|C=c)

智能体处于状态 s(S=s)s(S=s) 时,它的策略 π\pi 执行动作 a(A=a)a(A=a) 的概率

π(as)\pi(a|s)

# 期望

也称随机变量 𝑿 的期待值

E[X]=ixip(X=xi)\mathbb{E}[X]=\sum_i x_i p(X=x_i)

# 条件期望

Y=y 条件下,X 的期望

E[XY=y]=ixip(X=xiY=y)\mathbb{E}[X|Y=y]=\sum_i x_i p (X=x_i |Y=y)

# 简单值函数

直觉上看,为找到一个好的动作决策,需为每个动作赋值,只考虑了即时奖励,未考虑长期过程

有了值函数 q (a) ,决策问题即可转为优化问题求解,找最优动作:

argmaxaq(a)\underset{a}{argmax} \ q(a)

# 定义

q(a)=E[RA=a]a{1,...,k}=rp(ra)rq(a)=\mathbb{E}[R|A=a]\forall a\in \{ 1,...,k \}=\sum_r p(r|a)r

其中 p (r | a) 是在执行 a 动作情况下观察到智能体获得奖励 r 的概率;R 字母表示 Reward 奖励。

# 采样平均

除了按定义方法计算,还可以通过采样平均方法计算,即通过以往动作所获奖励情况来估算值函数

Q(a)=获得奖励的总和动作a的执行次数Q(a)=\frac{\text{获得奖励的总和}}{\text{动作a的执行次数}}

具体计算公式为:

Q(a)=i=1n1Ri1Ai=ai=1n11Ai=aQ(a)=\frac{\sum_{i=1}^{n-1}R_i \mathbb{1}_{A_i = a}}{\sum_{i=1}^{n-1}\mathbb{1}_{A_i = a}}

其中 1Ai=a\mathbb{1}_{A_i = a} 表示 Ai=aA_i =a 时取 1 ,否则取 0

# 问题

未考虑长期过程,实际并不采用

# 简单收益

# 简单收益的定义

G 字母表示 Gain 收益,也叫做回报 Return

Gt=Rt+1+Rt+2+Rt+3+...+RTG_t= R_{t+1}+R_{t+2}+R_{t+3}+...+R_T

# 简单收益的期望

# 离散动作的简单收益

E[Gt]=E[Rt+1+Rt+2+Rt+3+...+RT]\mathbb{E}[G_t]=\mathbb{E}[R_{t+1}+R_{t+2}+R_{t+3}+...+R_T]

# 连续动作的简单收益

E[Gt]=E[Rt+1+Rt+2+Rt+3+...]\mathbb{E}[G_t]=\mathbb{E}[R_{t+1}+R_{t+2}+R_{t+3}+...]

# 存在问题

若奖励都是非负的,收益会不会是无穷大?

# 折扣目标收益

引入折扣因子 γ\gammaγ<1\gamma \lt 1,得到折扣回报(Discounted Return):

Gt=Rt+1+γRt+2+γ2Rt+3+...+γk1Rt+k+...=k=0γkRt+k+1G_t= R_{t+1}+\gamma R_{t+2}+\gamma^2 R_{t+3}+...+ \gamma^{k-1}R_{t+k}+...=\sum_{k=0}^{\infin}\gamma^k R_{t+k+1}

γ=0\gamma =0 时:

Gt=Rt+1+0Rt+2+02Rt+3+...+0k1Rt+k+...=Rt+1G_t= R_{t+1}+0 R_{t+2}+0^2 R_{t+3}+...+ 0^{k-1}R_{t+k}+...=R_{t+1}

RmaxR_{max} 为智能体所能获取的最大奖励,则:

Rt+k+1RmaxR_{t+k+1}\leq R_{max}

Gt=k=0γkRt+k+1k=0γkRmax=Rmaxk=0γk=Rmax1γG_t = \sum_{k=0}^{\infin}\gamma^k R_{t+k+1} \leq \sum_{k=0}^{\infin}\gamma^k R_{max} = R_{max}\sum_{k=0}^{\infin}\gamma^k = \frac{R_{max}}{1-\gamma}

# 目标收益的递归计算

定义 GT=0,tTG_T = 0\ ,\ t\leq T

Gt=Rt+1+γRt+2+γ2Rt+3+...+γk1Rt+k+...=Rt+1+γ(Rt+2+γRt+3+...+γk2Rt+k+...)=Rt+1+γGt+1\begin{aligned} G_t &= R_{t+1}+\gamma R_{t+2}+\gamma^2 R_{t+3}+...+ \gamma^{k-1}R_{t+k}+... \\ &= R_{t+1}+\gamma (R_{t+2}+\gamma R_{t+3}+...+ \gamma^{k-2}R_{t+k}+...) \\ &= R_{t+1}+\gamma G_{t+1} \end{aligned}

从而有:

Gt=Rt+1+γGt+1G_t =R_{t+1}+\gamma G_{t+1}

特别地,t = T 时有 GT=0G_T = 0 ;t = T-1 时有 Gt1=RT+γGTG_{t-1} = R_T +\gamma G_T

# 标准状态值函数

一个状态在策略 𝝅 下的值函数,记作:vπ(s)v_\pi(s),是指该策略 𝝅 下,智能体在时间步 t 和所处状态 s 所能获得的奖励的期望值,即该值是从 s 开始,按照策略 𝝅 执行动作的预期收益

vπ(s)Eπ[GtSt=s]=Eπ[k=0γkRt+k+1St=s]v_\pi(s)\doteq \mathbb{E}_\pi[G_t|S_t=s]=\mathbb{E}_\pi[\sum_{k=0}^{\infin}\gamma^k R_{t+k+1} |S_t=s]

符号 \doteq 表示两个数值、表达式、模型之间的近似等于关系。这个符号通常用于数学和工程领域,在数值计算中,表示两个量非常接近或几乎相等,但并非完全相等。

# 标准动作值函数

一个动作在某个策略下的值函数,记作:qπ(s,a)q_\pi(s,a) ,是指该策略 𝝅 下,智能体在时间步 t 和所处状态 s,执行动作 𝒂 所获得的奖励的期望值,即从 s 开始,按照策略 𝝅 执行动作 𝒂 ,能获得的预期收益

qπ(s,a)Eπ[GtSt=s,At=a]=Eπ[k=0γkRt+k+1St=s,At=a]q_\pi(s,a)\doteq \mathbb{E}_\pi[G_t|S_t=s,A_t=a]=\mathbb{E}_\pi[\sum_{k=0}^{\infin}\gamma^k R_{t+k+1} |S_t=s,A_t=a]

两种值函数的区别在于:

  • 状态值函数相当于将当前状态 s 所有可能执行的动作都执行一遍,求奖励总和的期望
  • 动作值函数相当于只求当前状态 s 下某一个动作 a 的奖励的期望

# V 值的递归计算 (贝尔曼方程)

由目标收益的递归计算可以得到 V 值的递归计算:

V(s)=R(s)+γsSP(ss)V(s)V(s)=R(s)+\gamma\cdot\sum_{s'\in S}P(s'|s)\cdot V(s')

其中:

  • V(s)V(s)ss 状态的 V 值;
  • R(s)R(s)ss 状态的奖励
  • P(ss)P(s'|s) 是从 ss 状态转变为到 ss' 的概率;
  • γ\gamma 是折扣因子。

该式子也就是贝尔曼方程 (?)

# 从值函数计算最优策略

  • 策略:在每一个状态下执行何种动作?
  • 最优策略:能够最大化收益的动作执行决策

# 策略

# 确定性策略

在某一状态下智能体的策略采取动作 a 的概率是 100% ,即 π(s)=a\pi(s)=a

确定性策略

# 不确定性策略(随机策略)

也叫随机策略

π(as)\pi (a|s) 是在状态 ss 上采取的动作的概率分布,也可以表示某一状态 s 采取可能的行为 a 的概率,满足:

aA(s)π(as)=1π(as)0\sum_{a\in \mathcal{A}(s)}\pi (a|s)=1 \\ \pi (a|s)\geq 0

# 值函数的应用

值函数最高值的策略

无终止态MDP的状态值函数求法举例

# 贝尔曼期望方程

贝尔曼方程 (Bellman Equation) 就是递归计算值函数得到的值函数的方程。

贝尔曼期望方程(Bellman Expectation Equation)是为了与接下来的贝尔曼最优方程进行区分

# 状态值函数

v 值

vπ(s)Eπ[GtSt=s]=Eπ[Rt+1+γGt+1St=s]=Eπ[Rt+1]+γEπ[Gt+1St=s]Eπ[Rt+1]=aπ(as)srp(s,rs,a)[r]γEπ[Gt+1St=s]=aπ(as)srp(s,rs,a)[γEπ[Gt+1St+1=s]]=aπ(as)srp(s,rs,a)[γvπ(s)]vπ(s)=aπ(as)srp(s,rs,a)[r+γvπ(s)]\begin{aligned} v_\pi(s) &\doteq \mathbb{E}_\pi[G_t|S_t=s] \\ &= \mathbb{E}_\pi[R_{t+1}+\gamma G_{t+1}|S_t =s] \\ &= \mathbb{E}_\pi[R_{t+1}] + \gamma \mathbb{E}_\pi [G_{t+1}|S_t =s] \\ \mathbb{E}_\pi[R_{t+1}] &=\sum_a \pi (a|s)\sum_{s'}\sum_{r}p(s',r|s,a)[r] \\ \gamma \mathbb{E}_\pi [G_{t+1}|S_t =s] &= \sum_a \pi (a|s)\sum_{s'}\sum_{r}p(s',r|s,a)[\gamma \mathbb{E}_\pi [G_{t+1}|S_{t+1} =s']]\\ &= \sum_a \pi (a|s)\sum_{s'}\sum_{r}p(s',r|s,a)[\gamma v_{\pi}(s')]\\ v_\pi(s) &= \sum_a \pi (a|s)\sum_{s'}\sum_{r}p(s',r|s,a)[r+\gamma v_{\pi}(s')] \end{aligned}

其中 St+1=sS_{t+1}=s'

# 动作值函数

q 值

qπ(s,a)Eπ[GtSt=s,At=a]=srp(s,rs,a)[r+γEπ[Gt+1St+1=s]]=srp(s,rs,a)[r+γaπ(as)Eπ[Gt+1St+1=s,At+1=a]]qπ(s,a)=srp(s,rs,a)[r+γaπ(as)qπ(s,a)]\begin{aligned} q_\pi(s,a) &\doteq \mathbb{E}_\pi[G_t|S_t=s ,A_t=a] \\ &=\sum_{s'}\sum_{r}p(s',r|s,a)[r+\gamma \mathbb{E}_\pi [G_{t+1}|S_{t+1} =s']]\\ &=\sum_{s'}\sum_{r}p(s',r|s,a)[r+\gamma\sum_{a'}\pi (a'|s') \mathbb{E}_\pi [G_{t+1}|S_{t+1} =s',A_{t+1}=a']]\\ q_\pi(s,a) &=\sum_{s'}\sum_{r}p(s',r|s,a)[r+\gamma\sum_{a'}\pi (a'|s') q_{\pi}(s',a')]\\ \end{aligned}

其中 St+1=s,At+1=aS_{t+1}=s',A_{t+1}=a'

# 最优策略的数学表达

sS:π1π2vπ1(s)vπ2(s)\forall s\in \mathcal{S}:\pi_1\geq\pi_2\lrArr v_{\pi_1}(s)\geq v_{\pi_2}(s)

sS,aA:π1π2qπ1(s,a)qπ2(s,a)\forall s\in \mathcal{S},\forall a\in\mathcal{A}:\pi_1\geq\pi_2\lrArr q_{\pi_1}(s,a)\geq q_{\pi_2}(s,a)

从而有:

vπ(s)Eπ[GtSt=s]=maxπvπ(s),sSv_{\pi_*}(s)\doteq\mathbb{E}_{\pi_*}[G_t|S_t=s]=\max_\pi v_{\pi}(s),\forall s\in\mathcal{S}

qπ(s,a)=Eπ[GtSt=s,At=a]=maxπqπ(s,a),sS,aAq_{\pi_*}(s,a)=\mathbb{E}_{\pi_*}[G_{t}|S_{t} =s,A_{t}=a]=\max_\pi q_{\pi}(s,a),\forall s\in\mathcal{S},\forall a\in\mathcal{A}

* 表示最优的,这里 π\pi_* 表示最优策略

# 贝尔曼最优方程

v(s)=vπ(s)?v_*(s)=v_{\pi_*}(s)? 代替原来的 vπ(s)v_{\pi}(s)qπ(s,a)q_{\pi_*}(s,a) 代替原来的 qπ(s,a)q_{\pi}(s,a) ,那么原本的贝尔曼方程就变成贝尔曼最优方程,得到的表达式就是最优值函数

参考:https://blog.csdn.net/november_chopin/article/details/106589197,有 demo 更方便掌握。

# 最优状态值函数

v(s)=a=π(s)π(as)srp(s,rs,a)[r+γv(s)]=maxasrp(s,rs,a)[r+γv(s)]\begin{aligned} v_{*}(s) &= \sum_{a=\pi_{*}(s)} \pi_{*} (a|s)\sum_{s'}\sum_{r}p(s',r|s,a)[r+\gamma v_{*} (s')]\\ &= \max_{a}\sum_{s'}\sum_{r}p(s',r|s,a)[r+\gamma v_{*} (s')] \end{aligned}

# 最优状态值函数

qπ(s,a)=srp(s,rs,a)[r+γa=π(s)π(as)qπ(s,a)]=srp(s,rs,a)[r+γmaxaqπ(s,a)]\begin{aligned} q_{\pi_*}(s,a)&=\sum_{s'}\sum_{r}p(s',r|s,a)[r+\gamma\sum_{a'=\pi_*(s')}\pi_* (a'|s') q_{\pi_*}(s',a')]\\ &=\sum_{s'}\sum_{r}p(s',r|s,a)[r+\gamma \max_{a'} q_{\pi_*}(s',a')] \end{aligned}

# 两种最优值函数的关系

v(s)=maxasrp(s,rs,a)[r+γv(s)]=maxaEπ[Rt+1+γv(St+1)St=s,At=a]=maxaEπ[Rt+1+γGt+1St=s,At=a]=maxaEπ[GtSt=s,At=a]v(s)=maxaA(s)qπ(s,a)\begin{aligned} v_{*}(s) &= \max_{a}\sum_{s'}\sum_{r}p(s',r|s,a)[r+\gamma v_{*} (s')]\\ &= \max_{a}\mathbb{E}_\pi[R_{t+1}+\gamma v_{*}(S_{t+1})|S_t =s,A_t=a]\\ &= \max_{a}\mathbb{E}_{\pi_*}[R_{t+1}+\gamma G_{t+1}|S_t =s,A_t=a]\\ &= \max_{a}\mathbb{E}_{\pi_*}[G_t|S_t =s,A_t=a]\\ v_{*}(s) &= \max_{a\in\mathcal{A(s)}}q_{\pi_{*}}(s,a) \end{aligned}

# TDL

Temporal Difference Learning 差分学习

TD target :

yt=rt+γQ(st+1,at+1;w)=rt+γmaxaQ(st+1,a;w)\begin{aligned} y_t&=r_t+\gamma\cdot Q(s_{t+1},a_{t+1};w)\\ &=r_t+\gamma\cdot \max_aQ(s_{t+1},a;w) \end{aligned}

损失函数:

L=12[Q(w)y]2L=\frac{1}{2}[Q(w)-y]^2

梯度:

Lw=[Q(w)y]Q(w)w\frac{\partial L}{\partial w}=[Q(w)-y]\cdot\frac{\partial Q(w)}{\partial w}

梯度下降计算:

wt+1=wtαLtww=wtw_{t+1}=w_t-\alpha\cdot\frac{\partial L_t}{\partial w}\large|_{w=w_t}

# 算法流程

  1. Observe state St=stS_t=s_t and perform action At=atA_t=a_t
  2. Predict the value: qt=Q(st,at;wt)q_t=Q(s_t,a_t;w_t)
  3. Differentiate the value network: dt=Q(st,at;w)ww=wtd_t=\frac{\partial Q(s_t,a_t;w)}{\partial w}\large|_{w=w_t}
  4. Environment provides new state st+1s_{t+1} and reward rtr_t
  5. Compute TD target: yt=rt+γmaxa𝑄(st+1,a;wt)y_t=r_t+\gamma\cdot\max_a𝑄(s_{t+1},a;w_t)
  6. Gradient descent: wt+1=wtα(qtyt)dtw_{t+1}=w_t-\alpha\cdot(q_t-y_t)\cdot d_t

# 值估计

# Sarsa

State-Action-Reward-State-Action (SARSA)

使用 (st,at,rt,at+1)(s_t,a_t,r_t,a_{t+1}) 更新 QπQ_{\pi}

# 表格

Tabular version

# 神经网络

Value network version

# Q-Learning

# 表格

Tabular version

# 神经网络

DQN version

# 策略

Edited on Views times
\ No newline at end of file +强化学习基础 - 人工智能 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

强烈推荐:https://hrl.boyuai.com/

# 基本概念

RL, in a nutshell, is to “learn to make good sequences of decisions through trail-and-errors”

# 机器学习范式

  • 监督

    • 给定训练数据 training data
    • 和期望的输出(标签)desired outputs (with labels)
  • 半监督

    • 给定训练数据 training data
    • 和部分期望的输出 some labels
  • 无监督

    • 仅给定训练数据 training data
    • without labels
  • 强化学习 Reinforce Learning (RL)

    • 从序列决策中获取奖励 observations and periodic rewards as the agent takes sequential actions;

强化学习是一种自动化目标导向的计算方法,与其他机器学习范式最大的区别在于:

  • 交互:与环境之间存在交互,从环境中学习
  • 决策序列:以前的决策会影响以后的决策,也就是学习的过程
  • 探索:不断试错、探索的学习过程

# 强化学习算法构成

# 给定

# 三大集合

  • 状态集合
  • 动作集合
  • 奖励值集合

# 两大模型 (两大主角)

两大模型

# 环境模型(optional)

述了环境的具体表现。在给定状态和动作情况下,模型可能会预测结果的下一个状态和下一个奖励

  • 有环境模型:定义了环境模型
  • 无环境模型:不定义环境模型,靠智能体感知实时状态来理解环境
# 主角模型(智能体)

概念来源于分布式人工智能思想。通常意义上,智能体是一个能够感知并施加某种作用到自身和环境、有生命周期的一个物理的或抽象的计算实体。智能体接受一个任务指令后,经过学习,可自主完成该任务

# 目标

智能体获取从任务开始到任务结束过程中每一步的最优动作

# 三大特征

  • 探索试错
  • 延迟回报
  • 长期目标

强化学习的关键特征是长期目标优化,智能体关注整体学习问题,优化整个过程

大多强化学习算法的关键特征是值函数的使用

# 特有挑战

探索和开发 (exploration and exploitation):短期和长期目标的权衡

# 强化学习的问题建模

# 问题模型

# 状态空间

智能体所处的实际环境所有状态集合

# 当前状态

智能体感知到的环境当前所处的状态

# 动作策略

智能体在某时间的行为

# 奖励信号

智能体收到的关于动作效果的反馈

# 价值函数

奖励信号 只能说明 动作的即时效果,而 价值函数 则指明什么策略的 长期效果 是好还是差。

# 马尔科夫决策问题模型

# 马尔可夫奖励过程 (MRP)

马尔可夫奖励过程(Markov reward process)

# 动作序列集

动作序列集(Action Sequence Set)是指一组按特定顺序排列的动作序列。它通常用于描述某个系统、机器人或智能体在特定环境中的行为。动作序列集可以包含一系列离散的动作步骤,也可以包含连续的动作序列。

动作序列集可以被分为两类:

  • 有终止态的离散动作(动作幕)
  • 无终止态的连续动作

# 动作幕 (Episode)

一个完整的有起始终止状态动作序列集(Episode,简称动作幕

  • 从标准起始态或从标准分布的初始态集合中抽取初始态开始
  • 以终止态结束
  • 不同动作序列集都可以认为是在同一个终止态结束,只是序列具体元素不同

区分奖励和收益

# 简单收益

Return (aka cumulative future reward) ,未来的累积奖励,也就是简单收益

# 折扣回报

Discounted Return ,折扣回报,也就是引入了折扣因子的简单收益。

# 马尔科夫决策过程 (MDP)

马尔可夫决策过程(Markov decision process,MDP)

MDP 与 MRP 非常相像,主要区别为 MDP 中的状态转移函数和奖励函数都比 MRP 多了动作 aa 作为自变量。

# 问题模型

MDP(马尔科夫决策过程)是一个描述智能体和环境交互的四元组 (S, A, 𝑅, 𝑝)

  • S :Status 状态的集合

  • A :Action 动作的集合

  • R :Reward 奖励的集合

  • p :当前状态执行了一个动作跳转到下一个状态的概率

  • 对象

    • 智能体
    • 环境
  • 交互

    • 动作
    • 状态
    • 奖励

MDP模型

# 状态 - 奖励表

扫地机器人状态 - 奖励表

扫地机器人状态-奖励表

# 状态迁移图

扫地机器人状态迁移图

扫地机器人状态迁移图

# 游戏轨迹

(state, action, reward) trajectory 游戏轨迹

s1,a1,r1,s2,a2,r2,sT,aT,rTs_1,a_1,r_1,s_2,a_2,r_2,\cdots s_T,a_T,r_T

即时奖励和长期目标的关系

某一决策的即时奖励较高不一定带来较高的长期目标

追求长期目标 ≠ 追求即时奖励

# 强化学习的值函数

# 基础概统知识

以下定义针对离散型随机变量:

# 概率

随机变量 A = a 发生的概率

p(A=a)p(A=a)

# 条件概率

在 C = c 条件下,A = a 的概率

p(A=aC=c)p(A=a|C=c)

智能体处于状态 s(S=s)s(S=s) 时,它的策略 π\pi 执行动作 a(A=a)a(A=a) 的概率

π(as)\pi(a|s)

# 期望

也称随机变量 𝑿 的期待值

E[X]=ixip(X=xi)\mathbb{E}[X]=\sum_i x_i p(X=x_i)

# 条件期望

Y=y 条件下,X 的期望

E[XY=y]=ixip(X=xiY=y)\mathbb{E}[X|Y=y]=\sum_i x_i p (X=x_i |Y=y)

# 简单值函数

直觉上看,为找到一个好的动作决策,需为每个动作赋值,只考虑了即时奖励,未考虑长期过程

有了值函数 q (a) ,决策问题即可转为优化问题求解,找最优动作:

argmaxaq(a)\underset{a}{argmax} \ q(a)

# 定义

q(a)=E[RA=a]a{1,...,k}=rp(ra)rq(a)=\mathbb{E}[R|A=a]\forall a\in \{ 1,...,k \}=\sum_r p(r|a)r

其中 p (r | a) 是在执行 a 动作情况下观察到智能体获得奖励 r 的概率;R 字母表示 Reward 奖励。

# 采样平均

除了按定义方法计算,还可以通过采样平均方法计算,即通过以往动作所获奖励情况来估算值函数

Q(a)=获得奖励的总和动作a的执行次数Q(a)=\frac{\text{获得奖励的总和}}{\text{动作a的执行次数}}

具体计算公式为:

Q(a)=i=1n1Ri1Ai=ai=1n11Ai=aQ(a)=\frac{\sum_{i=1}^{n-1}R_i \mathbb{1}_{A_i = a}}{\sum_{i=1}^{n-1}\mathbb{1}_{A_i = a}}

其中 1Ai=a\mathbb{1}_{A_i = a} 表示 Ai=aA_i =a 时取 1 ,否则取 0

# 问题

未考虑长期过程,实际并不采用

# 简单收益

# 简单收益的定义

G 字母表示 Gain 收益,也叫做回报 Return

Gt=Rt+1+Rt+2+Rt+3+...+RTG_t= R_{t+1}+R_{t+2}+R_{t+3}+...+R_T

# 简单收益的期望

# 离散动作的简单收益

E[Gt]=E[Rt+1+Rt+2+Rt+3+...+RT]\mathbb{E}[G_t]=\mathbb{E}[R_{t+1}+R_{t+2}+R_{t+3}+...+R_T]

# 连续动作的简单收益

E[Gt]=E[Rt+1+Rt+2+Rt+3+...]\mathbb{E}[G_t]=\mathbb{E}[R_{t+1}+R_{t+2}+R_{t+3}+...]

# 存在问题

若奖励都是非负的,收益会不会是无穷大?

# 折扣目标收益

引入折扣因子 γ\gammaγ<1\gamma \lt 1,得到折扣回报(Discounted Return):

Gt=Rt+1+γRt+2+γ2Rt+3+...+γk1Rt+k+...=k=0γkRt+k+1G_t= R_{t+1}+\gamma R_{t+2}+\gamma^2 R_{t+3}+...+ \gamma^{k-1}R_{t+k}+...=\sum_{k=0}^{\infin}\gamma^k R_{t+k+1}

γ=0\gamma =0 时:

Gt=Rt+1+0Rt+2+02Rt+3+...+0k1Rt+k+...=Rt+1G_t= R_{t+1}+0 R_{t+2}+0^2 R_{t+3}+...+ 0^{k-1}R_{t+k}+...=R_{t+1}

RmaxR_{max} 为智能体所能获取的最大奖励,则:

Rt+k+1RmaxR_{t+k+1}\leq R_{max}

Gt=k=0γkRt+k+1k=0γkRmax=Rmaxk=0γk=Rmax1γG_t = \sum_{k=0}^{\infin}\gamma^k R_{t+k+1} \leq \sum_{k=0}^{\infin}\gamma^k R_{max} = R_{max}\sum_{k=0}^{\infin}\gamma^k = \frac{R_{max}}{1-\gamma}

# 目标收益的递归计算

定义 GT=0,tTG_T = 0\ ,\ t\leq T

Gt=Rt+1+γRt+2+γ2Rt+3+...+γk1Rt+k+...=Rt+1+γ(Rt+2+γRt+3+...+γk2Rt+k+...)=Rt+1+γGt+1\begin{aligned} G_t &= R_{t+1}+\gamma R_{t+2}+\gamma^2 R_{t+3}+...+ \gamma^{k-1}R_{t+k}+... \\ &= R_{t+1}+\gamma (R_{t+2}+\gamma R_{t+3}+...+ \gamma^{k-2}R_{t+k}+...) \\ &= R_{t+1}+\gamma G_{t+1} \end{aligned}

从而有:

Gt=Rt+1+γGt+1G_t =R_{t+1}+\gamma G_{t+1}

特别地,t = T 时有 GT=0G_T = 0 ;t = T-1 时有 Gt1=RT+γGTG_{t-1} = R_T +\gamma G_T

# 标准状态值函数

一个状态在策略 𝝅 下的值函数,记作:vπ(s)v_\pi(s),是指该策略 𝝅 下,智能体在时间步 t 和所处状态 s 所能获得的奖励的期望值,即该值是从 s 开始,按照策略 𝝅 执行动作的预期收益

vπ(s)Eπ[GtSt=s]=Eπ[k=0γkRt+k+1St=s]v_\pi(s)\doteq \mathbb{E}_\pi[G_t|S_t=s]=\mathbb{E}_\pi[\sum_{k=0}^{\infin}\gamma^k R_{t+k+1} |S_t=s]

符号 \doteq 表示两个数值、表达式、模型之间的近似等于关系。这个符号通常用于数学和工程领域,在数值计算中,表示两个量非常接近或几乎相等,但并非完全相等。

# 标准动作值函数

一个动作在某个策略下的值函数,记作:qπ(s,a)q_\pi(s,a) ,是指该策略 𝝅 下,智能体在时间步 t 和所处状态 s,执行动作 𝒂 所获得的奖励的期望值,即从 s 开始,按照策略 𝝅 执行动作 𝒂 ,能获得的预期收益

qπ(s,a)Eπ[GtSt=s,At=a]=Eπ[k=0γkRt+k+1St=s,At=a]q_\pi(s,a)\doteq \mathbb{E}_\pi[G_t|S_t=s,A_t=a]=\mathbb{E}_\pi[\sum_{k=0}^{\infin}\gamma^k R_{t+k+1} |S_t=s,A_t=a]

两种值函数的区别在于:

  • 状态值函数相当于将当前状态 s 所有可能执行的动作都执行一遍,求奖励总和的期望
  • 动作值函数相当于只求当前状态 s 下某一个动作 a 的奖励的期望

# V 值的递归计算 (贝尔曼方程)

由目标收益的递归计算可以得到 V 值的递归计算:

V(s)=R(s)+γsSP(ss)V(s)V(s)=R(s)+\gamma\cdot\sum_{s'\in S}P(s'|s)\cdot V(s')

其中:

  • V(s)V(s)ss 状态的 V 值;
  • R(s)R(s)ss 状态的奖励
  • P(ss)P(s'|s) 是从 ss 状态转变为到 ss' 的概率;
  • γ\gamma 是折扣因子。

该式子也就是贝尔曼方程 (?)

# 从值函数计算最优策略

  • 策略:在每一个状态下执行何种动作?
  • 最优策略:能够最大化收益的动作执行决策

# 策略

# 确定性策略

在某一状态下智能体的策略采取动作 a 的概率是 100% ,即 π(s)=a\pi(s)=a

确定性策略

# 不确定性策略(随机策略)

也叫随机策略

π(as)\pi (a|s) 是在状态 ss 上采取的动作的概率分布,也可以表示某一状态 s 采取可能的行为 a 的概率,满足:

aA(s)π(as)=1π(as)0\sum_{a\in \mathcal{A}(s)}\pi (a|s)=1 \\ \pi (a|s)\geq 0

# 值函数的应用

值函数最高值的策略

无终止态MDP的状态值函数求法举例

# 贝尔曼期望方程

贝尔曼方程 (Bellman Equation) 就是递归计算值函数得到的值函数的方程。

贝尔曼期望方程(Bellman Expectation Equation)是为了与接下来的贝尔曼最优方程进行区分

# 状态值函数

v 值

vπ(s)Eπ[GtSt=s]=Eπ[Rt+1+γGt+1St=s]=Eπ[Rt+1]+γEπ[Gt+1St=s]Eπ[Rt+1]=aπ(as)srp(s,rs,a)[r]γEπ[Gt+1St=s]=aπ(as)srp(s,rs,a)[γEπ[Gt+1St+1=s]]=aπ(as)srp(s,rs,a)[γvπ(s)]vπ(s)=aπ(as)srp(s,rs,a)[r+γvπ(s)]\begin{aligned} v_\pi(s) &\doteq \mathbb{E}_\pi[G_t|S_t=s] \\ &= \mathbb{E}_\pi[R_{t+1}+\gamma G_{t+1}|S_t =s] \\ &= \mathbb{E}_\pi[R_{t+1}] + \gamma \mathbb{E}_\pi [G_{t+1}|S_t =s] \\ \mathbb{E}_\pi[R_{t+1}] &=\sum_a \pi (a|s)\sum_{s'}\sum_{r}p(s',r|s,a)[r] \\ \gamma \mathbb{E}_\pi [G_{t+1}|S_t =s] &= \sum_a \pi (a|s)\sum_{s'}\sum_{r}p(s',r|s,a)[\gamma \mathbb{E}_\pi [G_{t+1}|S_{t+1} =s']]\\ &= \sum_a \pi (a|s)\sum_{s'}\sum_{r}p(s',r|s,a)[\gamma v_{\pi}(s')]\\ v_\pi(s) &= \sum_a \pi (a|s)\sum_{s'}\sum_{r}p(s',r|s,a)[r+\gamma v_{\pi}(s')] \end{aligned}

其中 St+1=sS_{t+1}=s'

# 动作值函数

q 值

qπ(s,a)Eπ[GtSt=s,At=a]=srp(s,rs,a)[r+γEπ[Gt+1St+1=s]]=srp(s,rs,a)[r+γaπ(as)Eπ[Gt+1St+1=s,At+1=a]]qπ(s,a)=srp(s,rs,a)[r+γaπ(as)qπ(s,a)]\begin{aligned} q_\pi(s,a) &\doteq \mathbb{E}_\pi[G_t|S_t=s ,A_t=a] \\ &=\sum_{s'}\sum_{r}p(s',r|s,a)[r+\gamma \mathbb{E}_\pi [G_{t+1}|S_{t+1} =s']]\\ &=\sum_{s'}\sum_{r}p(s',r|s,a)[r+\gamma\sum_{a'}\pi (a'|s') \mathbb{E}_\pi [G_{t+1}|S_{t+1} =s',A_{t+1}=a']]\\ q_\pi(s,a) &=\sum_{s'}\sum_{r}p(s',r|s,a)[r+\gamma\sum_{a'}\pi (a'|s') q_{\pi}(s',a')]\\ \end{aligned}

其中 St+1=s,At+1=aS_{t+1}=s',A_{t+1}=a'

# 最优策略的数学表达

sS:π1π2vπ1(s)vπ2(s)\forall s\in \mathcal{S}:\pi_1\geq\pi_2\lrArr v_{\pi_1}(s)\geq v_{\pi_2}(s)

sS,aA:π1π2qπ1(s,a)qπ2(s,a)\forall s\in \mathcal{S},\forall a\in\mathcal{A}:\pi_1\geq\pi_2\lrArr q_{\pi_1}(s,a)\geq q_{\pi_2}(s,a)

从而有:

vπ(s)Eπ[GtSt=s]=maxπvπ(s),sSv_{\pi_*}(s)\doteq\mathbb{E}_{\pi_*}[G_t|S_t=s]=\max_\pi v_{\pi}(s),\forall s\in\mathcal{S}

qπ(s,a)=Eπ[GtSt=s,At=a]=maxπqπ(s,a),sS,aAq_{\pi_*}(s,a)=\mathbb{E}_{\pi_*}[G_{t}|S_{t} =s,A_{t}=a]=\max_\pi q_{\pi}(s,a),\forall s\in\mathcal{S},\forall a\in\mathcal{A}

* 表示最优的,这里 π\pi_* 表示最优策略

# 贝尔曼最优方程

v(s)=vπ(s)?v_*(s)=v_{\pi_*}(s)? 代替原来的 vπ(s)v_{\pi}(s)qπ(s,a)q_{\pi_*}(s,a) 代替原来的 qπ(s,a)q_{\pi}(s,a) ,那么原本的贝尔曼方程就变成贝尔曼最优方程,得到的表达式就是最优值函数

参考:https://blog.csdn.net/november_chopin/article/details/106589197,有 demo 更方便掌握。

# 最优状态值函数

v(s)=a=π(s)π(as)srp(s,rs,a)[r+γv(s)]=maxasrp(s,rs,a)[r+γv(s)]\begin{aligned} v_{*}(s) &= \sum_{a=\pi_{*}(s)} \pi_{*} (a|s)\sum_{s'}\sum_{r}p(s',r|s,a)[r+\gamma v_{*} (s')]\\ &= \max_{a}\sum_{s'}\sum_{r}p(s',r|s,a)[r+\gamma v_{*} (s')] \end{aligned}

# 最优状态值函数

qπ(s,a)=srp(s,rs,a)[r+γa=π(s)π(as)qπ(s,a)]=srp(s,rs,a)[r+γmaxaqπ(s,a)]\begin{aligned} q_{\pi_*}(s,a)&=\sum_{s'}\sum_{r}p(s',r|s,a)[r+\gamma\sum_{a'=\pi_*(s')}\pi_* (a'|s') q_{\pi_*}(s',a')]\\ &=\sum_{s'}\sum_{r}p(s',r|s,a)[r+\gamma \max_{a'} q_{\pi_*}(s',a')] \end{aligned}

# 两种最优值函数的关系

v(s)=maxasrp(s,rs,a)[r+γv(s)]=maxaEπ[Rt+1+γv(St+1)St=s,At=a]=maxaEπ[Rt+1+γGt+1St=s,At=a]=maxaEπ[GtSt=s,At=a]v(s)=maxaA(s)qπ(s,a)\begin{aligned} v_{*}(s) &= \max_{a}\sum_{s'}\sum_{r}p(s',r|s,a)[r+\gamma v_{*} (s')]\\ &= \max_{a}\mathbb{E}_\pi[R_{t+1}+\gamma v_{*}(S_{t+1})|S_t =s,A_t=a]\\ &= \max_{a}\mathbb{E}_{\pi_*}[R_{t+1}+\gamma G_{t+1}|S_t =s,A_t=a]\\ &= \max_{a}\mathbb{E}_{\pi_*}[G_t|S_t =s,A_t=a]\\ v_{*}(s) &= \max_{a\in\mathcal{A(s)}}q_{\pi_{*}}(s,a) \end{aligned}

# TDL

Temporal Difference Learning 差分学习

TD target :

yt=rt+γQ(st+1,at+1;w)=rt+γmaxaQ(st+1,a;w)\begin{aligned} y_t&=r_t+\gamma\cdot Q(s_{t+1},a_{t+1};w)\\ &=r_t+\gamma\cdot \max_aQ(s_{t+1},a;w) \end{aligned}

损失函数:

L=12[Q(w)y]2L=\frac{1}{2}[Q(w)-y]^2

梯度:

Lw=[Q(w)y]Q(w)w\frac{\partial L}{\partial w}=[Q(w)-y]\cdot\frac{\partial Q(w)}{\partial w}

梯度下降计算:

wt+1=wtαLtww=wtw_{t+1}=w_t-\alpha\cdot\frac{\partial L_t}{\partial w}\large|_{w=w_t}

# 算法流程

  1. Observe state St=stS_t=s_t and perform action At=atA_t=a_t
  2. Predict the value: qt=Q(st,at;wt)q_t=Q(s_t,a_t;w_t)
  3. Differentiate the value network: dt=Q(st,at;w)ww=wtd_t=\frac{\partial Q(s_t,a_t;w)}{\partial w}\large|_{w=w_t}
  4. Environment provides new state st+1s_{t+1} and reward rtr_t
  5. Compute TD target: yt=rt+γmaxa𝑄(st+1,a;wt)y_t=r_t+\gamma\cdot\max_a𝑄(s_{t+1},a;w_t)
  6. Gradient descent: wt+1=wtα(qtyt)dtw_{t+1}=w_t-\alpha\cdot(q_t-y_t)\cdot d_t

# 值估计

# Sarsa

State-Action-Reward-State-Action (SARSA)

使用 (st,at,rt,at+1)(s_t,a_t,r_t,a_{t+1}) 更新 QπQ_{\pi}

# 表格

Tabular version

# 神经网络

Value network version

# Q-Learning

# 表格

Tabular version

# 神经网络

DQN version

# 策略

Edited on Views times
\ No newline at end of file diff --git a/cs/ai/search/index.html b/cs/ai/search/index.html index bebc6d98..9d250de3 100644 --- a/cs/ai/search/index.html +++ b/cs/ai/search/index.html @@ -1,4 +1,4 @@ -搜索技术 - 人工智能 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

思维导图

# 搜索相关概念

对于给定的问题,智能系统的行为一般是找到能够达到所希望目标的动作序列,并使其所付出的代价最小、性能最好。

基于给定的问题,问题求解的第一步是目标的表示。

搜索就是找到智能系统的动作序列的过程

搜索算法的:

  • 输入是给定的问题
    • 搜索什么(目标)
    • 在哪里搜索(搜索空间)
      • 和通常的搜索空间不同,人工智能中大多数问题的状态空间在问题求解之前不是全部知道的
  • 输出是表示为动作序列的方案。

一旦有了方案,就可以执行该方案所给出的动作了。(执行阶段)

求解问题包括:

  • 目标表示
  • 搜索
  • 执行

# 搜索的分类

根据是否使用启发式信息分为

  • 盲目搜索
    • 只是可以区分出哪个是目标状态
    • 一般是按预定的搜索策略进行搜索
    • 没有考虑到问题本身的特性,这种搜索具有很大的盲目性,效率不高,不便于复杂问题的求解
  • 启发式搜索
    • 是在搜索过程中加入了与问题有关的启发式信息,用于指导搜索朝着最有希望的方向前进,加速问题的求解并找到最优解。

按表示方式分为

  • 状态空间搜索
    • 用状态空间法来求解问题所进行的搜索
  • 与或树搜索
    • 用问题规约方法来求解问题时所进行的搜索

# 搜索策略评价标准

# 完备性

如果存在一个解答,该策略是否保证能够找到?

# 时间复杂性

需要多长时间可以找到解答?

# 空间复杂性

执行搜索需要多少存储空间?

# 最优性

如果存在不同的几个解答,该策略是否可以发现最高质量的解答?

# 搜索控制策略

  • 不可撤回的控制策略 —— 可能无解
  • 试探性控制策略
    • 回溯型
    • 图搜索

# 盲目搜索方法

定义一个四元组,以此来表示状态空间:

w raw
{ nodes ,arc ,goal ,current }
  • nodes 表示当前搜索空间中现有状态的集合
  • arc 表示可应用与当前状态的操作符,把当前状态转换为另一个状态
  • goal 表示需要到达的状态,是 nodes 中的一个状态
  • current 表示现在生成的用于和目标状态比较的状态(包括初始状态)

各种算法的完备性、时间复杂度、空间复杂度、能否找到最优解的比较如下:

盲目搜索算法评价

# 宽度优先搜索

优点:

目标节点如果存在,用宽度优先搜索算法总可以找到该目标节点,而且是最小(即最短路径)的节点。

问题:

宽度优先搜索是一种盲目搜索,时间和空间复杂度都比较高,当目标节点距离初始节点较时会产生许多无用的节点,搜索效率低。

宽度优先搜索中,时间需求是一个很大的问题,特别是当搜索的深度比较大时,尤为严重,但是空间需求是比执行时间更严重的问题。

# 深度优先搜索

也称为回溯搜索,它总是首先扩展树的最深层次上的某个节点,只是当搜索遇到一个死亡节点(非目标节点而且不可扩展),搜索方法才会返回并扩展浅层次的节点

上述原理对树中的每一节点是递归实现的 (实现该递归用栈)

优点:

是比宽度优先搜索算法需要较少的空间,该算法只需要保存搜索树的一部分,它由当前正在搜索的路径和该路径上还没有完全展开的节点标志所组成

深度优先搜索的存储器要求深度约束的线性函数

# 有界深度优先搜索

Deep-limited Search

若状态空间无限,DFS 可能会出现循环,搜索失败;通过预定一个深度限制来解决这个问题

有界深度优先搜索过程总体上按深度优先算法方法进行,但对搜索深度需要给出一个深度限制 dmd_m ,当深度达到了 dmd_m 的时候,如果还没有找到解答,就停止对该分支的搜索,换到另外一个分支进行搜索

深度限制 dmd_m 很重要。当问题有解,且解的路径长度小于或等于 dmd_m 时,则搜索过程一定能够找到解,但是和深度优先搜索一样这并不能保证最先找到的是最优解

  • 但是当 dmd_m 取得太小,解的路径长度大于 dmd_m 时,则搜索过程中就找不到解,即这时搜索过程甚至是不完备的
  • dmd_m 不能太大。当 dmd_m 太大时,搜索过程会产生过多的无用节点,既浪费了计算机资源,又降低了搜索效率

有界深度搜索的主要问题是深度限制值 dmd_m 的选取

# 迭代加深搜索

Iterative Deepening Search

在有界深度优先搜索的基础上改进,试图尝试所有可能的深度限制

先任意给定一个较小的数作为 dmd_m,然后按有界深度算法搜索,通过典型的深度优先算法,生成深度为 m 的树:

  • 若在此深度限制内找到了解,则算法结束
  • 若在此限制内没有找到问题的解,则增大深度限制 dmd_m ,继续搜索
  • 宽度优先搜索需要指数数量的空间
  • 深度优先搜索的空间复杂度和最大搜索深度呈线性关系
  • 迭代加深搜索对一棵深度受控的树采用深度优先的搜索。它结合了宽度优先和深度优先搜索的优点:
    • 和宽度优先搜索一样,它是最优的,也是完备的
    • 但对空间要求和深度优先搜索一样是适中的

# 双向搜索

Bidirectional search

同时进行两个搜索:

  • 一个是从初始状态向前搜索;
  • 另一个则从目标向后搜索;

当两者在中间相遇时停止搜索,在一定程度上能减小复杂度

# 启发式搜索

启发式搜索用于两种不同类型的问题:

  • 前向推理
    • 一般用于状态空间的搜索。在前向推理中,推理是从预选定义的初始状态出发向目标状态方向执行
  • 反向推理
    • 反向推理一般用于问题规约中。在反向推理中,推理是从给定的目标状态向初始状态执行

启发式搜索:如果在选择节点时能充分利用与问题有关的特征信息,估计出节点的重要性,就能在搜索时选择重要性较高的节点,以便求得最优解。

# 启发性信息和评估函数

# 评估函数

用来评估节点重要性的函数称为评估函数 f(x)f(x) ,定义为从初始节点 S0S_0 出发,约束地经过节点 x 到达目标节点 SgS_g 的所有路径中最小路径代价的估计值:

f(x)=g(x)+h(x)f(x)=g(x)+h(x)

  • g(x)g(x)—— 从初始节点 S0S_0 到节点 xx实际代价

  • h(x)h(x)—— 从 x 到目标节点 SgS_g 的最优路径的评估代价,它体现了问题的启发式信息,其形式要根据问题的特性确定,h(x)h(x) 称为启发式函数

  • 在正方形网格中,允许向 4 邻域的移动,使用曼哈顿距离(L1L1

  • 在正方形网格中,允许向 8 邻域的移动,使用对角线距离(LL∞)等等

# OR 图

大多数前向推理问题可以表示为 OR 图,其中图中的节点表示问题的状态,弧表示应用于当前状态的规则,该规则引起状态的转换;

当有多个规则可用于当前状态的时候,可以从该状态的各个子状态中选择一个比较好的状态作为下一个状态。

# 最好优先搜索算法

从最有希望的节点开始,并且生成其所有的子节点

  • 计算每个节点的性能(合适性)
  • 选择最有希望的节点进行扩展,而不是仅仅从当前节点所生成的子节点中进行选择
  • 如果在早期选择了一个错误的节点,最好优先搜索就提供了一个修改的机会

最好优先搜索算法并没有显式地给出如何定义启发式函数,

它不能保证当从起始节点到目标节点的最短路径存在时,一定能够找到它。

# 通用图搜索算法

图搜索算法只记录状态空间中那些被搜索过的状态,它们组成一个搜索图 G。

G 由两种节点组成:

  • Open 节点,如果该节点已经生成,而且启发式函数值 h(x) 已经计算出来,但是它没有扩展。这些节点也称为未考察节点
  • Closed 节点,如果该节点已经扩展并生成了其子节点。Closed 节点是已经考察过的节点。

可以给出两个数据结构 OPEN 和 CLOSED 表,分别存放了 Open 节点和 Closed 节点。

节点 x 总的费用函数 f (x) 是 g (x) 和 h (x) 之和。

生成费用 g (x) 可以比较容易地得到,如, 如果节点 x 是从初始节点经过 m 步得到,则 g (x) 应该和 m 成正比(或者就是 m)。

h (x) 只是一个预测值。

上述图搜索算法生成一个明确的图 G(称为搜索图)和一个 G 的子集 T(称为搜索树),图 G 中的每一个节点也在树 T 上。

搜索树是由返回指针来确定的。

G 中的每一个节点(除了初始节点 S0)都有一个指向 G 中一个父辈节点的指针。该父辈节点就是树中那个节点的惟一父辈节点。

最好优先搜索算法伪代码:

最好优先搜索算法伪代码

算法的特例:

f(n)=g(n)+h(n)f(n)=g(n)+h(n)

  • 当 g (n)=0 时,f (n)=h (n),称为贪婪最好优先搜索算法
  • 当 h (n)=0 时,f (n)=g (n),变成宽度优先搜索算法

# A 算法和 A * 算法

如果最好优先搜索算法中的 f (n) 被实例化为 f (n)=g (n)+h (n),则称为 A 算法。

进一步细化,如果启发函数 h 满足对于任一结点 n,h (n) 的值都不大于 n 到目标结点的最优代价,则称此类 A 算法为 A * 算法。A * 算法在一些条件下能够保证找到最优解。

在图搜索策略的基础上,A * 算法可以看作是 BFS 算法的升级版

评估函数 ff^*

f(n)=g(n)+h(n)f^*(n)=g^*(n)+h^*(n)

  • g(n)g^*(n) 为起始节点到节点 n 的最短路径的代价
  • h(n)h^*(n) 是从 n 到目标节点的最短路径的代价。

这样 f(n)f^*(n) 就是从起始节点出发通过节点 n 到达目标节点的最佳路径的总代价的估值。

把估价函数 f(n)f(n)f(n)f^*(n) 相比较,g(n)g^*(n) 是对 g(n)g(n) 的估价;h(n)h^*(n) 是对 h(n)h(n) 的估价。

在这两个估价中,尽管 g(n)g(n) 容易计算,但它不一定就是从起始节点 S0S_0 到节点 n 的真正的最短路径的代价,很可能从初始节点 S0S_0 到节点 n 的真正最短路径还没有找到,所以一般都有:

g(n)g(n)g(n)\geq g^*(n)

有了g(n)g^*(n)h(n)h^*(n) 的定义,如果对最好优先的启发式搜索算法中的 g(n)g(n)h(n)h(n) 做如下的限制:

  • g(n)g(n) 是对 g(n)g^*(n) 估计,且 g(n)g(n) >0
  • h(n)h(n)h(n)h^*(n) 的下界,即对任意节点 n 均有 h(n)h(n)h(n)\leq h^*(n)
  • (这个限制保证 A * 算法能够找到最优解)

则称这样得到的算法为 A * 算法

# IDA * 算法

也叫迭代加深 A * 算法

  • 迭代加深搜索算法,它以深度优先的方式在有限制的深度内搜索目标节点。在每个深度上,该算法在每个深度上检查目标节点是否出现,如果出现则停止,否则深度加 1 继续搜索。
  • 而 A * 算法是选择具有最小估价函数值的节点扩展。

迭代加深 A * 搜索算法 是 上述两种算法的结合,这里启发式函数用做深度的限制,而不是选择扩展节点的排序

伪代码:

IDA_star伪代码

# 问题归约和 AND-OR 图启发式搜索

# 问题归约的描述

问题归约可以用三元组表示:(S0OP)(S_0,O,P) ,其中:

  • S0S_0 是初始问题
  • PP 是本原问题集 (不用证明的,公理、已知事实,或已证明过的问题)
  • OO 是操作算子集 (操作算子把一个问题化成若干个子问题)

问题归约表示方法就是由初始问题出发,运用操作算子产生一些子问题,对子问题再运用操作算子产生子问题的子问题,这样一直进行到产生的问题均为本原问题,则问题得解。所有问题归约的最终目的是产生本原问题

# ⭐️AND-OR 图

# 超图

超图

q 指向 p 和 t 的两条有向边被一个圆弧连接,用于表示 q 被分解 (归约) 为 p 与 t : 只有当 p 和 t 对应的问题都被解决时,q 才能被解决。把圆弧连接的有向边看作 一个整体,把有向边 <q,m> 看作另一个整体,这两个整体表示可以将 q 按照前一个整体进行分解,或者,将 q 按照后一个整体进行分解

超图 (hypergrah) 用二元组 (N,H)(N,H) 表示,其中 :

  • N 为结点的有穷集合
  • H 为 “超边” (Hyperarce) 的集合

一个超边表示为 <s,D><s,D> ,也称为 “k 连接符”(k-coNnN_nector),其中 k=Dk=|D| 其中 :

  • sNs\in N ,称 s 为该超边的源结点
  • DND\in N ,称 D 该超边的目的结点集。每个超边 <s,D><s,D> 都表示结点 s 对应的问题的一个可行的分 解方法.
    • D=1|D|=1 ,则该超边称为 “或弧”,同时称 D 中的结点为 s 的 “或子结点”(OR-node),也称它为 s 的 “或后继”(OR-descendents),也就是或节点
    • D>1|D|>1 ,则该超边称为 “与弧”,同时称 D 中的结点为 s 的 “与子结点”(AND-node),也称它们为 s 的 “与后继”(AND-descendents),也就是与节点

# 与或图 (AND-OR)

用 AND-OR 图把问题归约为子问题替换集合,将:

  • 初始节点表示初始问题描述
  • 对应于本原问题的节点称为叶节点

它的四元组表示为 (N,n0,H,T)(N,n_0,H,T) ,其中:

  • N 是结点集合,其中每个结点都对应一个唯一的问题
  • n0Nn_0\in N ,对应于初始问题
  • H 是超边的集合,其中每个超边 <S,D> 都表示结点 s 对应的问题的一个可行的分解方法
    • 若 | D|=1,则该超边称为 “或弧”,同时称 D 中的结点为;的 “或子结点”(OR-node),也称它为 s 的 “或后继”(OR-descendents),也就是或节点
    • 若 | D|>1,则该超边称为 “与弧”,同时称 D 中的结点为;的 “与子结点”(AND-node),也称它们力;的 “与后继”(AND-descendents),也就是与节点
  • T 是 N 的子集,其中每个结点对应的问题都为本原问题,T 中的结点也称为叶结点

# 可解节点

可解节点可递归地定义如下:

  • 叶节点是可解节点
  • 如果某节点为或子节点,那么该节点可解当且仅当至少有一个子节点为可解节点
  • 如果某节点为与子节点,那么该节点可解当且仅当所有子节点均为可解节点

# 不可解节点

不可解节点可递归定义如下:

  • 有后裔节点的非叶节点是不可解节点
  • 或节点是不可解节点(含有或后继节点),当且仅当它的所有子节点都是不可解节点
  • 与节点是不可解节点(含有与后继节点),当且仅当它的子节点中至少有一个是不可解节点。

# 解图

能导致初始节点可解的那些可解节点及有关连线组成的子图称为该 AND-OR 图的解图

# 博弈

这里讲的博弈是二人博弈,二人零和、全信息、非偶然博弈,博弈双方的利益是完全对立的;

(1)对垒的双方 MAX 和 MIN 轮流采取行动,博弈的结果只能有 3 种情况:MAX 胜、MIN 败;MAX 败,MIN 胜;和局。

(2)在对垒过程中,任何一方都了解当前的格局和过去的历史。

(3)任何一方在采取行动前都要根据当前的实际情况,进行得失分析,选择对自己最为有利而对对方最不利的对策,在不存在 “碰运气” 的偶然因素,即双方都很理智地决定自己的行动。

# 博弈树

把双人博弈过程用图的形式表示出来,这样就可以得到一棵 AND-OR 树,这种 AND-OR 树称为博弈树;

博弈树中的节点分为:

  • MAX 节点 —— 下一步该 MAX 走的节点,所有 MAX 节点都是或节点
  • MIN 节点 —— 下一步该 MIN 走的节点,所有 MIN 节点都是与节点

博弈树特点:

  • 博弈的初始状态是初始节点
  • 博弈树的 “与” 节点 和 “或” 节点 是逐层交替出现的
  • 整个博弈过程始终站在某一方的立场上,所以能使自己一方获胜的终局都是本原问题,相应的节点也是可解节点,所有使对方获胜的节点都是不可解节点

# 极大极小过程

极大极小过程是考虑双方对弈若干步之后,从可能的走法中选一步相对好的走法来走,即在有限的搜索深度范围内进行求解

定义一个静态估价函数 ff , 以便对棋局的态势做出评估,这个函数可以根据棋局的态势特征进行定义。假定对弈双方分别为 MAX 和 MIN ,规定:

  • 有利于 MAX 方的态势:f(p)f(p) 取正值
  • 有利于 MIN 方的态势:f(p)f(p) 取负值
  • 态势均衡的时候:f(p)f(p) 取零

其中 pp 代表棋局

MINMAX 基本思想:

  1. 当轮到 MIN 走步的节点时,MIN 应考虑最坏的情况(即 f(p)f(p) 取极小值)
  2. 当轮到 MAX 走步的节点时,MAX 应考虑最好的情况(即 f(p)f(p) 取极大值)
  3. 评价往回倒推时,相应于两位棋手的对抗策略,交替使用(1)和(2)两种方法传递倒推值

所以这种方法称为极大极小过程

# 井字棋

设只进行两层,即每方只走一步

估价函数 e(p)e(p)

  • 若格局 p 对任何一方都不是获胜的,则:e(p)e(p) = (所有空格都放上 MAX 的棋子之后三子成一线的总数) –(所有空格都放上 MIN 的棋子后三子成一线的总数)
  • 若 p 是 MAX 获胜,则:e(p)=+e(p) = +∞
  • 若 p 是 MIN 获胜,则:e(p)=e(p) = -∞

井字棋

在生成后继节点时,可以利用棋盘的对称性,省略了从对称上看是相同的格局

极大极小方法是把搜索树的生成和估值这两个过程完全分开,只有在已经生成树之后才开始进行估值,这一分离导致了低效率的策略

# α\alpha-β\beta 过程

α\alpha-β\beta 剪枝技术是极大极小方法的改进,是一种提高博弈树搜索效率的方法

α\alpha-β\beta 剪枝技术是一种边生成节点,边计算估值和倒推值的方法,从而剪去某些分枝

  • 对于一个 “与”(MIN)节点,它取当前子节点中的最小的倒推值作为它的倒推值的上界,称此值为 β\beta 值。也就是说 β\beta 值可以等于其后继节点当前最小的最终倒推值.“与” 节点的 β\beta 值是永远不会增加的
  • 对于一个 “或”(MAX)节点,它取当前子节点中得最大的倒推值作为它的倒推值的下界,称此值为 α\alpha 值。也就是说 α\alpha 值可以等于其后继节点当前最大的最终倒推值.“或” 节点的 α\alpha 值是永远不会减少的.

β\beta 剪枝:任何 “或”(MAX)节点 x 的 α\alpha 值如果大于等于其父节点的 β\beta 值,则:

  • 对节点 x 以下的分枝可以停止搜索
  • 并使 x 的倒推值为 α\alpha

α\alpha 剪枝:任何 “与”(MIN)节点 x 的 β\beta 值如果小于等于其父节点的 α\alpha 值,则:

  • 对节点 x 以下的分枝可以停止搜索
  • 并使 x 的倒推值为 β\beta

要进行 α\alpha-β\beta 剪枝,至少必须使某一部分的搜索树生长到最大深度。因为 α\alphaβ\beta 值必须以某个端节点的静态估值为依据。因此采用 α\alpha-β\beta 过程都要使用某种深度优先的搜索方法

alpha-beta剪枝

剪枝并不影响最后的结果

好的招数序列可以改进剪枝的效率

如果可以找到最好的剪枝,则:

  • 时间复杂度为:O(bm2)O(b^{\frac{m}{2}})
  • 有效的分支系数为 b\sqrt{b}

其中树的深度为 dd,且每个非叶结点的分枝系数为 bb

α\alpha-β\beta 搜索向前看的走步数是极大极小搜索的两倍

α\alpha-β\beta 过程就是把生成后继和倒推值估计结合起来,及时剪掉一些无用分支

# 约束满足搜索

约束满足问题(CSP)就是为一组变量寻找满足约束的赋值。

例如,N - 皇后问题就是一个约束满足问题。这里的问题就是为 N 个变量赋值,每个变量的值表示每行上皇后的问题,值域均为 [1N][1,N],约束就是 N 个皇后谁也 “吃” 不到谁。

类似的,还有地图着色问题

一个约束满足问题表述为一个三元组 (V,D,C)(V,D,C)

  • VV:n 个变量的集合 V=v1,,vnV={v_1,\dots,v_n}
  • DD:变量 vi(i=1,2,,n)v_i (i=1,2,\dots,n) 相应的取值集合
    D={D1,,Dn}D=\{D_1,\dots,D_n\}
  • CC:约束的有限集合,其中每个约束对若干变量同时可取的值做出限制。

如果至少存在一个解答满足某个约束,则称该约束是可满足的。

如果找不到一个变量值的组合,使之满足所有的约束,则可以找到一个满足最大数目约束的解,这种情况称为最大约束满足问题。

# 蒙特卡洛树搜索

分为四个阶段:

  • 选择(Selection)
  • 扩展(Expansion)
  • 模拟(Simulation)
  • 反向传播(Backpropagation)

蒙特卡洛树搜索

推荐观看【双语字幕】MCTS 蒙特卡洛树搜索算法详细步骤解释

# 选择 (Selection)

在选择阶段,需要从根节点,也就是要做决策的局面 R 出发向下选择出一个最急迫需要被拓展的节点 N,局面 R 是是每一次迭代中第一个被检查的节点;对于被检查的局面而言,他可能有三种可能:

  • 该节点所有可行动作都已经被拓展过
    • 使用 UCB 公式 计算该节点所有子节点的 UCB 值,并找到值最大的一个子节点继续检查。反复向下迭代
  • 该节点有可行动作还未被拓展过
    • 认为这个节点就是本次迭代的的目标节点 N,并找出 N 还未被拓展的动作 A。执行模拟
  • 这个节点游戏已经结束了
    • 从该节点直接执行反向传播

# 信任度上限树

N 表示总模拟次数,W 表示胜局次数。每次都选择胜率最大的节点进行模拟。但是这样会导致新节点无法被探索到。为了在最大胜率和新节点探索上保持平衡,UCT(Upper Confidence Bound,上限置信区间算法)被引入:

wini+clntni\frac{w_i}{n_i}+c\sqrt{\frac{\ln{t}}{n_i}} ;在实际中通常可凭经验选择;

  • tt 代表仿真总次数,等于所有 nin_i 的和;
  • 目前蒙特卡洛树搜索的实现大多是基于 UCT 的一些变形

    # 拓展 (Expansion)

    在选择阶段结束时候,我们查找到了一个最迫切被拓展的节点 N,以及他一个尚未拓展的动作 A。在搜索树中创建一个新的节点NnN_n 作为 N 的一个新子节点。NnN_n 的局面就是节点 N 在执行了动作 A 之后的局面

    # 模拟 (Simulation)

    为了让NnN_n 得到一个初始的评分。我们从NnN_n 开始,让游戏随机进行,直到得到一个游戏结局,这个结局将作为NnN_n 的初始评分。一般使用胜利 / 失败来作为评分,只有 1 或者 0。

    # 反向传播 (Back Propagation)

    NnN_n 的模拟结束之后,它的父节点 N 以及从根节点到 N 的路径上的所有节点都会根据本次模拟的结果来添加自己的累计评分


    有些搜索算法在内存中保留一条或多条路径,并且记录哪些是已经探索过的,哪些是还没有探索过的。当找到目标时,到达目标的路径同时也构成了这个问题的一个解。

    但是在许多问题中,问题的与到达目标的路径无关的。例如,在八皇后问题中,重要的是最终皇后的布局,而不是加入皇后的次序

    局部搜索算法从单独的一个当前状态出发,通常只移动到与之相邻的状态。典型情况下,搜索的路径是不保留的

    • 只用很少的内存
    • 通常能在很大状态空间中找到合理的解

    # 爬山法

    贪婪局部搜索

    一直向值增加的方向持续移动,将会在到达一个 “峰顶” 时终止

    爬山法有许多变化的形式:

    • 随机爬山法,它在上山移动中随机地选择下一步;选择的概率随着上山移动的陡峭程度而变化。
    • 首选爬山法,它在实现随机爬山法的基础上,采用的方式是随机地生成后继节点直到生成一个优于当前节点的后继。这个算法在有很多后继节点的情况下有很好的效果。
    • 随机重新开始爬山法,它通过随机生成的初始状态来进行一系列的爬山法搜索,找到目标时停止搜索。这个算法是完备的,概率接近于 1

    # 模拟退火算法

    S.A. (Simulated annealing) 算法,是一种通用概率算法,适用于一定时间内寻找在一个很大搜寻空间中的近似最优解

    P=eδtP=e^{-\frac{\delta}{t}}

    其中 δ\delta 为邻近解与当前解的目标函数之差;tt温度参数

    # 初始化

    由一个产生函数从当前解产生一个位于解空间的新解,并定义一个足够大的数值作为初始温度。

    # 迭代过程

    迭代过程是模拟退火算法的核心步骤,分为新解的产生接受新解两部分:

    1. 由一个产生函数从当前解产生一个位于解空间的新解;为便于后续的计算和接受,减少算法耗时,通常选择由当前新解经过简单地变换即可产生新解的方法,如对构成新解的全部或部分元素进行置换、互换等,注意到产生新解的变换方法决定了当前新解的邻域结构,因而对冷却进度表的选取有一定的影响。
      • 计算与新解所对应的目标函数差。因为目标函数差仅由变换部分产生,所以目标函数差的计算最好按增量计算。事实表明,对大多数应用而言,这是计算目标函数差的最快方法。
    2. 判断新解是否被接受,判断的依据是一个接受准则
      • 最常用的接受准则是 Metropolis 准则:
        • δ<0\delta\lt0 则接受 SS' 作为新的当前解 SS
        • 否则以概率 P=eδtP=e^{-\frac{\delta}{t}} 接受 SS' 作为新的当前解 SS
      • 当新解被确定接受时,用新解代替当前解,这只需将当前解中对应于产生新解时的变换部分予以实现,同时修正目标函数值即可。此时,当前解实现了一次迭代。可在此基础上开始下一轮试验。
      • 而当新解被判定为舍弃时,则在原当前解的基础上继续下一轮试验。

    模拟退火算法与初始值无关,算法求得的解与初始解状态 S(是算法迭代的起点)无关;模拟退火算法具有渐近收敛性,已在理论上被证明是一种以概率 1 收敛于全局最优解的全局优化算法;模拟退火算法具有并行性。

    # 停止准则

    迭代过程的一般停止准则:温度 T 降低至某阈值时,或连续若干次迭代均未接受新解时,停止迭代,接受当前寻找的最优解为最终解。

    # 退火方案

    在某个温度状态 TT 下,当一定数量的迭代操作完成后,降低温度 T,在新的温度状态下执行下一个批次的迭代操作。

    # 伪代码

    伪代码
    // Initialization
    // Randomly generate a solution X0,
    // and calculate its fitness value f(X0).
    Xbest=X0;
    k=0;
    t(k)=T;
    while not stop
      //The search loop under the temperature tk
      for i=1 to L : //The loop times
        // Generate a new solution Xnew based on the current
        // solution Xk, and calculate its fitness value f(Xnew).
        if f(Xnew) < f(X(k)) :
        	Xk = Xnew;
        if f(X(k)) < f(Xbest) Xbest=X(k) :
          continue;
        end if
        Calculate P(tk) = exp[(f(Xnew)-f(X(k)))/t(k)];
        if random(0,1) < P :
          X(k) = Xnew;
        end if
      end for
      //Drop down the temperature
      t(k+1) = drop(t(k));
      k=k+1;
      end while
    print Xbest

    # 遗传算法⭐️

    遗传算法具有并行性 —— 遗传算法按并行方式搜索一个种群数目的点,而不是单点。

    遗传算法不需要求导或其他辅助知识,而只需要影响搜索方向的目标函数和相应的适应度函数。

    遗传算法强调概率转换规则,而不是确定的转换规则。

    由于遗传算法是基于随机搜索的算法,遗传算法对给定问题,可以产生许多的潜在解,最终由使用者决定最终解。只运行一次算法的结果并不能反映算法的性能。为了更好地分析遗传算法的性能,应该以不同的初始随机种子或用不同的参数 (例如种群数量,变异概率等) 多次运行算法

    (在某些特殊不止一个解存在情况下,如多目标优化问题有一组 pareto 最优解。这种遗传算法对于确认可替代解集而言是特别合适的)

    解的染色体表示

    • 随机排列

      对于 n 个城市的 TSP 问题,对 n 个城市从 1 开始编号,解表示为 1 到 n 的排列

      染色体表示1

    • 随机数排序

      生成 n 个 (0,1)(0,1) 范围内的随机数对 n 个城市编号,这 n 个随机数的排序对应旅游城市的顺序

      染色体表示2

    # 选择

    选择是用来确定重组或交叉个体,以及被选个体将产生多少个子代个体。

    首先计算适应度:

    • (1) 按比例的适应度计算;
    • (2) 基于排序的适应 s 计算;

    适应度计算之后是实际的选择,按照适应度进行父代个体的选择。

    选择策略如下:

    # 轮盘赌选择

    (Roulette-wheel selection)

    # 锦标赛选择

    (Tournament selection)

    遗传算法中最流行的选择策略

    • 更小的复杂度
    • 易并行化处理
    • 不易陷入局部最优点
    • 不需要对所有的适应度值进行排序处理

    # 截断选择

    (Truncation selection)

    # 蒙特卡洛选择

    (Monte Carlo selection)

    # 概率选择

    (Probability selection)

    # 线性排序选择

    (Linear-rank selection)

    # 指数排序选择

    (Exponential-rank selection)

    # 玻尔兹曼选择

    (Boltzmann selection)

    # 随机遍历选择

    (Stochastic-universal selection)

    # 精英选择

    (Elite selection)

    # 交叉 (基因重组)

    # 部分映射交叉

    (Partial-Mapped Crossover ,PMX)

    • step 1 : 随机选择下标 s,t
    • step 2 : 交叉子串,2 个 parent 的 s-t 部分子串交换
    • step 3 : 确定映射关系,参考置换群的操作
    • step 4 : 生成后代染色体

    # 其他交叉操作

    order crossover (OX)

    cycle crossover (CX)

    position-based crossover

    order-based crossover

    # 变异

    (mutation)

    交叉之后子代经历的变异,实际上是子代基因按小概率扰动产生的变化。

    # 倒置变异

    (Inversion Mutation)

    • step 1: 随机选择下标 s,t
    • step 2 : 将 parent 的 s-t 中间部分倒置,得到 offspring

    # 流程

    遗传算法流程图

    1. 随机产生初始种群,个体数目一定,每个个体表示为染色体的基因编码
    2. 计算个体的适应度,并判断是否符合优化准则,若符合,输出最佳个体及其代表的最优解,并结束计算;否则转向第 3 步
    3. 依据适应度选择再生个体,适应度高的个体被选中的概率高,适应度低的个体可能被淘汰
    4. 按照一定的交叉概率和交叉方法,生成新的个体
    5. 按照一定的变异概率和变异方法,生成新的个体
    6. 由交叉和变异产生新一代的种群,返回到第 2 步

    # 伪代码

    伪代码
    输入 n 个城市的坐标 (这里单纯以欧式距离作为路程代价)
    (还可以输入距离矩阵,在算法运行前计算距离)
    随机生成10个不同个体的初始种群 P(0)
    for t=0 to 最大选代轮数 :
      初始化后代种群 C(t)
      for i = 1 to 每一代后代数目 :
        依据适应度,从 P(t) 中选取一对父母 p1,p2
        利用交叉操作,从 p1,p2 产生后代 c1,c2
        以一定概率对 c1,c2 进行变异操作
        C(t) <= {c1,c2}
      end for
      依据适应度,从 Union{P(t),C(t)} 选取优异个体作为 P(t+1)
    end for
    依据适应度,从最后一代种群中选择最优个体作问题的解

    # TSP⭐️

    用遗传算法 (GA) 旅行商问题 (TSP) 。(python 中可以直接用 GA.tsp ...)

    可供测试的数据集:National Traveling Salesman Problems (uwaterloo.ca)

    # 自然启发式优化搜索算法

    群体智能(Swarm intelligence,SI)和生物启发(u)

    This part is to be completed

    Edited on Views times
    \ No newline at end of file +M834 80h400000v40h-400000z"/> ;在实际中通常可凭经验选择;
  • tt 代表仿真总次数,等于所有 nin_i 的和;
  • 目前蒙特卡洛树搜索的实现大多是基于 UCT 的一些变形

    # 拓展 (Expansion)

    在选择阶段结束时候,我们查找到了一个最迫切被拓展的节点 N,以及他一个尚未拓展的动作 A。在搜索树中创建一个新的节点NnN_n 作为 N 的一个新子节点。NnN_n 的局面就是节点 N 在执行了动作 A 之后的局面

    # 模拟 (Simulation)

    为了让NnN_n 得到一个初始的评分。我们从NnN_n 开始,让游戏随机进行,直到得到一个游戏结局,这个结局将作为NnN_n 的初始评分。一般使用胜利 / 失败来作为评分,只有 1 或者 0。

    # 反向传播 (Back Propagation)

    NnN_n 的模拟结束之后,它的父节点 N 以及从根节点到 N 的路径上的所有节点都会根据本次模拟的结果来添加自己的累计评分


    有些搜索算法在内存中保留一条或多条路径,并且记录哪些是已经探索过的,哪些是还没有探索过的。当找到目标时,到达目标的路径同时也构成了这个问题的一个解。

    但是在许多问题中,问题的与到达目标的路径无关的。例如,在八皇后问题中,重要的是最终皇后的布局,而不是加入皇后的次序

    局部搜索算法从单独的一个当前状态出发,通常只移动到与之相邻的状态。典型情况下,搜索的路径是不保留的

    • 只用很少的内存
    • 通常能在很大状态空间中找到合理的解

    # 爬山法

    贪婪局部搜索

    一直向值增加的方向持续移动,将会在到达一个 “峰顶” 时终止

    爬山法有许多变化的形式:

    • 随机爬山法,它在上山移动中随机地选择下一步;选择的概率随着上山移动的陡峭程度而变化。
    • 首选爬山法,它在实现随机爬山法的基础上,采用的方式是随机地生成后继节点直到生成一个优于当前节点的后继。这个算法在有很多后继节点的情况下有很好的效果。
    • 随机重新开始爬山法,它通过随机生成的初始状态来进行一系列的爬山法搜索,找到目标时停止搜索。这个算法是完备的,概率接近于 1

    # 模拟退火算法

    S.A. (Simulated annealing) 算法,是一种通用概率算法,适用于一定时间内寻找在一个很大搜寻空间中的近似最优解

    P=eδtP=e^{-\frac{\delta}{t}}

    其中 δ\delta 为邻近解与当前解的目标函数之差;tt温度参数

    # 初始化

    由一个产生函数从当前解产生一个位于解空间的新解,并定义一个足够大的数值作为初始温度。

    # 迭代过程

    迭代过程是模拟退火算法的核心步骤,分为新解的产生接受新解两部分:

    1. 由一个产生函数从当前解产生一个位于解空间的新解;为便于后续的计算和接受,减少算法耗时,通常选择由当前新解经过简单地变换即可产生新解的方法,如对构成新解的全部或部分元素进行置换、互换等,注意到产生新解的变换方法决定了当前新解的邻域结构,因而对冷却进度表的选取有一定的影响。
      • 计算与新解所对应的目标函数差。因为目标函数差仅由变换部分产生,所以目标函数差的计算最好按增量计算。事实表明,对大多数应用而言,这是计算目标函数差的最快方法。
    2. 判断新解是否被接受,判断的依据是一个接受准则
      • 最常用的接受准则是 Metropolis 准则:
        • δ<0\delta\lt0 则接受 SS' 作为新的当前解 SS
        • 否则以概率 P=eδtP=e^{-\frac{\delta}{t}} 接受 SS' 作为新的当前解 SS
      • 当新解被确定接受时,用新解代替当前解,这只需将当前解中对应于产生新解时的变换部分予以实现,同时修正目标函数值即可。此时,当前解实现了一次迭代。可在此基础上开始下一轮试验。
      • 而当新解被判定为舍弃时,则在原当前解的基础上继续下一轮试验。

    模拟退火算法与初始值无关,算法求得的解与初始解状态 S(是算法迭代的起点)无关;模拟退火算法具有渐近收敛性,已在理论上被证明是一种以概率 1 收敛于全局最优解的全局优化算法;模拟退火算法具有并行性。

    # 停止准则

    迭代过程的一般停止准则:温度 T 降低至某阈值时,或连续若干次迭代均未接受新解时,停止迭代,接受当前寻找的最优解为最终解。

    # 退火方案

    在某个温度状态 TT 下,当一定数量的迭代操作完成后,降低温度 T,在新的温度状态下执行下一个批次的迭代操作。

    # 伪代码

    伪代码
    // Initialization
    // Randomly generate a solution X0,
    // and calculate its fitness value f(X0).
    Xbest=X0;
    k=0;
    t(k)=T;
    while not stop
      //The search loop under the temperature tk
      for i=1 to L : //The loop times
        // Generate a new solution Xnew based on the current
        // solution Xk, and calculate its fitness value f(Xnew).
        if f(Xnew) < f(X(k)) :
        	Xk = Xnew;
        if f(X(k)) < f(Xbest) Xbest=X(k) :
          continue;
        end if
        Calculate P(tk) = exp[(f(Xnew)-f(X(k)))/t(k)];
        if random(0,1) < $p$ :
          X(k) = Xnew;
        end if
      end for
      //Drop down the temperature
      t(k+1) = drop(t(k));
      k=k+1;
      end while
    print Xbest

    # 遗传算法⭐️

    遗传算法的特点:

    • 自组织、自适应和自学习性 (智能性)
    • 遗传算法具有并行性 —— 遗传算法按并行方式搜索一个种群数目的点,而不是单点
    • 遗传算法不需要求导或其他辅助知识,而只需要影响搜索方向的目标函数和相应的适应度函数
    • 遗传算法强调概率转换规则,而不是确定的转换规则
    • 遗传算法可以更加直接地被应用
    • 遗传算法对给定问题,可以产生许多潜在解,最终的选择由使用者确定

    由于遗传算法是基于随机搜索的算法,遗传算法对给定问题,可以产生许多的潜在解,最终由使用者决定最终解。只运行一次算法的结果并不能反映算法的性能。为了更好地分析遗传算法的性能,应该以不同的初始随机种子或用不同的参数 (例如种群数量,变异概率等) 多次运行算法

    (在某些特殊不止一个解存在情况下,如多目标优化问题有一组 pareto 最优解。这种遗传算法对于确认可替代解集而言是特别合适的)

    采用多种群 (即有子种群) 的算法往往会获得更好的结果。每个子种群像单种群遗传算法一样独立地演算若干代后,在子种群之间进行个体交换。这种多种群遗传算法更贴近于自然种族的进化,称为并行遗传算法 (ParallelingGeneticAlgorithm,PGA)。

    遗传算法包括 3 个基本操作:

    • 选择
    • 交叉或基因重组
    • 变异

    # 编码

    解的染色体表示

    • 随机排列

      对于 NN 个城市的 TSP 问题,对 NN 个城市从 1 开始编号,解表示为 1 到 NN 的排列

      染色体表示1

    • 随机数排序

      生成 NN(0,1)(0,1) 范围内的随机数对 NN 个城市编号,这 NN 个随机数的排序对应旅游城市的顺序

      染色体表示2

    # 选择

    选择是用来确定重组或交叉个体,以及被选个体将产生多少个子代个体。

    首先计算适应度:

    • (1) 按比例的适应度计算;
    • (2) 基于排序的适应 s 计算;

    适应度计算之后是实际的选择,按照适应度进行父代个体的选择。

    选择策略如下:

    # 轮盘赌选择

    (Roulette-wheel selection)

    # 锦标赛选择

    (Tournament selection)

    遗传算法中最流行的选择策略

    • 更小的复杂度
    • 易并行化处理
    • 不易陷入局部最优点
    • 不需要对所有的适应度值进行排序处理

    # 截断选择

    (Truncation selection)

    # 蒙特卡洛选择

    (Monte Carlo selection)

    # 概率选择

    (Probability selection)

    # 线性排序选择

    (Linear-rank selection)

    # 指数排序选择

    (Exponential-rank selection)

    # 玻尔兹曼选择

    (Boltzmann selection)

    # 随机遍历选择

    (Stochastic-universal selection)

    # 精英选择

    (Elite selection)

    # 交叉 (基因重组)

    # 单点交叉

    随机产生一个交叉点位置,父个体 1 和父个体 2 在交叉点位置之右 (后) 的部分基因码互换,形成子个体 1 和子个体 2。

    # 部分映射交叉

    (Partial-Mapped Crossover ,PMX)

    • step 1 : 随机选择下标 s,t
    • step 2 : 交叉子串,2 个 parent 的 s-t 部分子串交换
    • step 3 : 确定映射关系,参考置换群的操作
    • step 4 : 生成后代染色体

    # 其他交叉操作

    order crossover (OX)

    cycle crossover (CX)

    position-based crossover

    order-based crossover

    # 变异

    (mutation)

    交叉之后子代经历的变异,实际上是子代基因按小概率扰动产生的变化。

    # 倒置变异

    (Inversion Mutation)

    • step 1: 随机选择下标 s,t
    • step 2 : 将 parent 的 s-t 中间部分倒置,得到 offspring

    # 流程

    遗传算法流程图

    1. 随机产生初始种群,个体数目一定,每个个体表示为染色体的基因编码
    2. 计算个体的适应度,并判断是否符合优化准则,若符合,输出最佳个体及其代表的最优解,并结束计算;否则转向第 3 步
    3. 依据适应度选择再生个体,适应度高的个体被选中的概率高,适应度低的个体可能被淘汰
    4. 按照一定的交叉概率和交叉方法,生成新的个体
    5. 按照一定的变异概率和变异方法,生成新的个体
    6. 由交叉和变异产生新一代的种群,返回到第 2 步

    # 伪代码

    伪代码
    输入 $N$ 个城市的坐标 (这里单纯以欧式距离作为路程代价)
    (还可以输入距离矩阵,在算法运行前计算距离)
    随机生成10个不同个体的初始种群 P(0)
    for t=0 to 最大选代轮数 :
      初始化后代种群 C(t)
      for i = 1 to 每一代后代数目 :
        依据适应度,从 P(t) 中选取一对父母 p1,p2
        利用交叉操作,从 p1,p2 产生后代 c1,c2
        以一定概率对 c1,c2 进行变异操作
        C(t) <= {c1,c2}
      end for
      依据适应度,从 Union{P(t),C(t)} 选取优异个体作为 P(t+1)
    end for
    依据适应度,从最后一代种群中选择最优个体作问题的解

    # TSP⭐️

    用遗传算法 (GA) 旅行商问题 (TSP) 。(python 中可以直接用 GA.tsp ...)

    可供测试的数据集:National Traveling Salesman Problems (uwaterloo.ca)

    # 自然启发式优化搜索算法

    群体智能(Swarm intelligence,SI)和生物启发(u)

    This part is to be completed

    Edited on Views times
    \ No newline at end of file diff --git a/cs/ai/uncentain-knowledge-presentation-and-resolution/index.html b/cs/ai/uncentain-knowledge-presentation-and-resolution/index.html index 0f54111a..982d525b 100644 --- a/cs/ai/uncentain-knowledge-presentation-and-resolution/index.html +++ b/cs/ai/uncentain-knowledge-presentation-and-resolution/index.html @@ -1 +1 @@ -不确定性知识表示与推理 - 人工智能 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    按照所用知识的确定性,可以分为确定性和不确定性推理:

    • 确定性推理是建立在经典逻辑基础上的,经典逻辑的基础之一就是集合论。这在很多实际情况中是很难做到的,如高、矮、胖、瘦就很难精确地分开;
    • 不确定性推理就是从不确定性初始证据出发,通过运用不确定性的知识,最终推出具有一定程度的不确定性但却是合理或者近乎合理的结论的思维过程。

    常识(common sense)具有不确定性。一个常识可能有众多的例外,一个常识可能是一种尚无理论依据或者缺乏充分验证的经验。常识往往对环境有极强的依存性。

    不确定知识表示和推理方法:

    • 确定性理论
      • 可信度表示不确定性
    • 证据理论
      • 信任函数度量不确定性
    • 模糊逻辑和模糊推理
      • 模糊集合论应用于近似或模糊推理,形成了可能性理论
      • 模糊逻辑可以看作是多值逻辑的扩展。模糊推理是在一组可能不精确的前提下推出一个可能不精确的结论
    • 概率论处理的是由随机性引起的不确定性;
    • 可能性理论处理的是由模糊性引起的不确定性

    # 概率图模型

    # 基础知识

    这是概率论与图论结合的产物,为统计推理和学习提供了一个统一的灵活框架

    • 节点表示变量节点
    • 之间的边表示局部变量间的概率依赖关系
    • 局部变量分布的连乘积表示系统的联合概率分布

    该表示框架不仅避免了对复杂系统的联合概率分布直接进行建模,而且易于引入先验知识

    概率图模型统一了目前广泛应用的许多统计模型和方法:

    • 马尔科夫随机场(MRF)
    • 条件随机场(CRF)
    • 隐马尔科夫模型(HMM)
    • 多元高斯模型
    • 卡尔曼滤波、粒子滤波、变分推理 \cdots \cdots

    概率论基础【回顾】

    • 频率论学派
      • 事件的概率是当我们无限次重复试验时,事件发生次数的比值
        • 投掷硬币、掷骰子等
    • 贝叶斯学派
      • 将事件的概率视为一种主观置信度
        • 我认为明天下雨的概率是 30%;他认为明天下雨的概率是 80%

    基础:

    • 乘法 / 链式法则 求 联合概率
    • 加法法则
    • 贝叶斯定理:后验概率=先验概率×似然度标准化常量\text{后验概率}=\frac{\text{先验概率}\times\text{似然度}}{\text{标准化常量}}

    三个概念:

    • 联合概率分布
    • 边缘概率分布
    • 最大后验概率状态

    # 分类

    贝叶斯网络以外不作重点,那基本是不考了

    • 有向图模型 Directed graphs
      • ⭐️贝叶斯网络
      • 隐马尔科夫模型
      • 卡尔曼滤波
    • 无向图模型 Undirected graphs
      • 马尔可夫随机场
      • 条件随机场
    • 因子图模型 Factor graphs

    # 贝叶斯网络⭐️

    a,ba,b 相互独立:a\perp\llap{\perp}b;

    a,ba,b 在给定 cc 的条件下相互独立(条件独立):a\perp\llap{\perp}b|c;

    # 定义

    贝叶斯网络(Bayesian Network,简称 BN)是不确定知识表示与推理的一种有效方法,属于有向图模型,它由这些组成:

    • 一个有向无环图(Directed Acyclic Graph, DAG)
    • 一系列条件概率表(衡量了上述关系的强度)

    G 为定义在 {X1,X2,,XN}\{X_1,X_2,\cdots,X_N\} 上的一个贝叶斯网络,其联合概率分布可以表示为各个节点的条件概率分布的乘积

    p(X)=ipi(XiParG(Xi))p(X)=\prod_{i}p_i(X_i|Par_G(X_i))

    联合概率分布结构化分解,原本表示 nn 个二元随机变量的联合分布需要 2n12^n-1 个参数,用贝叶斯网络建模,假设每个节点最多有 kk 个父节点,所需要的参数最多为 n2kn\ 2^k ,所需参数大幅减少


    贝叶斯网络举例

    节点定义:

    • 试题难度(Difficulty)
    • 智力(Intelligence)
    • 考试等级(SAT)
    • 高考成绩(Grade)
    • 是否得到推荐信(Letter)

    贝叶斯网络

    # 盘式记法

    贝叶斯网络的一种简洁的表示方法,其将相互独立的、由相同机制生成的多个变量放在一个方框(盘)内,并在方框中标出类似变量重复出现的个数。方框可以嵌套,且通常用阴影标注出可观察到的变量

    盘式记法

    # 举例

    # 一元语言模型

    假设文本中每个词都和其他词独立,和它的上下文无关。通过一元语言模型,我们可以计算一个序列的概率,从而判断该序列是否符合自然语言的语法和语义规则。

    # 一元混合语言模型

    假设给定文本主题(z)的前提下,该文本中的每个词都和其他词条件独立。其中,z 为隐变量。

    # 概率潜在语义分析

    # 潜在狄利克雷分配

    # 构建步骤

    • 步骤 1:

      在某种变量顺序下,对所有变量的联合概率应用链式法则

      Pr(X1,,Xn)=Pr(XnX1,,Xn1)Pr(Xn1X1,,Xn2)Pr(X1)Pr(X_1,\cdots ,X_n)=Pr(X_n|X_1,\cdots ,X_{n-1})Pr(X_{n-1}|X_1,\cdots ,X_{n-2})\cdots Pr(X_1)

    • 步骤 2:

      对于每个变量 XiX_i ,考虑该变量的条件集合 ${X_1,\cdots,X_{i-1}} $,采用如下方法递归地判断条件集合中的每个变量 XjX_j 是否可以删除:

      如果给定其余变量的集合,XiX_iXjX_j 是条件独立的,则将 XiX_iXjX_j 的条件集合中删除。

      经过这一步骤,可以得到下式:

      Pr(X1,,Xn)=Pr(XnPar(Xn))Pr(Xn1Par(Xn1))Pr(X1)Pr(X_1,\cdots,X_n)=Pr(X_n|Par(X_n))Pr(X_{n-1}|Par(X_{n-1}))\cdots Pr(X_1)

    • 步骤 3:

      基于上述公式,构建一个有向无环图。其中,对于每个用节点表示的变量 XiX_i ,其父节点为 Par(Xi)Par(X_i) 中的变量集合。

    • 步骤 4:

      为每个家庭(即变量及其父节点集合)确定条件概率表的取值

    # D - 分离⭐️

    对于一个有向无环图,D - 分离(D-Separation)是一种用来判断其变量是否条件独立的图形化方法。在基于贝叶斯网络的不确定性知识推理中,采用 D - 分离方法可以简化概率计算,提高运行速度

    D - 分离可用于判断任意两个节点的相关性和独立性。

    • 若存在一条路径将这两个节点(直接)连通,则称这两个节点是有向连接(d-connected)的,即这两个节点是相关的
    • 若不存在这样的路径将这两个节点连通,则这两个节点不是有向连接的,则称这两个节点是有向分离的(d-separated),即这两个节点相互独立

    定义:路径 𝑝 被限定集 𝑍 阻塞(block)当且仅当:

    • 路径 𝑝 含有链结构 A→𝐵→C 或分连结构 A←𝐵→C 且中间节点 𝐵 在 𝑍 中
    • 或者路径 𝑝 含有汇连结构 A→𝐵←C 且汇连节点 𝐵 及其后代都不在 𝑍 中

    定义:若 𝑍 阻塞了节点 X 和节点 𝑌 之间的每一条路径,则称给定 𝑍 时,X 和 𝑌 是 D - 分离,即给定 𝑍 时,X 和 𝑌 条件独立

    引理:父节点已知时,该节点与其所有非后代的节点( non-descendants)满足 D-separated

    定理:父节点已知时,该节点与其所有非后代的节点( non-descendants)条件独立

    # 具体例子

    • 单一:Naive Bayes(朴素贝叶斯)

    • 混合:GMM (混合高斯模型)

    • 加入时间维度

      • Markov Chain
      • Gaussian Process(无限维高斯分布)
    • 连续:Gaussian Bayesian Network

    • 混合模型与时间结合起来:动态模型

      • HMM(离散)
      • Karman Filter(连续 [高斯]、线性)
      • Particle Filter(非高斯、非线性)

    总的来说,用两句话总结这些模型:从单一到混合、从有限到无限

    两个角度:空间(随机变量的取值从离散到连续)、时间(加入时间维度,从某一时刻延长到无限时间)

    # 概率图模型的推理

    (贝叶斯网络为主)

    # 变量消元算法 (VE)

    变量消除(Variable Elimination,VE)基本思想:通过从联合概率分别逐步消除变量,来求边缘概率

    给定一个贝叶斯网络的一系列条件概率表 F\mathbf F,查询变量 QQ,证据变量的取值 E=e\mathbf E=e ,剩余变量 Z\mathbf Z ,需要计算 Pr(QE)Pr(Q|\mathbf E)

    # 因子及相关操作

    f(X,Y)f(\mathbf{X},\mathbf{Y}) 指代涉及变量集合 XY\mathbf{X}\cup\mathbf{Y} 的因子,如条件概率表中的 Pr(CA)Pr(C|A) 可以用因子 f(C,A)f(C,A) 或者 f(A,C)f(A,C) 来表示;

    • 限制:给定 f(X,Y)f(X,\mathbf{Y})X=aX=a ,则 fX=a=h(Y)=f(a,Y)f_{X=a}=h(\mathbf{Y})=f(a,\mathbf{Y})

    • 相乘:给定 f(X,Y)f(\mathbf{X},\mathbf{Y})g(Y,Z)g(\mathbf{Y},\mathbf{Z}) ,则

      h(X,Y,Z)=f(X,Y)×g(Y,Z)h(\mathbf{X},\mathbf{Y},\mathbf{Z})=f(\mathbf{X},\mathbf{Y})\times g(\mathbf{Y},\mathbf{Z})

    • 求和:给定 f(X,Y)f(X,\mathbf{Y}) ,则该因子在变量 XX 上求和将得到

      h(Y)=xDom(X)f(x,Y)h(\mathbf{Y})=\sum_{x\in Dom(X)}f(x,\mathbf{Y})

    # 算法流程

    • 对于任意因子 fFf\in\mathbf F ,如果该因子涉及 EE 中的一个或多个变量,则将其替换为限制后的因子 f_

    • 给定一个消元顺序,对于变量 ZjZZ_j\in\mathbf Z

      • f1,f2,,fkf_1,f_2,\cdots,f_kF\mathbf F 中包含 ZjZ_j 的所有因子;

      • 将这些因子相乘并在 ZjZ_j 上求和后得到新的因子:

        gj=Zjf1×f2××fkg_j=\sum_{Z_j}f_1\times f_2\times\cdots\times f_k

      • F\mathbf F 中删除 f1,f2,,fkf_1,f_2,\cdots,f_k ,并将 gjg_j 加入到 F\mathbf F

    • 剩下的因子将只涉及查询变量 QQ,对这些因子相乘并归一化后得到 Pr(QE)Pr(Q|\mathbf E)

    # 算法理解

    第一种角度是理解为乘法对加法的分配律

    第二种角度是 message 的传播过程,也就是局部的消息传递

    # 信念传播算法 (BP)

    信念传播算法(Belief Propagation, BP),基于局部的消息传递,可以视为 VE 算法 + 缓存 的推广:BP = VE + Caching

    • 在树状图模型上能收敛,且能达到精确推理
    • 但在一般图模型上,BP 为近似推理算法

    BP

    mba(xa)=bφabφbmcb(xb)mdb(xb)m_{b\to a}(x_a)=\sum_{b}\varphi_{ab}\cdot\varphi_{b}\cdot m_{c\to b}(x_b)\cdot m_{d\to b}(x_b)

    φbmcb(xb)mdb(xb)\varphi_{b}\cdot m_{c\to b}(x_b)\cdot m_{d\to b}(x_b)信念 (belief)

    • φb\varphi_{b} 来自 bb 自己;
    • mcb(xb)mdb(xb)m_{c\to b}(x_b)\cdot m_{d\to b}(x_b) 来自 bb 的 children

    一般化形式:

    mji(xi)=xjφijφjkNB(j)imkj(xj)P(xi)=φijNB(i)mji(xi)=φijNB(i)xjφijφjkNB(j)imkj(xj)\begin{aligned} m_{j\to i}(x_i) &=\sum_{x_j}\varphi_{ij}\cdot\varphi_{j}\prod_{k\in\text{NB}(j)-i}m_{k\to j}(x_j)\\ P(x_i) &=\varphi_i\prod_{j\in\text{NB}(i)}m_{j\to i}(x_i)\\ &=\varphi_i\prod_{j\in\text{NB}(i)}\sum_{x_j}\varphi_{ij}\cdot\varphi_{j}\prod_{k\in\text{NB}(j)-i}m_{k\to j}(x_j) \end{aligned}

    # 因子图 BP

    # 概率图模型的学习

    (不考)

    条件概率表的学习方法:

    • 极大似然
    • 期望最大化

    贝叶斯网络结构学习常用方法:

    • 基于约束的结构学习 (constraint-based structure learning):将贝叶斯网络看作是一种独立关系的表示。尝试对数据中的条件依赖和条件独立关系进行检验,找到能够对这些依赖和独立关系给出最好解释的某个网络。
    • 基于得分的结构学习 (score-based structure learning):定义模型对观测数据拟合程度的得分函数,通过最大化得分函数完成结构学习。
      • 结构斯然得分
      • 贝叶斯得分
    \ No newline at end of file +不确定性知识表示与推理 - 人工智能 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    按照所用知识的确定性,可以分为确定性和不确定性推理:

    • 确定性推理是建立在经典逻辑基础上的,经典逻辑的基础之一就是集合论。这在很多实际情况中是很难做到的,如高、矮、胖、瘦就很难精确地分开;
    • 不确定性推理就是从不确定性初始证据出发,通过运用不确定性的知识,最终推出具有一定程度的不确定性但却是合理或者近乎合理的结论的思维过程。

    常识(common sense)具有不确定性。一个常识可能有众多的例外,一个常识可能是一种尚无理论依据或者缺乏充分验证的经验。常识往往对环境有极强的依存性。

    不确定知识表示和推理方法:

    • 确定性理论
      • 可信度表示不确定性
    • 证据理论
      • 信任函数度量不确定性
    • 模糊逻辑和模糊推理
      • 模糊集合论应用于近似或模糊推理,形成了可能性理论
      • 模糊逻辑可以看作是多值逻辑的扩展。模糊推理是在一组可能不精确的前提下推出一个可能不精确的结论
    • 概率论处理的是由随机性引起的不确定性;
    • 可能性理论处理的是由模糊性引起的不确定性

    # 概率图模型

    # 基础知识

    这是概率论与图论结合的产物,为统计推理和学习提供了一个统一的灵活框架

    • 节点表示变量节点
    • 之间的边表示局部变量间的概率依赖关系
    • 局部变量分布的连乘积表示系统的联合概率分布

    该表示框架不仅避免了对复杂系统的联合概率分布直接进行建模,而且易于引入先验知识

    概率图模型统一了目前广泛应用的许多统计模型和方法:

    • 马尔科夫随机场(MRF)
    • 条件随机场(CRF)
    • 隐马尔科夫模型(HMM)
    • 多元高斯模型
    • 卡尔曼滤波、粒子滤波、变分推理 \cdots \cdots

    概率论基础【回顾】

    • 频率论学派
      • 事件的概率是当我们无限次重复试验时,事件发生次数的比值
        • 投掷硬币、掷骰子等
    • 贝叶斯学派
      • 将事件的概率视为一种主观置信度
        • 我认为明天下雨的概率是 30%;他认为明天下雨的概率是 80%

    基础:

    • 乘法 / 链式法则 求 联合概率
    • 加法法则
    • 贝叶斯定理:后验概率=先验概率×似然度标准化常量\text{后验概率}=\frac{\text{先验概率}\times\text{似然度}}{\text{标准化常量}}

    三个概念:

    • 联合概率分布
    • 边缘概率分布
    • 最大后验概率状态

    # 分类

    贝叶斯网络以外不作重点,那基本是不考了

    • 有向图模型 Directed graphs
      • ⭐️贝叶斯网络
      • 隐马尔科夫模型
      • 卡尔曼滤波
    • 无向图模型 Undirected graphs
      • 马尔可夫随机场
      • 条件随机场
    • 因子图模型 Factor graphs

    # 贝叶斯网络⭐️

    a,ba,b 相互独立:a\perp\llap{\perp}b;

    a,ba,b 在给定 cc 的条件下相互独立(条件独立):a\perp\llap{\perp}b|c;

    # 定义

    贝叶斯网络(Bayesian Network,简称 BN)是不确定知识表示与推理的一种有效方法,属于有向图模型,它由这些组成:

    • 一个有向无环图(Directed Acyclic Graph, DAG)
    • 一系列条件概率表(衡量了上述关系的强度)

    G 为定义在 {X1,X2,,XN}\{X_1,X_2,\cdots,X_N\} 上的一个贝叶斯网络,其联合概率分布可以表示为各个节点的条件概率分布的乘积

    p(X)=ipi(XiParG(Xi))p(X)=\prod_{i}p_i(X_i|Par_G(X_i))

    联合概率分布结构化分解,原本表示 nn 个二元随机变量的联合分布需要 2n12^n-1 个参数,用贝叶斯网络建模,假设每个节点最多有 kk 个父节点,所需要的参数最多为 n2kn\ 2^k ,所需参数大幅减少


    贝叶斯网络举例

    节点定义:

    • 试题难度(Difficulty)
    • 智力(Intelligence)
    • 考试等级(SAT)
    • 高考成绩(Grade)
    • 是否得到推荐信(Letter)

    贝叶斯网络

    # 盘式记法

    贝叶斯网络的一种简洁的表示方法,其将相互独立的、由相同机制生成的多个变量放在一个方框(盘)内,并在方框中标出类似变量重复出现的个数。方框可以嵌套,且通常用阴影标注出可观察到的变量

    盘式记法

    # 举例

    # 一元语言模型

    假设文本中每个词都和其他词独立,和它的上下文无关。通过一元语言模型,我们可以计算一个序列的概率,从而判断该序列是否符合自然语言的语法和语义规则。

    # 一元混合语言模型

    假设给定文本主题(z)的前提下,该文本中的每个词都和其他词条件独立。其中,z 为隐变量。

    # 概率潜在语义分析

    # 潜在狄利克雷分配

    # 构建步骤

    • 步骤 1:

      在某种变量顺序下,对所有变量的联合概率应用链式法则

      Pr(X1,,Xn)=Pr(XnX1,,Xn1)Pr(Xn1X1,,Xn2)Pr(X1)Pr(X_1,\cdots ,X_n)=Pr(X_n|X_1,\cdots ,X_{n-1})Pr(X_{n-1}|X_1,\cdots ,X_{n-2})\cdots Pr(X_1)

    • 步骤 2:

      对于每个变量 XiX_i ,考虑该变量的条件集合 ${X_1,\cdots,X_{i-1}} $,采用如下方法递归地判断条件集合中的每个变量 XjX_j 是否可以删除:

      如果给定其余变量的集合,XiX_iXjX_j 是条件独立的,则将 XiX_iXjX_j 的条件集合中删除。

      经过这一步骤,可以得到下式:

      Pr(X1,,Xn)=Pr(XnPar(Xn))Pr(Xn1Par(Xn1))Pr(X1)Pr(X_1,\cdots,X_n)=Pr(X_n|Par(X_n))Pr(X_{n-1}|Par(X_{n-1}))\cdots Pr(X_1)

    • 步骤 3:

      基于上述公式,构建一个有向无环图。其中,对于每个用节点表示的变量 XiX_i ,其父节点为 Par(Xi)Par(X_i) 中的变量集合。

    • 步骤 4:

      为每个家庭(即变量及其父节点集合)确定条件概率表的取值

    # D - 分离⭐️

    对于一个有向无环图,D - 分离(D-Separation)是一种用来判断其变量是否条件独立的图形化方法。在基于贝叶斯网络的不确定性知识推理中,采用 D - 分离方法可以简化概率计算,提高运行速度

    D - 分离可用于判断任意两个节点的相关性和独立性。

    • 若存在一条路径将这两个节点(直接)连通,则称这两个节点是有向连接(d-connected)的,即这两个节点是相关的
    • 若不存在这样的路径将这两个节点连通,则这两个节点不是有向连接的,则称这两个节点是有向分离的(d-separated),即这两个节点相互独立

    定义:路径 𝑝 被限定集 𝑍 阻塞(block)当且仅当:

    • 路径 𝑝 含有链结构 A→𝐵→C 或分连结构 A←𝐵→C 且中间节点 𝐵 在 𝑍 中
    • 或者路径 𝑝 含有汇连结构 A→𝐵←C 且汇连节点 𝐵 及其后代都不在 𝑍 中

    定义:若 𝑍 阻塞了节点 X 和节点 𝑌 之间的每一条路径,则称给定 𝑍 时,X 和 𝑌 是 D - 分离,即给定 𝑍 时,X 和 𝑌 条件独立

    引理:父节点已知时,该节点与其所有非后代的节点( non-descendants)满足 D-separated

    定理:父节点已知时,该节点与其所有非后代的节点( non-descendants)条件独立

    # 具体例子

    • 单一:Naive Bayes(朴素贝叶斯)

    • 混合:GMM (混合高斯模型)

    • 加入时间维度

      • Markov Chain
      • Gaussian Process(无限维高斯分布)
    • 连续:Gaussian Bayesian Network

    • 混合模型与时间结合起来:动态模型

      • HMM(离散)
      • Karman Filter(连续 [高斯]、线性)
      • Particle Filter(非高斯、非线性)

    总的来说,用两句话总结这些模型:从单一到混合、从有限到无限

    两个角度:空间(随机变量的取值从离散到连续)、时间(加入时间维度,从某一时刻延长到无限时间)

    # 概率图模型的推理

    (贝叶斯网络为主)

    # 变量消元算法 (VE)

    变量消除(Variable Elimination,VE)基本思想:通过从联合概率分别逐步消除变量,来求边缘概率

    给定一个贝叶斯网络的一系列条件概率表 F\mathbf F,查询变量 QQ,证据变量的取值 E=e\mathbf E=e ,剩余变量 Z\mathbf Z ,需要计算 Pr(QE)Pr(Q|\mathbf E)

    # 因子及相关操作

    f(X,Y)f(\mathbf{X},\mathbf{Y}) 指代涉及变量集合 XY\mathbf{X}\cup\mathbf{Y} 的因子,如条件概率表中的 Pr(CA)Pr(C|A) 可以用因子 f(C,A)f(C,A) 或者 f(A,C)f(A,C) 来表示;

    • 限制:给定 f(X,Y)f(X,\mathbf{Y})X=aX=a ,则 fX=a=h(Y)=f(a,Y)f_{X=a}=h(\mathbf{Y})=f(a,\mathbf{Y})

    • 相乘:给定 f(X,Y)f(\mathbf{X},\mathbf{Y})g(Y,Z)g(\mathbf{Y},\mathbf{Z}) ,则

      h(X,Y,Z)=f(X,Y)×g(Y,Z)h(\mathbf{X},\mathbf{Y},\mathbf{Z})=f(\mathbf{X},\mathbf{Y})\times g(\mathbf{Y},\mathbf{Z})

    • 求和:给定 f(X,Y)f(X,\mathbf{Y}) ,则该因子在变量 XX 上求和将得到

      h(Y)=xDom(X)f(x,Y)h(\mathbf{Y})=\sum_{x\in Dom(X)}f(x,\mathbf{Y})

    # 算法流程

    • 对于任意因子 fFf\in\mathbf F ,如果该因子涉及 EE 中的一个或多个变量,则将其替换为限制后的因子 f_

    • 给定一个消元顺序,对于变量 ZjZZ_j\in\mathbf Z

      • f1,f2,,fkf_1,f_2,\cdots,f_kF\mathbf F 中包含 ZjZ_j 的所有因子;

      • 将这些因子相乘并在 ZjZ_j 上求和后得到新的因子:

        gj=Zjf1×f2××fkg_j=\sum_{Z_j}f_1\times f_2\times\cdots\times f_k

      • F\mathbf F 中删除 f1,f2,,fkf_1,f_2,\cdots,f_k ,并将 gjg_j 加入到 F\mathbf F

    • 剩下的因子将只涉及查询变量 QQ,对这些因子相乘并归一化后得到 Pr(QE)Pr(Q|\mathbf E)

    # 算法理解

    第一种角度是理解为乘法对加法的分配律

    第二种角度是 message 的传播过程,也就是局部的消息传递

    # 信念传播算法 (BP)

    信念传播算法(Belief Propagation, BP),基于局部的消息传递,可以视为 VE 算法 + 缓存 的推广:BP = VE + Caching

    • 在树状图模型上能收敛,且能达到精确推理
    • 但在一般图模型上,BP 为近似推理算法

    BP

    mba(xa)=bφabφbmcb(xb)mdb(xb)m_{b\to a}(x_a)=\sum_{b}\varphi_{ab}\cdot\varphi_{b}\cdot m_{c\to b}(x_b)\cdot m_{d\to b}(x_b)

    φbmcb(xb)mdb(xb)\varphi_{b}\cdot m_{c\to b}(x_b)\cdot m_{d\to b}(x_b)信念 (belief)

    • φb\varphi_{b} 来自 bb 自己;
    • mcb(xb)mdb(xb)m_{c\to b}(x_b)\cdot m_{d\to b}(x_b) 来自 bb 的 children

    一般化形式:

    mji(xi)=xjφijφjkNB(j)imkj(xj)P(xi)=φijNB(i)mji(xi)=φijNB(i)xjφijφjkNB(j)imkj(xj)\begin{aligned} m_{j\to i}(x_i) &=\sum_{x_j}\varphi_{ij}\cdot\varphi_{j}\prod_{k\in\text{NB}(j)-i}m_{k\to j}(x_j)\\ P(x_i) &=\varphi_i\prod_{j\in\text{NB}(i)}m_{j\to i}(x_i)\\ &=\varphi_i\prod_{j\in\text{NB}(i)}\sum_{x_j}\varphi_{ij}\cdot\varphi_{j}\prod_{k\in\text{NB}(j)-i}m_{k\to j}(x_j) \end{aligned}

    # 因子图 BP

    # 概率图模型的学习

    (不考)

    条件概率表的学习方法:

    • 极大似然
    • 期望最大化

    贝叶斯网络结构学习常用方法:

    • 基于约束的结构学习 (constraint-based structure learning):将贝叶斯网络看作是一种独立关系的表示。尝试对数据中的条件依赖和条件独立关系进行检验,找到能够对这些依赖和独立关系给出最好解释的某个网络。
    • 基于得分的结构学习 (score-based structure learning):定义模型对观测数据拟合程度的得分函数,通过最大化得分函数完成结构学习。
      • 结构斯然得分
      • 贝叶斯得分
    \ No newline at end of file diff --git a/cs/ctf/w4terctf2024/index.html b/cs/ctf/w4terctf2024/index.html index 51ba6b24..fa5264aa 100644 --- a/cs/ctf/w4terctf2024/index.html +++ b/cs/ctf/w4terctf2024/index.html @@ -1,4 +1,4 @@ -W4terCTF2024 WP - CTF - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    年轻人的第一场 CTF ,学到了很多,比较走运拿了第五。最遗憾的是密码学板块了,认真学了不少抽象代数但还是没有做出几题。

    rank

    队友们:(我起了个整活队名,其实是抄袭另一个叫做 “憧憬成为 CTFer” 的 )

    team

    # AI

    # Network Reverse

    网络结构长这样:

    Sequential(
    +W4terCTF2024 WP - CTF - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    年轻人的第一场 CTF ,学到了很多,比较走运拿了第五。最遗憾的是密码学板块了,认真学了不少抽象代数但还是没有做出几题。

    rank

    队友们:(我起了个整活队名,其实是抄袭另一个叫做 “憧憬成为 CTFer” 的 )

    team

    # AI

    # Network Reverse

    网络结构长这样:

    Sequential(
       (0): Conv2d(3, 4, kernel_size=(2, 2), stride=(1, 1))
       (1): GELU(approximate='none')
       (2): Conv2d(4, 8, kernel_size=(5, 5), stride=(1, 1))
    @@ -48,4 +48,4 @@
      [49. 49. 49. 49. 49. 49. 49. 49. 49. 49. 32. 32. 49. 49. 32. 32. 32.]
      [32. 32. 49. 49. 49. 49. 49. 49. 49. 49. 32. 32. 49. 49. 32. 32. 32.]]
     

    同样的方法,可以得到 byte_8020 的后 10 行

    # sub_1483

    代码的逻辑大致是根据输入的 flag 的后 17 位,构造 byte_8020

    刚才已经求出了 byte_8020,可以用下面的脚本求出 flag 后 17 位

    small_alpha_bet = [
    '      11    11    111111111111111111    11    11       ',
    '           111  1119911191119                          ',
    '          11  11111111119999991111111111  11           ',
    '  9   9111        9        111  9   99      9   111    ',
    '            111111111199111111991111111111             ',
    '111  9  9111  1111119111111   111         111  9       ',
    '99111  111      9           111111  111  111         11',
    '   111     9   111      1119            111   111  111 ',
    '   9         111111      9      9      111  99111   111',
    '   111     91111119     111         1119111   1119     ',
    '    111         111   9  111      1119     111        1',
    '        111   11191119    111   991119  111            ',
    '     9111111111              111      9111   111   1119',
    '1119  111111   111        111111111        9111  99111 ',
    '                        1119        111111   111  111  ',
    '999     111111        1119  111  11199   9        91119',
    '111        111      9     9111     1119  99   111111111',
    '        11    11111111111111111111111111    11         ',
    '          11  11                      11  11           ',
    '   99111     111111     111   111  9       111      9  ',
    '        1111119  111111   111              9    111    ',
    '   111      11191111119   111  99        111111   11111',
    '             111   111     11199           9     991111',
    '        9111      111      111111      9   99     111  ',
    '9   9      1111119  1119111      111  11111199       11',
    '        9   111111111     111111111      9111111    111']
    upper_alpha_bet = [
    '                   111111  99      111  111111111111111',
    '      11  11    11                  11    11  11       ',
    '  111     9  111      11199111111              11111111',
    '  11111111  11      1199  11  1199      11  11111111111',
    '1111111111119   999          111      9  9      111    ',
    '111     99   9111                9111111           1119',
    '     11111199111111        111111111      111  9111   1',
    '      11  111111        111111        111111  111      ',
    '        11111111111111111111111111111111111111         ',
    '       1119        111     111   9      111            ',
    '111   111         9      11199111      1119111         ',
    '     111        999   111111111         99     1119111 ',
    '  11          11111111      11111111          11    111',
    '1111111111  11            11            11  11111111111',
    '  1111  11    11        11  11        11    11  1111111',
    '111   9           111111           1119111111   9      ',
    '9111        11111191111111111119   111         111     ',
    '  9111    111111   111   9           111111   111111   ',
    '9111   911199     111111           9     9         9111',
    '       111   111  9  9   99   1111119      111        1',
    '  1111  11  11            11            11  11  1111111',
    '111        111  111111111        111111111           11',
    '    1111    11      9999  11  9999      1111  11111    ',
    '111111     1119111       9      99111   1111119111     ',
    '111     9   9   111        1119  111      1111111119111',
    '  111     111111111111         1119   111111   1111119 ']
    num_bet = [
    '111  9   111     99  111111   111           111     111',
    '   111   111111      111     9111   111   1111119111  1',
    '             9   9   9111111  9      111111111     111 ',
    '111       9111111111   111   111        1111111119  911',
    '99  9111        1119        1119  111        1119     9',
    '   9   111111   9111   9111111111                  9  1',
    '     111   111   111  111111              9     9111111',
    '          1111111111111111111111111111111111           ',
    '       9      111111     9    1111111111111119         ',
    '111   111111999111111  9   9111     111   9111        1']
    matrix_8020 = [
    [32, 32, 49, 49, 32, 32, 32, 32, 32, 32, 32, 32, 32, 32, 49, 49, 49],
    [32, 32, 49, 49, 49, 49, 32, 32, 49, 49, 32, 32, 32, 32, 49, 49, 32],
    [32, 32, 49, 49, 49, 49, 32, 32, 49, 49, 32, 32, 49, 49, 32, 32, 32],
    [49, 49, 49, 49, 49, 49, 49, 49, 49, 49, 32, 32, 49, 49, 32, 32, 32],
    [32, 32, 49, 49, 49, 49, 49, 49, 49, 49, 32, 32, 49, 49, 32, 32, 32],
    [32, 32, 32, 32, 49, 49, 49, 49, 32, 32, 32, 32, 49, 49, 32, 32, 32],
    [32, 32, 32, 32, 32, 32, 49, 49, 32, 32, 49, 49, 49, 49, 49, 49, 32],
    [32, 32, 32, 32, 32, 32, 32, 32, 49, 49, 49, 49, 49, 49, 49, 49, 49],
    [32, 32, 32, 32, 32, 32, 32, 32, 32, 32, 49, 49, 49, 49, 49, 49, 49],
    [32, 32, 32, 32, 32, 32, 32, 32, 32, 32, 32, 32, 49, 49, 49, 49, 49],
    [32, 32, 32, 32, 32, 32, 32, 32, 32, 32, 49, 49, 32, 32, 49, 49, 49],
    [32, 32, 32, 32, 32, 32, 32, 32, 49, 49, 32, 32, 32, 32, 49, 49, 49],
    [32, 32, 32, 32, 32, 32, 49, 49, 32, 32, 32, 32, 49, 49, 32, 32, 32],
    [32, 32, 32, 32, 32, 32, 49, 49, 32, 32, 49, 49, 32, 32, 32, 32, 49],
    [32, 32, 32, 32, 32, 32, 32, 32, 32, 32, 49, 49, 32, 32, 49, 49, 32]]
    str_array_8020 = [ (''.join(chr(x) for x in line )) for line in matrix_8020]
    # print(str_array_8020)
    for line in str_array_8020:
        for i in range(26):
            if small_alpha_bet[i][0:17] == line:
                print(chr(i+ord('a')),end=' ')
        for i in range(26):
            if upper_alpha_bet[i][0:17] == line:
                print(chr(i+ord('A')), end=' ')
        for i in range(10):
            if num_bet[i][0:17] == line:
                print(chr(i+ord('0')), end=' ')

    结果为

    M O U N D W H I 7 e c r a B s

    由此得到 flag

    W4terCTF{1E7_US_drAW_a_plC7ur3_WITH_MA7Rix_4nd_MOUND_WHI7e_craBs}

    # DouDou

    爬了 js 下来,用了 restringer 反混淆,得到关键函数如下:

    function check(q) {
      var V = JSON.parse(JSON.stringify(q)),
        x = []
      for (var P = 0; P < 12; P++) {
        x.push(
          e(q[P])
            .map((g) => g.toString(16).padStart(2, '0'))
            .join('')
        )
      }
      x = x.join('')
      if (x == r + A + s + y + B + F + w + Q + b + p + i + f) {
        // ...

    最后 x == ... 后面的表达式也求出来了,是一个超长 16 进制字符串,太长不贴了。

    分析前面调用 check(STEPS) 的环节可以得到, q 会是一个 [[int * 16] * 12] ,而函数 e ,结合 hint 最终判断为 AES 加密,key 是 W4terDr0pCTF2024 ,但是还没完,加密之前每个字节要先异或一个 153

    最后生成的 x 要等于那个超长字符串,才会进入 if 内生成 flag 的过程,得到 flag:

    flag = ''
        res = ''
        for (var P = 0; P < 12; P++) {
          res += V[P].map((g) => String.fromCharCode(g))
        }
        res = res.replaceAll(',', '')
        for (var P = 0; P < res.length; P += 4) {
          flag += String.fromCharCode(parseInt(res.slice(P, P + 4), 4))
        }

    大概就是 res 四个一组,parse 成一个四位四进制整数,再转 ascii 即可。这也就意味着 res 必须是 ^[0123]*$ ,这也可以检验得到 res 的过程是否正确,反正我们一次就出来了:

    final_x = "91d48f9e77505fe36b1537597d68a8be6ef1ced09bba7c9d4ea6a123042d47bbd5776d7b85eebbb7738de44b9af45f9b706640c8dac77d409ceb067e677e7290401e4852527e119b8095a636d0b628e1161142b7dfcbcf2ef890698ba5d279c78df2348e45d1aef635ebfb841284f6d67d1e9d0f0489bcdc5ba303bc791d0e18e46889fa856a463907d82a8137b7b2a863d1896d4dfcd6b84a2f55b50567f705ea7177187609da2e506182007b7e8244acf62c69de4f511760159ffc6dd8e9f5"
    aes = AES.new(b"W4terDr0pCTF2024", AES.MODE_ECB)
    l = []
    while final_x:
        part_s = final_x[:32]
        final_x = final_x[32:]
        l2 = []
        while part_s:
            x = int(part_s[:2], 16)
            l2.append(x)
            part_s = part_s[2:]
        l.append(l2)
    l = [[x ^ 153 for x in subl] for subl in l]
    l = [bytes(subl) for subl in l]
    l = [aes.decrypt(b) for b in l]
    ans = ""
    for s in l:
        while s:
            b = int(s[:4], 4)
            s = s[4:]
            ans += chr(b)
    ans
    'jS_ls_In7erEStlng_ANd_QuaTeRN4rY_IS_al50_1Unny!!'
    -

    # 古老的语言

    用 VB 反编译器逆向,关键函数 Fxxxtel 如下:

    Public Function Fxxxtel(raw) '40F35C
      'Data Table: 40E3A8
      Dim var_B0 As Long
      Dim var_B4 As Long
      Dim var_B8 As Long
      Dim var_BC As Long
      Dim var_86 As Integer
      loc_40F031: For var_AC = 0 To 9 Step 3: var_A6 = var_AC 'Integer
      loc_40F047:   var_B4 = raw(CLng(var_A6))
      loc_40F055:   var_B8 = raw(CLng((var_A6 + 1)))
      loc_40F063:   var_BC = raw(CLng((var_A6 + 2)))
      loc_40F06D:   For var_C0 = 1 To &H20: var_A8 = var_C0 'Integer
      loc_40F089:     var_B0 = AddLong(0, -1640531527)
      loc_40F0FB:     var_B4 = var_B4 Xor AddLong(AddLong(LeftRotateLong(var_B8, 4) Xor -559038737, var_B8 Xor var_B0), RightRotateLong(var_B8, 5) Xor -1161901314)
      loc_40F16D:     var_B8 = var_B8 Xor AddLong(AddLong(LeftRotateLong(var_BC, 4) Xor -559038737, var_BC Xor var_B0), RightRotateLong(var_BC, 5) Xor -1161901314)
      loc_40F1DF:     var_BC = var_BC Xor AddLong(AddLong(LeftRotateLong(var_B4, 4) Xor -559038737, var_B4 Xor var_B0), RightRotateLong(var_B4, 5) Xor -1161901314)
      loc_40F1E5:   Next var_C0 'Integer
      loc_40F1F4:   var_A0(CLng(var_A6)) = var_B4
      loc_40F202:   var_A0(CLng((var_A6 + 1))) = var_B8
      loc_40F210:   var_A0(CLng((var_A6 + 2))) = var_BC
      loc_40F214: Next var_AC 'Integer
      loc_40F21B: var_86 = &HFF

    用 C 整理出的等价版本:

    void encrypt(uint32_t *v) {
      uint32_t a = v[0], b = v[1], c = v[2];
      for (int i = 0; i < 32; i++) {
        a = a ^ round_fn(b);
        b = b ^ round_fn(c);
        c = c ^ round_fn(a);
      }
      v[0] = a;
      v[1] = b;
      v[2] = c;
    }
    uint32_t round_fn(uint32_t b) {
      const uint32_t delta = 0x9e3779b9;
      return ((left_rotate(b, 4) ^ (0xdeadbeef)) + (b ^ delta) +
              (right_rotate(b, 5) ^ (0xbabecafe)));

    在这之后是一长串的 IF,判断 var_A0 的各个位是否等于一些硬编码的 32 位常数,将这些常数解密即可。

    可以看出这和 TEA 加密非常像,但是是三个一组的循环混淆,而非经典 TEA 的两个一组,而且把 TEA 轮函数加法和异或交换了,以及每次异或的是固定的 delta 而不是 delta 的累加。每次取三个 u32 一组,像这样交叉异或 32 轮。

    三个一组问题不大,把加密的顺序反过来异或回去可以了,原理是一样的:

    void decrypt(uint32_t *v) {
      uint32_t a = v[0], b = v[1], c = v[2];
      for (int i = 0; i < 32; i++) {
        c = c ^ round_fn(a);
        b = b ^ round_fn(c);
        a = a ^ round_fn(b);
      }
      v[0] = a;
      v[1] = b;
      v[2] = c;
    }

    然而这样不对。hint 说要关掉的那个选项我看到了,关了之后重新复现了一遍,发现跟之前一点区别没有,心态炸了。

    最后一天晚九点,突发奇想把轮函数改了一下,把 sum 做异或,相当于只把轮函数的加法和异或互换的 TEA 算法,保留 sum 的部分不改,然后居然过了,有瞎蒙的成分在。

    事后注意到这一行:

    loc_40F089:     var_B0 = AddLong(0, -1640531527)

    那就意味着这一行的逆向是错的,应该是 var_B0 = AddLong(var_B0, -1640531527) ,不过没发现为什么。

    Edited on Views times
    \ No newline at end of file +

    # 古老的语言

    用 VB 反编译器逆向,关键函数 Fxxxtel 如下:

    Public Function Fxxxtel(raw) '40F35C
      'Data Table: 40E3A8
      Dim var_B0 As Long
      Dim var_B4 As Long
      Dim var_B8 As Long
      Dim var_BC As Long
      Dim var_86 As Integer
      loc_40F031: For var_AC = 0 To 9 Step 3: var_A6 = var_AC 'Integer
      loc_40F047:   var_B4 = raw(CLng(var_A6))
      loc_40F055:   var_B8 = raw(CLng((var_A6 + 1)))
      loc_40F063:   var_BC = raw(CLng((var_A6 + 2)))
      loc_40F06D:   For var_C0 = 1 To &H20: var_A8 = var_C0 'Integer
      loc_40F089:     var_B0 = AddLong(0, -1640531527)
      loc_40F0FB:     var_B4 = var_B4 Xor AddLong(AddLong(LeftRotateLong(var_B8, 4) Xor -559038737, var_B8 Xor var_B0), RightRotateLong(var_B8, 5) Xor -1161901314)
      loc_40F16D:     var_B8 = var_B8 Xor AddLong(AddLong(LeftRotateLong(var_BC, 4) Xor -559038737, var_BC Xor var_B0), RightRotateLong(var_BC, 5) Xor -1161901314)
      loc_40F1DF:     var_BC = var_BC Xor AddLong(AddLong(LeftRotateLong(var_B4, 4) Xor -559038737, var_B4 Xor var_B0), RightRotateLong(var_B4, 5) Xor -1161901314)
      loc_40F1E5:   Next var_C0 'Integer
      loc_40F1F4:   var_A0(CLng(var_A6)) = var_B4
      loc_40F202:   var_A0(CLng((var_A6 + 1))) = var_B8
      loc_40F210:   var_A0(CLng((var_A6 + 2))) = var_BC
      loc_40F214: Next var_AC 'Integer
      loc_40F21B: var_86 = &HFF

    用 C 整理出的等价版本:

    void encrypt(uint32_t *v) {
      uint32_t a = v[0], b = v[1], c = v[2];
      for (int i = 0; i < 32; i++) {
        a = a ^ round_fn(b);
        b = b ^ round_fn(c);
        c = c ^ round_fn(a);
      }
      v[0] = a;
      v[1] = b;
      v[2] = c;
    }
    uint32_t round_fn(uint32_t b) {
      const uint32_t delta = 0x9e3779b9;
      return ((left_rotate(b, 4) ^ (0xdeadbeef)) + (b ^ delta) +
              (right_rotate(b, 5) ^ (0xbabecafe)));

    在这之后是一长串的 IF,判断 var_A0 的各个位是否等于一些硬编码的 32 位常数,将这些常数解密即可。

    可以看出这和 TEA 加密非常像,但是是三个一组的循环混淆,而非经典 TEA 的两个一组,而且把 TEA 轮函数加法和异或交换了,以及每次异或的是固定的 delta 而不是 delta 的累加。每次取三个 u32 一组,像这样交叉异或 32 轮。

    三个一组问题不大,把加密的顺序反过来异或回去可以了,原理是一样的:

    void decrypt(uint32_t *v) {
      uint32_t a = v[0], b = v[1], c = v[2];
      for (int i = 0; i < 32; i++) {
        c = c ^ round_fn(a);
        b = b ^ round_fn(c);
        a = a ^ round_fn(b);
      }
      v[0] = a;
      v[1] = b;
      v[2] = c;
    }

    然而这样不对。hint 说要关掉的那个选项我看到了,关了之后重新复现了一遍,发现跟之前一点区别没有,心态炸了。

    最后一天晚九点,突发奇想把轮函数改了一下,把 sum 做异或,相当于只把轮函数的加法和异或互换的 TEA 算法,保留 sum 的部分不改,然后居然过了,有瞎蒙的成分在。

    事后注意到这一行:

    loc_40F089:     var_B0 = AddLong(0, -1640531527)

    那就意味着这一行的逆向是错的,应该是 var_B0 = AddLong(var_B0, -1640531527) ,不过没发现为什么。

    Edited on Views times
    \ No newline at end of file diff --git a/cs/data-structure-and-algorithm/b-tree/index.html b/cs/data-structure-and-algorithm/b-tree/index.html index c07d3df3..efd7d64e 100644 --- a/cs/data-structure-and-algorithm/b-tree/index.html +++ b/cs/data-structure-and-algorithm/b-tree/index.html @@ -1 +1 @@ -B - 树 - 数据结构与算法 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # Multiway Tree

    多叉树任意节点的所有子树也都是有顺序的

    多叉树可以按固定的方法向二叉树转化:

    1. 对所有节点只保留与最左侧子节点的连接,删除其他连接
    2. 从左到右依次连接同一层的兄弟节点 (siblings)

    # B-Tree

    Banlanced Multiway Tree, 顾名思义,B - 树是一种平衡的多路查找树,是平衡二叉树的外延概念。

    B-Tree 是为外部查找(如磁盘)设计的一种平衡查找树。

    系统从磁盘读取数据到内存时是以存储块 (block) 为基本单位的,位于同一个磁盘块中的数据会被一次性读取出来,而不是需要什么取什么。B-Tree 结构的数据可以让系统高效的找到数据所在的磁盘块。

    为了描述 B-Tree,首先定义一条记录为一个二元组[key,data][key,data],key 为记录的健值,对应表中的主键值,data 为一行记录中除主键外的数据。对于不同的记录,key 值互不相同。

    # 定义

    一颗 m 阶 B - 树 (m 路 B - 树) 满足:

    1. 所有叶子节点在同一层 (level),所有叶节点实际上不存在,都是空节点,平衡因子 = 0
      • 根节点若非空 (不是叶节点 / 空节点),则有至少 2 颗子树
      • 除 根节点 以外所有 内部节点 的 子节点 / 子树 数量XX 满足: m2Xm\left\lceil\frac{m}{2}\right\rceil \leq X \leq m
      • 根节点至少有 11 个关键字,至多有 m1m-1 个关键字
      • 非根节点至少有 m21\left\lceil\frac{m}{2}\right\rceil-1 个关键字,至多有 m1m-1 个关键字
    2. 每个节点中的关键字都按照从小到大 (升序) 排列,每个关键字的左子树中的所有关键字都小于它,右子树中的所有关键字都大于它 (也可以反过来,只要满足查找树的定义)

    # 推论

    高度为hhmm 阶 B - 树关键字NN 取值范围为:
    2m2h11Nmh12\left\lceil\frac{m}{2}\right\rceil^{h-1}-1\leq N\leq m^h-1

    NN 个关键字的mm 阶 B - 树的高度hh 的取值范围为:
    logm(N+1)hlogm2N+12+1log_m(N+1)\leq h\leq log_{\left\lceil\frac{m}{2}\right\rceil}\frac{N+1}{2}+1

    \ No newline at end of file +B - 树 - 数据结构与算法 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # Multiway Tree

    多叉树任意节点的所有子树也都是有顺序的

    多叉树可以按固定的方法向二叉树转化:

    1. 对所有节点只保留与最左侧子节点的连接,删除其他连接
    2. 从左到右依次连接同一层的兄弟节点 (siblings)

    # B-Tree

    Banlanced Multiway Tree, 顾名思义,B - 树是一种平衡的多路查找树,是平衡二叉树的外延概念。

    B-Tree 是为外部查找(如磁盘)设计的一种平衡查找树。

    系统从磁盘读取数据到内存时是以存储块 (block) 为基本单位的,位于同一个磁盘块中的数据会被一次性读取出来,而不是需要什么取什么。B-Tree 结构的数据可以让系统高效的找到数据所在的磁盘块。

    为了描述 B-Tree,首先定义一条记录为一个二元组[key,data][key,data],key 为记录的健值,对应表中的主键值,data 为一行记录中除主键外的数据。对于不同的记录,key 值互不相同。

    # 定义

    一颗 m 阶 B - 树 (m 路 B - 树) 满足:

    1. 所有叶子节点在同一层 (level),所有叶节点实际上不存在,都是空节点,平衡因子 = 0
      • 根节点若非空 (不是叶节点 / 空节点),则有至少 2 颗子树
      • 除 根节点 以外所有 内部节点 的 子节点 / 子树 数量XX 满足: m2Xm\left\lceil\frac{m}{2}\right\rceil \leq X \leq m
      • 根节点至少有 11 个关键字,至多有 m1m-1 个关键字
      • 非根节点至少有 m21\left\lceil\frac{m}{2}\right\rceil-1 个关键字,至多有 m1m-1 个关键字
    2. 每个节点中的关键字都按照从小到大 (升序) 排列,每个关键字的左子树中的所有关键字都小于它,右子树中的所有关键字都大于它 (也可以反过来,只要满足查找树的定义)

    # 推论

    高度为hhmm 阶 B - 树关键字NN 取值范围为:
    2m2h11Nmh12\left\lceil\frac{m}{2}\right\rceil^{h-1}-1\leq N\leq m^h-1

    NN 个关键字的mm 阶 B - 树的高度hh 的取值范围为:
    logm(N+1)hlogm2N+12+1log_m(N+1)\leq h\leq log_{\left\lceil\frac{m}{2}\right\rceil}\frac{N+1}{2}+1

    \ No newline at end of file diff --git a/cs/data-structure-and-algorithm/binary-tree/index.html b/cs/data-structure-and-algorithm/binary-tree/index.html index ff5a2480..ca066e81 100644 --- a/cs/data-structure-and-algorithm/binary-tree/index.html +++ b/cs/data-structure-and-algorithm/binary-tree/index.html @@ -1 +1 @@ -二叉树 - 数据结构与算法 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 二叉树的一些规范 (考研)

    节点的度数:节点出发指向下一层节点的边的数量 (出度)

    N0,N1,N2N_0,N_1,N_2:分别表示度数为 1,2,3 的顶点数量

    # 结论

    任意二叉树满足:

    • n=N0+N1+N2n=N_0+N_1+N_2

    • n1=0×N0+1×N1+2×N2=N1+2N2n-1=0\times N_0+1\times N_1+2\times N_2=N_1+2N_2

    • n=1+N1+2N2n=1+N_1+2N_2

    • N2=N01N_2=N_0-1

    正则二叉树还满足:

    • N1=0N_1=0

    • n+1=2N0n+1=2N_0

    • n1=2N2n-1=2N_2

    # 树的遍历

    • 先序:根、左、右

    • 中序:左、根、右

    • 后序:左、右、根

    # 递归做法

    三行代码:遍历左子树、节点、右子树,变换顺序即可得到三种遍历方式

    # 非递归做法

    # 已知二叉树的两种遍历结果确定 (还原) 二叉树

    • 即还原这颗二叉树,当且仅当这两种遍历中有一个是中序遍历!

    • 如果遍历结果是打印出来的元素结果,还需要每个元素不一样?!结点序列的话由于是指针则不需要?

    • 基本思路:已知先序和后序遍历都能容易确定谁是树根,再由中序遍历才确定左右子树

    # 线索二叉树 Threaded binary trees

    一个具有 n 个结点的二叉树若采用二叉链表存储结构, 在 2n 个指针域中只有 n-1 个指针域是用来存储结点孩 子的地址,而另外 n+1 个指针域存放的都是 NULL。

    因此,可以利用某结点空的左指针域 (lchild) 指出该 结点在某种遍历序列中的直接前驱结点的存储地址, 利用结点空的右指针域 (rchild) 指出该结点在某种遍历序列中的直接后继结点的存储地址;对于那些非空 的指针域,则仍然存放指向该结点左、右孩子的指针。 这样,就得到了一棵线索二叉树。

    由于序列可由不同的遍历方法得到,因此,线索树有 先序线索二叉树、中序线索二叉树、后序线索二叉树 三种。把二叉树改造成线索二叉树的过程称为线索化。

    # 哈夫曼树

    哈夫曼树 (Huffman Tree) 是指对于一组带有确定权值的叶结点,构造的具有最小带权路径长度的二叉树,也称最优二叉树 (Optimal Binary Tree)。

    设二叉树具有 n 个带权值的叶结点,那么从根结点到各个叶结点的路径长度与 相应结点权值的乘积之和,称为二叉树的带权路径长度,记为:

    WPL=k=1nWPL=\sum_{k=1}^{n}

    其中 WkW_k 为第 k 个叶结点的权值,LkL_k 为第 k 个叶结点的路径长度。

    在给定一组具有确定权值的叶结点,可以构造出不同的带权二叉树。这些形状不同的二叉树的带权路径长度 WPL 将各不相同。

    哈夫曼树是正则二叉树,即没有度数为 1 的顶点。

    # 构造

    由相同权值的一组叶结点所构成的二叉树有不同的形态和不同的带权路径长度,那么如何找到带权路径长度最小的二叉树呢?

    根据哈夫曼树的定义,一棵二叉树要使其 WPL 值最小,必须使权值越大的叶结点越靠近根结点,而权值越小的叶结点越远离根结点。

    哈夫曼 (Huffman) 依据这一特点提出了一种方法。

    1. 由给定的 n 个权值 {W1W_1,W2W_2,...,WnW_n} 构造 n 棵只有一个叶结点的二叉树,从而得到 一个二叉树的集合 F={T1T_1,T2T_2,...,TnT_n};

    2. 在 F 中选取根结点的权值最小和次小的两棵二叉树作为左、右子树构造一棵新的二叉树,这棵新的二叉树根结点的权值为其左、右子树根结点权值之和;

    3. 在集合 F 中删除作为左、右子树的两棵二叉树,并将新建立的二叉树加入到集合 F 中;

    4. 重复 (2)、(3) 两步,当 F 中只剩下一棵二叉树时,这棵二叉树便是所要建立的哈夫曼树。

    # 哈夫曼编码

    哈夫曼编码 (Huffman Coding) 是一种编码方式,是一种用于无损数据压 缩的熵编码 (权编码) 算法,是 David A. Huffman 发明的。

    # 哈夫曼编码代码

    c++
    typedef struct  // 哈夫曼编码的存储结构
    { char bits[N];  // 保存编码的数组 
        int start;   // 编码有效起始位置,该位置之后的 01 串为字符的编码 
        char ch;  // 字符
    } CodeType ;
    CodeType code[N+1]; // 字符编码数组,下标为 0 的单元空出
    c++
    huffmanCode(HufmTree tree[],CodeType code[]) /* 利用哈夫曼树求字符的哈夫曼编码 */
    {
        int i,c,p;
        for(i = 1; i<=N; i++)  // N 次循环,分别得到 N 个字符的编码
        {
            code[i].start = N; c = i;
            p = tree[i].parent; // 获取字符的双亲
            while (p != 0) // 一直往上层查找,直到根结束
            {
                code[i].start--;
                if ( tree[p].lchild ==c ) code[i].bits[code[i].start] = '0';
                else code[i].bits[code[i].start] = '1';
                c = p;
                p = tree[p].parent;
            }
        }
    }

    # 二叉查找树

    BST (Binary Search Tree) 递归定义:

    1. 根节点:左子树所有节点的值 < 根节点的值 < 右子树所有节点的值

    2. 任何节点的左子树右子树也都是 BST

    # 二叉树的查找 (二叉查找树实现二分查找函数)

    递归实现:

    c++
    //Recursive auxiliary function
    template <class Record>
    Binary_node<Record>*
    Search_tree<Record> :: search_for_node
    (Binary_node<Record>*root1, const Record &target)const
    {
        if (root1 == NULL || root1->data == target) return root;
        if (root1->data < target)return search_for_node(root1->right, target);
        return search_for_node(root1->left, target);
    }
    c++
    //Nonrecursive version
    template <elass Record>
    Binary_node<Record>*
    Search_tree<Record> :: search_for_node
    (Binary_node<Record>*root1, const Record &target) const
    {
        while (root1 != NULL && root1->data != target){
            if (root1->data < target) root1 = root1->right;
            else root1 = root1->left;
        }
        return root1:
    }
    c++
    //Public method for tree search
    template <class Record>
    Error_code
    Search_tree<Record> :: tree_search(Record &target) const
    {
        Error_code_result = success;
        Binary_node<Record> *found = search_for_node(root, target);
        if (found == NULL) result = not_present;
        else target = found->data;
        return result;
    }

    # BST 的插入

    c++
    template <class Record>
    Error_code
    Search_tree<Record> :: insert(const Record & new_data)
    {
        return search_and_insert(root, new_data);
    }
    c++
    template <class Record>
    Error_code
    Search_tree<Record> :: search_and_insert
    (Binary_node<Record> *&root1, const Record & new_data)
    {
        if (root1 == NULL)
        {
            root1 = new Binary_node<Record>(new_data);
            return success;
        }
        if (root1->data == new_data) return duplicate_error;
        if (root1->data > new_data) return search_and_insert(root1->left, new_data);
        return search_and_insert root1->right, new_data);
    }

    # BST 的遍历

    BST 中序遍历的结果就是所有数据升序排序结果

    # BST 的排序

    实际上与快排等价

    平均情况下 BST 排序 (Treesort) 的时间复杂度为:

    2nln(n)+O(n)1.39nlgn+O(n)2nln(n)+O(n)\approx1.39nlgn+O(n)

    # BST 的删除

    1. 要删除的节点没有子节点
      直接删除即可

    2. 要删除的节点只有一个子节点(子树)
      以子树根节点替换该节点再删除该节点即可

    3. 要删除的节点有两个子节点(子树)
      要删除节点 x
      有两种互相对称的做法:

      沿着左子树的右链走到尽头的节点 w 的元素是 x 左子树里最大的元素
      只需将 w 元素赋值给 x,再删除 w 节点即可

      沿着右子树的左链走到尽头的节点 w 的元素是 x 右子树里最小的元素
      只需将 w 元素赋值给 x,再删除 w 节点即可

    c++
    template <class Record>
    Error_code
    Search_tree<Record> :: search_and_destroy
    (Binary_node<Record>*&root1, const Record &target)
    {
        if (root1 == NULL || root1->data == target)return remove_root(root1);
        if (target < root1->data)return search_and_destroy(root1->left, target);
        return search_and_destroy(root1->right, target);
    }
    c++
    template <class Record>
    Error_code
    Search_tree<Record> :: remove_root
    (Binary_node<Record>*&root1)
    {
        if (root1 == NULL) return not_present;
        Binary_node<Record> *to_delete = root1;
        if (root1->right == NULL) root1 = root1->left;
        else if (root1->left == NULL) root1 = root1->right;
        else
        {
            to_delete = root1->left;
            Binary_node<Record> *parent= root1;
            while (to_delete->right != NULL)
            {
                parent = to_delete;
                to_delete = to_delete->right;
            }
            root1->data = to_delete->data;
            if (parent = root1) root1->left = to_delete->left; // 重点
            else parent->right = to_delete->left; // 重点
        }
        delete to_delete; return success;
    }

    # 平衡二叉树

    • 二叉树结构会影响查找效率

      • 二叉树结构越趋向于完全二叉树则时间复杂度越接近O(log(n))O(log(n)),效率越好

      • 二叉树结构越趋向于线性链表则时间复杂度越接近于O(n)O(n),效率越差

        • 最坏的情况会退化为O(n)O(n) 的线性查找复杂度

    平均访问节点次数:2ln(n)1.39log(n)2ln(n)\approx1.39log(n)

    平均比较节点次数:4ln(n)2.77log(n)4ln(n)\approx2.77log(n)

    也就是为什么我们需要自平衡二叉查找树

    # AVL Trees

    AVL 是一种平衡二叉树 (自平衡二叉查找树),过于复杂,没有太大优势,实际更常用的是红黑树来实现

    # 平衡因子 BF (Balance Factor)

    BF=左子树深度右子树深度BF=左子树深度-右子树深度

    # AVL 定义

    任意节点的 BF 只能为 - 1、0 或 1

    # 插入元素

    插入元素会改变 BF,需要调整来维持二叉树的平衡

    AVL 树采用四种 “旋转” 调整操作来实现

    # 代码实现:

    https://blog.csdn.net/weixin_57761086/article/details/127152576

    此处定义:BF=右子树深度左子树深度BF=右子树深度-左子树深度

    # 数据结构定义
    c++
    // 结点类
    template <typename T>
    struct AVLTreeNode {
    	AVLTreeNode<T>* left;// 左孩子指针
    	AVLTreeNode<T>* right;// 右孩子指针
    	AVLTreeNode<T>* parent;// 双亲指针
    	T value;// 存储元素的变量
    	int bf;// 平衡因子
     
    	// 用传入的值构造结点
    	AVLTreeNode(const T& _value = T())
    		:left(nullptr)
    		,right(nullptr)
    		,parent(nullptr)
    		,value(_value)
    		,bf(0)
    	{}
    };
    c++
    //AVL 树类
    template<typename T>
    class AVLTree {
    	// 给结点类起别名,方便使用
    	typedef AVLTreeNode<T> Node;
    private:
    	// 指向根结点的结点指针
    	Node* root;
    };
    c++
    // 无参构造
    // 让根结点指针指向空即可
    AVLTree():root(nullptr){}
    # 左旋转(RR)

    当新节点插入到较高右子树的右侧时 (右右 / RR),需要对最先失去平衡的节点进行一次左旋,使其右子树成为新的根节点,原根节点成为其左子树。

    c++
    // 左单旋
    void RotateLeft(Node* parent) {
    	// 这里的指针设计在讲解旋转的时候已经解释过
    	Node* subR = parent->right;
    	Node* subRL = subR->left;
     
    	// 用 pparent (祖父结点) 来保存 parent 结点的双亲
    	// 因为 parent 并不一定是整棵树的根,还有可能只是一棵子树的根
    	Node* pparent = parent->parent;
     
    	// 第一步:首先让双亲的右孩子指针指向 subRL
    	parent->right = subRL;
     
    	// 之所以要判断是因为 subRL 结点可能不存在
    	if (subRL)
    		// 更新双亲
    		subRL->parent = parent;
     
    	// 第二步:让 subR 的左孩子指针指向 parent
    	subR->left = parent;
     
    	// 不要忘记给这些结点更新它们的双亲
    	parent->parent = subR;
    	subR->parent = pparent;
     
    	// 祖父结点是空,说明 parent 结点是整棵树的根结点
    	if (pparent == nullptr) {
    		root = subR;
    	}
    	// 说明 parent 是子树的根结点
    	else {
    		// 原本祖父结点的孩子是 parent
    		// 经过旋转他的孩子变成了 subR
    		// 需要判断 parent 是祖父结点的左孩子还是右孩子,然后让对应指针指向 subR 结点
    		if (pparent->left == parent) {
    			pparent->left = subR;
    		}
    		else {
    			pparent->right = subR;
    		}
    	}
    	// 更新平衡因子
    	parent->bf = subR->bf = 0;
    }
    # 右旋转(LL)

    当新节点插入到较高左子树的左侧时 (左左 / LL),需要对最先失去平衡的节点进行一次右旋,使其左子树成为新的根节点,原根节点成为其右子树。

    c++
    // 右单旋
    void RotateRight(Node* parent) {
    	Node* subL = parent->left;
    	Node* subLR = subL->right;
     
    	// 祖父结点
    	Node* pparent = parent->parent;
     
    	// 第一步:让 parent 的左孩子指针指向 subLR
    	parent->left = subLR;
    	// 之所以要判断是因为 subLR 结点可能不存在
    	if (subLR)
    		subLR->parent = parent;
    	// 第二步:让 subL 的右孩子指针指向 parent
    	subL->right = parent;
     
    	// 更新双亲结点
    	parent->parent = subL;
    	subL->parent = pparent;
     
    	// 祖父结点是空,说明 parent 是整棵树的根结点
    	if (pparent == nullptr) {
    		root = subL;
    	}
    	//parent 是子树的根结点
    	else {
    		// 这里和左单旋一样
    		if (pparent->left == parent) {
    			pparent->left = subL;
    		}
    		else {
    			pparent->right = subL;
    		}
    	}
    	// 更新平衡因子
    	parent->bf = subL->bf = 0;
    }
    # 左右双旋(LR)

    当新节点插入到较高左子树的右侧时 (左右 / LR),需要先对左子树进行一次左旋,这时就变为 RR 情形,再对最先失去平衡的节点进行一次右旋,使其左子树的右子树成为新的根节点。

    c++
    // 左右双旋
    void RotateLR(Node* parent) {
    	Node* subL = parent->left;
    	Node* subLR = subL->right;
    	// 保存 subLR 的平衡因子,后面需要根据它来更新别的平衡因子
    	int BF = subLR->bf;
     
    	// 对以 subL 为根结点的树进行左单旋
    	RotateLeft(subL);
    	// 对以 parent 为根结点的树进行右单旋
    	RotateRight(parent);
     
    	// 根据保存的平衡因子更新
    	// 当 subLR 平衡因子为 - 1,将 parent 的平衡因子更新为 1
    	if (BF == -1) parent->bf = 1;
    	// 当 subLR 平衡因子不是 - 1,将 subL 的平衡因子更新为 - 1
    	else subL->bf = -1;
    }
    # 右左双旋(RL)

    当新节点插入到较高右子树的左侧时 (右左 / RL),需要先对右子树进行一次右旋,这时就变为 LL 情形,再对最先失去平衡的节点进行一次左旋,使其右子树的左子树成为新的根节点。

    c++
    // 右左双旋
    void RotateRL(Node* parent) {
    	Node* subR = parent->right;
    	Node* subRL = subR->left;
    	// 保存 subRL 的平衡因子,后面根据它更新其他结点平衡因子
    	int BF = subRL->bf;
     
    	// 对以 subR 为根的树进行右单旋
    	RotateRight(subR);
    	// 对以 parent 为根的树进行左单旋
    	RotateLeft(parent);
     
    	// 根据保存的 subRL 平衡因子的不同来进行不同的更新。
    	if (BF == -1) subR->bf = 1;
    	else parent->bf = -1;
    }
    # 插入过程
    c++
    // 插入函数
    bool Insert(const T& _value) {
    	// 树为空,将新结点作为根结点即可
    	if (root == nullptr) {
    		root = new Node(_value);
    		return true;
    	}
     
    	// 树非空
    	Node* cur = root;
    	// 用来保存 cur 的双亲结点
    	Node* parent = nullptr;
     
    	// 为新结点的插入寻找合适的位置
    	// 同时判断是否树中有相同结点
    	while (cur) {
    		parent = cur;
    		if (_value < cur->value) {
    			cur = cur->left;
    		}
    		else if (_value > cur->value) {
    			cur = cur->right;
    		}
    		else {
    			return false;
    		}
    	}
     
    	// 构造新结点
    	cur = new Node(_value);
    	// 判断是新结点应该插在双亲的左边还是右边
    	if (_value < parent->value) {
    		parent->left = cur;
    	}
    	else {
    		parent->right = cur;
    	}
    	// 更新新结点的双亲
    	cur->parent = parent;
     
    	// 更新平衡因子
    	// 从下往上
    	while (parent) {
    		//cur 是双亲左孩子
    		// 左子树高度变高,平衡因子就减 1
    		// 平衡因子:右子树高度减左子树高度
    		if (cur == parent->left) {
    			parent->bf--;
    		}
    		// 右子树变高,平衡因子加 1
    		else {
    			parent->bf++;
    		}
     
    		// 判断更新后的平衡因子是否满足 AVL 树
    		// 证明 parent 之前只有一个孩子,而新结点插到另一个空着的位置了
    		if (parent->bf == 0) {
    			break;
    		}
    		// 证明 parent 之前没有孩子,新结点插入后只有一个孩子
    		else if (-1 == parent->bf || 1 == parent->bf) {
    			cur = parent;
    			parent = cur->parent;
    		}
    		// 如下就是 parent 的平衡因子变成 2 或 - 2 了
    		// 需要根据图片进行理解
    		else {
    			// 说明 parent 的右子树高
    			if (parent->bf == 2) {
    				// 根据 cur 的平衡因子判断是左单旋还是右左双旋
    				if (cur->bf == 1)
    					RotateLeft(parent);
    				else
    					RotateRL(parent);
    			}
    			//parent 左子树高
    			else {
    				if (cur->bf == -1)
    					RotateRight(parent);
    				else
    					RotateLR(parent);
    			}
     
    			// 使用双旋调整后就不需要向上继续调整了,可以直接退出循环
    			break;
    		}
    	}
    	return true;
    }
    \ No newline at end of file +二叉树 - 数据结构与算法 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 二叉树的一些规范 (考研)

    节点的度数:节点出发指向下一层节点的边的数量 (出度)

    N0,N1,N2N_0,N_1,N_2:分别表示度数为 1,2,3 的顶点数量

    # 结论

    任意二叉树满足:

    • n=N0+N1+N2n=N_0+N_1+N_2

    • n1=0×N0+1×N1+2×N2=N1+2N2n-1=0\times N_0+1\times N_1+2\times N_2=N_1+2N_2

    • n=1+N1+2N2n=1+N_1+2N_2

    • N2=N01N_2=N_0-1

    正则二叉树还满足:

    • N1=0N_1=0

    • n+1=2N0n+1=2N_0

    • n1=2N2n-1=2N_2

    # 树的遍历

    • 先序:根、左、右

    • 中序:左、根、右

    • 后序:左、右、根

    # 递归做法

    三行代码:遍历左子树、节点、右子树,变换顺序即可得到三种遍历方式

    # 非递归做法

    # 已知二叉树的两种遍历结果确定 (还原) 二叉树

    • 即还原这颗二叉树,当且仅当这两种遍历中有一个是中序遍历!

    • 如果遍历结果是打印出来的元素结果,还需要每个元素不一样?!结点序列的话由于是指针则不需要?

    • 基本思路:已知先序和后序遍历都能容易确定谁是树根,再由中序遍历才确定左右子树

    # 线索二叉树 Threaded binary trees

    一个具有 n 个结点的二叉树若采用二叉链表存储结构, 在 2n 个指针域中只有 n-1 个指针域是用来存储结点孩 子的地址,而另外 n+1 个指针域存放的都是 NULL。

    因此,可以利用某结点空的左指针域 (lchild) 指出该 结点在某种遍历序列中的直接前驱结点的存储地址, 利用结点空的右指针域 (rchild) 指出该结点在某种遍历序列中的直接后继结点的存储地址;对于那些非空 的指针域,则仍然存放指向该结点左、右孩子的指针。 这样,就得到了一棵线索二叉树。

    由于序列可由不同的遍历方法得到,因此,线索树有 先序线索二叉树、中序线索二叉树、后序线索二叉树 三种。把二叉树改造成线索二叉树的过程称为线索化。

    # 哈夫曼树

    哈夫曼树 (Huffman Tree) 是指对于一组带有确定权值的叶结点,构造的具有最小带权路径长度的二叉树,也称最优二叉树 (Optimal Binary Tree)。

    设二叉树具有 n 个带权值的叶结点,那么从根结点到各个叶结点的路径长度与 相应结点权值的乘积之和,称为二叉树的带权路径长度,记为:

    WPL=k=1nWPL=\sum_{k=1}^{n}

    其中 WkW_k 为第 k 个叶结点的权值,LkL_k 为第 k 个叶结点的路径长度。

    在给定一组具有确定权值的叶结点,可以构造出不同的带权二叉树。这些形状不同的二叉树的带权路径长度 WPL 将各不相同。

    哈夫曼树是正则二叉树,即没有度数为 1 的顶点。

    # 构造

    由相同权值的一组叶结点所构成的二叉树有不同的形态和不同的带权路径长度,那么如何找到带权路径长度最小的二叉树呢?

    根据哈夫曼树的定义,一棵二叉树要使其 WPL 值最小,必须使权值越大的叶结点越靠近根结点,而权值越小的叶结点越远离根结点。

    哈夫曼 (Huffman) 依据这一特点提出了一种方法。

    1. 由给定的 n 个权值 {W1W_1,W2W_2,...,WnW_n} 构造 n 棵只有一个叶结点的二叉树,从而得到 一个二叉树的集合 F={T1T_1,T2T_2,...,TnT_n};

    2. 在 F 中选取根结点的权值最小和次小的两棵二叉树作为左、右子树构造一棵新的二叉树,这棵新的二叉树根结点的权值为其左、右子树根结点权值之和;

    3. 在集合 F 中删除作为左、右子树的两棵二叉树,并将新建立的二叉树加入到集合 F 中;

    4. 重复 (2)、(3) 两步,当 F 中只剩下一棵二叉树时,这棵二叉树便是所要建立的哈夫曼树。

    # 哈夫曼编码

    哈夫曼编码 (Huffman Coding) 是一种编码方式,是一种用于无损数据压 缩的熵编码 (权编码) 算法,是 David A. Huffman 发明的。

    # 哈夫曼编码代码

    c++
    typedef struct  // 哈夫曼编码的存储结构
    { char bits[N];  // 保存编码的数组 
        int start;   // 编码有效起始位置,该位置之后的 01 串为字符的编码 
        char ch;  // 字符
    } CodeType ;
    CodeType code[N+1]; // 字符编码数组,下标为 0 的单元空出
    c++
    huffmanCode(HufmTree tree[],CodeType code[]) /* 利用哈夫曼树求字符的哈夫曼编码 */
    {
        int i,c,p;
        for(i = 1; i<=N; i++)  // N 次循环,分别得到 N 个字符的编码
        {
            code[i].start = N; c = i;
            p = tree[i].parent; // 获取字符的双亲
            while (p != 0) // 一直往上层查找,直到根结束
            {
                code[i].start--;
                if ( tree[p].lchild ==c ) code[i].bits[code[i].start] = '0';
                else code[i].bits[code[i].start] = '1';
                c = p;
                p = tree[p].parent;
            }
        }
    }

    # 二叉查找树

    BST (Binary Search Tree) 递归定义:

    1. 根节点:左子树所有节点的值 < 根节点的值 < 右子树所有节点的值

    2. 任何节点的左子树右子树也都是 BST

    # 二叉树的查找 (二叉查找树实现二分查找函数)

    递归实现:

    c++
    //Recursive auxiliary function
    template <class Record>
    Binary_node<Record>*
    Search_tree<Record> :: search_for_node
    (Binary_node<Record>*root1, const Record &target)const
    {
        if (root1 == NULL || root1->data == target) return root;
        if (root1->data < target)return search_for_node(root1->right, target);
        return search_for_node(root1->left, target);
    }
    c++
    //Nonrecursive version
    template <elass Record>
    Binary_node<Record>*
    Search_tree<Record> :: search_for_node
    (Binary_node<Record>*root1, const Record &target) const
    {
        while (root1 != NULL && root1->data != target){
            if (root1->data < target) root1 = root1->right;
            else root1 = root1->left;
        }
        return root1:
    }
    c++
    //Public method for tree search
    template <class Record>
    Error_code
    Search_tree<Record> :: tree_search(Record &target) const
    {
        Error_code_result = success;
        Binary_node<Record> *found = search_for_node(root, target);
        if (found == NULL) result = not_present;
        else target = found->data;
        return result;
    }

    # BST 的插入

    c++
    template <class Record>
    Error_code
    Search_tree<Record> :: insert(const Record & new_data)
    {
        return search_and_insert(root, new_data);
    }
    c++
    template <class Record>
    Error_code
    Search_tree<Record> :: search_and_insert
    (Binary_node<Record> *&root1, const Record & new_data)
    {
        if (root1 == NULL)
        {
            root1 = new Binary_node<Record>(new_data);
            return success;
        }
        if (root1->data == new_data) return duplicate_error;
        if (root1->data > new_data) return search_and_insert(root1->left, new_data);
        return search_and_insert root1->right, new_data);
    }

    # BST 的遍历

    BST 中序遍历的结果就是所有数据升序排序结果

    # BST 的排序

    实际上与快排等价

    平均情况下 BST 排序 (Treesort) 的时间复杂度为:

    2nln(n)+O(n)1.39nlgn+O(n)2nln(n)+O(n)\approx1.39nlgn+O(n)

    # BST 的删除

    1. 要删除的节点没有子节点
      直接删除即可

    2. 要删除的节点只有一个子节点(子树)
      以子树根节点替换该节点再删除该节点即可

    3. 要删除的节点有两个子节点(子树)
      要删除节点 x
      有两种互相对称的做法:

      沿着左子树的右链走到尽头的节点 w 的元素是 x 左子树里最大的元素
      只需将 w 元素赋值给 x,再删除 w 节点即可

      沿着右子树的左链走到尽头的节点 w 的元素是 x 右子树里最小的元素
      只需将 w 元素赋值给 x,再删除 w 节点即可

    c++
    template <class Record>
    Error_code
    Search_tree<Record> :: search_and_destroy
    (Binary_node<Record>*&root1, const Record &target)
    {
        if (root1 == NULL || root1->data == target)return remove_root(root1);
        if (target < root1->data)return search_and_destroy(root1->left, target);
        return search_and_destroy(root1->right, target);
    }
    c++
    template <class Record>
    Error_code
    Search_tree<Record> :: remove_root
    (Binary_node<Record>*&root1)
    {
        if (root1 == NULL) return not_present;
        Binary_node<Record> *to_delete = root1;
        if (root1->right == NULL) root1 = root1->left;
        else if (root1->left == NULL) root1 = root1->right;
        else
        {
            to_delete = root1->left;
            Binary_node<Record> *parent= root1;
            while (to_delete->right != NULL)
            {
                parent = to_delete;
                to_delete = to_delete->right;
            }
            root1->data = to_delete->data;
            if (parent = root1) root1->left = to_delete->left; // 重点
            else parent->right = to_delete->left; // 重点
        }
        delete to_delete; return success;
    }

    # 平衡二叉树

    • 二叉树结构会影响查找效率

      • 二叉树结构越趋向于完全二叉树则时间复杂度越接近O(log(n))O(log(n)),效率越好

      • 二叉树结构越趋向于线性链表则时间复杂度越接近于O(n)O(n),效率越差

        • 最坏的情况会退化为O(n)O(n) 的线性查找复杂度

    平均访问节点次数:2ln(n)1.39log(n)2ln(n)\approx1.39log(n)

    平均比较节点次数:4ln(n)2.77log(n)4ln(n)\approx2.77log(n)

    也就是为什么我们需要自平衡二叉查找树

    # AVL Trees

    AVL 是一种平衡二叉树 (自平衡二叉查找树),过于复杂,没有太大优势,实际更常用的是红黑树来实现

    # 平衡因子 BF (Balance Factor)

    BF=左子树深度右子树深度BF=左子树深度-右子树深度

    # AVL 定义

    任意节点的 BF 只能为 - 1、0 或 1

    # 插入元素

    插入元素会改变 BF,需要调整来维持二叉树的平衡

    AVL 树采用四种 “旋转” 调整操作来实现

    # 代码实现:

    https://blog.csdn.net/weixin_57761086/article/details/127152576

    此处定义:BF=右子树深度左子树深度BF=右子树深度-左子树深度

    # 数据结构定义
    c++
    // 结点类
    template <typename T>
    struct AVLTreeNode {
    	AVLTreeNode<T>* left;// 左孩子指针
    	AVLTreeNode<T>* right;// 右孩子指针
    	AVLTreeNode<T>* parent;// 双亲指针
    	T value;// 存储元素的变量
    	int bf;// 平衡因子
     
    	// 用传入的值构造结点
    	AVLTreeNode(const T& _value = T())
    		:left(nullptr)
    		,right(nullptr)
    		,parent(nullptr)
    		,value(_value)
    		,bf(0)
    	{}
    };
    c++
    //AVL 树类
    template<typename T>
    class AVLTree {
    	// 给结点类起别名,方便使用
    	typedef AVLTreeNode<T> Node;
    private:
    	// 指向根结点的结点指针
    	Node* root;
    };
    c++
    // 无参构造
    // 让根结点指针指向空即可
    AVLTree():root(nullptr){}
    # 左旋转(RR)

    当新节点插入到较高右子树的右侧时 (右右 / RR),需要对最先失去平衡的节点进行一次左旋,使其右子树成为新的根节点,原根节点成为其左子树。

    c++
    // 左单旋
    void RotateLeft(Node* parent) {
    	// 这里的指针设计在讲解旋转的时候已经解释过
    	Node* subR = parent->right;
    	Node* subRL = subR->left;
     
    	// 用 pparent (祖父结点) 来保存 parent 结点的双亲
    	// 因为 parent 并不一定是整棵树的根,还有可能只是一棵子树的根
    	Node* pparent = parent->parent;
     
    	// 第一步:首先让双亲的右孩子指针指向 subRL
    	parent->right = subRL;
     
    	// 之所以要判断是因为 subRL 结点可能不存在
    	if (subRL)
    		// 更新双亲
    		subRL->parent = parent;
     
    	// 第二步:让 subR 的左孩子指针指向 parent
    	subR->left = parent;
     
    	// 不要忘记给这些结点更新它们的双亲
    	parent->parent = subR;
    	subR->parent = pparent;
     
    	// 祖父结点是空,说明 parent 结点是整棵树的根结点
    	if (pparent == nullptr) {
    		root = subR;
    	}
    	// 说明 parent 是子树的根结点
    	else {
    		// 原本祖父结点的孩子是 parent
    		// 经过旋转他的孩子变成了 subR
    		// 需要判断 parent 是祖父结点的左孩子还是右孩子,然后让对应指针指向 subR 结点
    		if (pparent->left == parent) {
    			pparent->left = subR;
    		}
    		else {
    			pparent->right = subR;
    		}
    	}
    	// 更新平衡因子
    	parent->bf = subR->bf = 0;
    }
    # 右旋转(LL)

    当新节点插入到较高左子树的左侧时 (左左 / LL),需要对最先失去平衡的节点进行一次右旋,使其左子树成为新的根节点,原根节点成为其右子树。

    c++
    // 右单旋
    void RotateRight(Node* parent) {
    	Node* subL = parent->left;
    	Node* subLR = subL->right;
     
    	// 祖父结点
    	Node* pparent = parent->parent;
     
    	// 第一步:让 parent 的左孩子指针指向 subLR
    	parent->left = subLR;
    	// 之所以要判断是因为 subLR 结点可能不存在
    	if (subLR)
    		subLR->parent = parent;
    	// 第二步:让 subL 的右孩子指针指向 parent
    	subL->right = parent;
     
    	// 更新双亲结点
    	parent->parent = subL;
    	subL->parent = pparent;
     
    	// 祖父结点是空,说明 parent 是整棵树的根结点
    	if (pparent == nullptr) {
    		root = subL;
    	}
    	//parent 是子树的根结点
    	else {
    		// 这里和左单旋一样
    		if (pparent->left == parent) {
    			pparent->left = subL;
    		}
    		else {
    			pparent->right = subL;
    		}
    	}
    	// 更新平衡因子
    	parent->bf = subL->bf = 0;
    }
    # 左右双旋(LR)

    当新节点插入到较高左子树的右侧时 (左右 / LR),需要先对左子树进行一次左旋,这时就变为 RR 情形,再对最先失去平衡的节点进行一次右旋,使其左子树的右子树成为新的根节点。

    c++
    // 左右双旋
    void RotateLR(Node* parent) {
    	Node* subL = parent->left;
    	Node* subLR = subL->right;
    	// 保存 subLR 的平衡因子,后面需要根据它来更新别的平衡因子
    	int BF = subLR->bf;
     
    	// 对以 subL 为根结点的树进行左单旋
    	RotateLeft(subL);
    	// 对以 parent 为根结点的树进行右单旋
    	RotateRight(parent);
     
    	// 根据保存的平衡因子更新
    	// 当 subLR 平衡因子为 - 1,将 parent 的平衡因子更新为 1
    	if (BF == -1) parent->bf = 1;
    	// 当 subLR 平衡因子不是 - 1,将 subL 的平衡因子更新为 - 1
    	else subL->bf = -1;
    }
    # 右左双旋(RL)

    当新节点插入到较高右子树的左侧时 (右左 / RL),需要先对右子树进行一次右旋,这时就变为 LL 情形,再对最先失去平衡的节点进行一次左旋,使其右子树的左子树成为新的根节点。

    c++
    // 右左双旋
    void RotateRL(Node* parent) {
    	Node* subR = parent->right;
    	Node* subRL = subR->left;
    	// 保存 subRL 的平衡因子,后面根据它更新其他结点平衡因子
    	int BF = subRL->bf;
     
    	// 对以 subR 为根的树进行右单旋
    	RotateRight(subR);
    	// 对以 parent 为根的树进行左单旋
    	RotateLeft(parent);
     
    	// 根据保存的 subRL 平衡因子的不同来进行不同的更新。
    	if (BF == -1) subR->bf = 1;
    	else parent->bf = -1;
    }
    # 插入过程
    c++
    // 插入函数
    bool Insert(const T& _value) {
    	// 树为空,将新结点作为根结点即可
    	if (root == nullptr) {
    		root = new Node(_value);
    		return true;
    	}
     
    	// 树非空
    	Node* cur = root;
    	// 用来保存 cur 的双亲结点
    	Node* parent = nullptr;
     
    	// 为新结点的插入寻找合适的位置
    	// 同时判断是否树中有相同结点
    	while (cur) {
    		parent = cur;
    		if (_value < cur->value) {
    			cur = cur->left;
    		}
    		else if (_value > cur->value) {
    			cur = cur->right;
    		}
    		else {
    			return false;
    		}
    	}
     
    	// 构造新结点
    	cur = new Node(_value);
    	// 判断是新结点应该插在双亲的左边还是右边
    	if (_value < parent->value) {
    		parent->left = cur;
    	}
    	else {
    		parent->right = cur;
    	}
    	// 更新新结点的双亲
    	cur->parent = parent;
     
    	// 更新平衡因子
    	// 从下往上
    	while (parent) {
    		//cur 是双亲左孩子
    		// 左子树高度变高,平衡因子就减 1
    		// 平衡因子:右子树高度减左子树高度
    		if (cur == parent->left) {
    			parent->bf--;
    		}
    		// 右子树变高,平衡因子加 1
    		else {
    			parent->bf++;
    		}
     
    		// 判断更新后的平衡因子是否满足 AVL 树
    		// 证明 parent 之前只有一个孩子,而新结点插到另一个空着的位置了
    		if (parent->bf == 0) {
    			break;
    		}
    		// 证明 parent 之前没有孩子,新结点插入后只有一个孩子
    		else if (-1 == parent->bf || 1 == parent->bf) {
    			cur = parent;
    			parent = cur->parent;
    		}
    		// 如下就是 parent 的平衡因子变成 2 或 - 2 了
    		// 需要根据图片进行理解
    		else {
    			// 说明 parent 的右子树高
    			if (parent->bf == 2) {
    				// 根据 cur 的平衡因子判断是左单旋还是右左双旋
    				if (cur->bf == 1)
    					RotateLeft(parent);
    				else
    					RotateRL(parent);
    			}
    			//parent 左子树高
    			else {
    				if (cur->bf == -1)
    					RotateRight(parent);
    				else
    					RotateLR(parent);
    			}
     
    			// 使用双旋调整后就不需要向上继续调整了,可以直接退出循环
    			break;
    		}
    	}
    	return true;
    }
    \ No newline at end of file diff --git a/cs/data-structure-and-algorithm/critical-path/index.html b/cs/data-structure-and-algorithm/critical-path/index.html index 112266ed..bf45526b 100644 --- a/cs/data-structure-and-algorithm/critical-path/index.html +++ b/cs/data-structure-and-algorithm/critical-path/index.html @@ -1 +1 @@ -关键路径问题 - 数据结构与算法 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # AOV: Activity On Vertex

    略(不考)

    # AOE: Activity On Edge

    与 AOV 网相对应的是 AOE (Activity On Edge) ,是边表示活动的有向无环图,如图所示。图中顶点表示事件 (Event),每个事件表示在其前的所有活动已经完成,其后的活动可以开始;弧表示活动,弧上的权值表示相应活动所需的时间或费用

    与 AOE 有关的研究问题

    • 完成整个工程至少需要多少时间?
    • 哪些活动是影响工程进度 (费用) 的关键?

    工程完成最短时间:从起点到终点的最长路径长度 (路径上各活动持续时间之和)

    长度最长的路径称为关键路径,关键路径上的活动称为关键活动。 关键活动是影响整个工程的关键。

    # 求 AOE 中关键路径和关键活动

    v0v_0 是起点,从v0v_0viv_i 的最长路径长度称为事件viv_i 的最早发生时间,即是 以viv_i 为尾的所有活动的最早发生时间。

    若活动aia_i 是弧<j,k><j,k>,持续时间是dut(<j,k>)dut(<j,k>), 设:

    e(i)e(i): 表示活动aia_i 的最早开始时间;

    l(i)l(i): 在不影响进度的前提下,表示活动aia_i 的最晚开始时间;

    l(i)e(i)l(i)-e(i) 表示活动aia_i 的时间余量,若l(i)e(i)=0l(i)-e(i)=0,表示活动aia_i 是关键活动

    ve(i)v_e(i): 表示事件viv_i 的最早发生时间,即从起点到顶点viv_i 的最长路径长度

    vl(i)v_l(i): 表示事件viv_i 的最晚发生时间

    则有以下关系:

    e(i)=ve(j)l(i)=vl(k)dut(<j,k>)\begin{aligned} e(i) &= v_e(j)\\ l(i) &= v_l(k)-dut(<j,k>) \end{aligned}

    并且:

    ve(j)={0Max{ve(i)+dut(<i,j>)<vi,vj>是网中的弧}j=0,表示vj是起点j0\begin{aligned} v_e(j) = \left\{ \begin{aligned} &0\\ &Max\{ve(i)+dut(<i, j>)|<v_i, v_j>是网中的弧\} \end{aligned} \begin{aligned} j=0,表示vj是起点\\ j\neq 0 \end{aligned} \right. \end{aligned}

    含义是:源点事件的最早发生时间设为 0; 除源点外,只有进入顶点vjv_j 的所有弧所代表的活 动全部结束后,事件vjv_j 才能发生。即只有vjv_j 的所有前驱事件viv_i 的最早发生时间ve(i)v_e(i) 计算出来 后,才能计算ve(j)v_e(j)

    方法是:对所有事件进行拓扑排序,然后依次按拓扑顺序计算每个事件的最早发生时间。

    vl(j)={ve(n1)Min{vl(k)dut(<j,k>)<vj,vk>是网中的弧}j=n1,表示vj是终点jn1\begin{aligned} v_l(j) = \left\{ \begin{aligned} &v_e(n-1) \\ &Min\{v_l(k)-dut(<j, k>)|<v_j, v_k>是网中的弧\} \end{aligned} \begin{aligned} j=n-1,表示v_j是终点\\ j\neq n-1 \end{aligned} \right. \end{aligned}

    含义是:只有vjv_j 的所有后继事件vkv_k 的最晚发生时间vl(k)v_l(k) 计算出来后,才能计算vl(j)v_l(j)

    方法是:按拓扑排序的逆顺序,依次计算每个事件的最晚发生时间。

    # 算法思想

    1. 利用拓扑排序求出 AOE 网的一个拓扑序列;

    2. 从拓扑排序的序列的第一个顶点 (源点) 开始,按拓扑顺序依次计算每个事件的最早发生时间ve(i)v_e(i);

    3. 从拓扑排序的序列的最后一个顶点 (汇点) 开始,按逆拓扑顺序依次计算每个事件的最晚发生时间vl(i)v_l(i);

    # 代码实现

    c++
    void critical_path(ALGraph *G) //Adjacency List Graph
    {
        int j, k, m ; LinkNode *p ;
        if (Topologic_Sort(G)==-1)
            printf("\nAOE网中存在回路,错误!!\n\n");
        else   
        {
            for (j=0; j<G->vexnum; j++)
                ve[j]=0; // 事件最早发生时间初始化 
            for (m=0 ; m<G->vexnum; m++)
            {
                j=topol[m] ;
                p=G->adjlist[i].firstarc ;
                for(; pl=NULL; p=p->nextarc )
                {
                    k=p->adjvex;
                    if (ve[j]+p->weight>ve[k])
                        ve[k]=ve[j]+p->weight ;
                }
            }// 计算每个事件的最早发生时间 ve 值
            for ( j=0; j<G->vexnum; j++)
                vl[j]=ve[j];// 事件最晚发生时间初始化
            for (m=G->vexnum-1; m>=0, m--)
            {
                j=topol[m] ;
                p=G->adjlist[i].firstarc;
                for (; pl=NULL; p=p->nextarc)
                {
                    k=p->adjvex ;
                    if (vl[k]-p->weight<vl[j])
                        vl[i]=$v_i$[k]-p->weight ;
                }
            }// 计算每个事件的最晚发生时间 vl 值
            for (m=0 , m<G->vexnum; m++)
            {
                p=G->adjlist[m].firstarc ;
                for(; p!=NULL; p=p->nextarc )
                {
                    k=p->adjvex;
                    if ( (ve[m]+p->weight)==l[k])
                        printf("<%d, %d>, m, j");
                }
            }// 输出所有的关键活动
        }//end of else
    }

    # 算法分析

    设 AOE 网有 n 个事件,e 个活动,则算法的主要执行是:

    • 进行拓扑排序:时间复杂度是O(n+e)O(n+e)

    • 求每个事件的 ve 值和 vl 值:时间复杂度是O(n+e)O(n+e)

    • 根据 ve 值和 vl 值找关键活动:时间复杂度是O(n+e)O(n+e)

    因此,整个算法的时间复杂度是O(n+e)O(n+e)

    \ No newline at end of file +关键路径问题 - 数据结构与算法 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # AOV: Activity On Vertex

    略(不考)

    # AOE: Activity On Edge

    与 AOV 网相对应的是 AOE (Activity On Edge) ,是边表示活动的有向无环图,如图所示。图中顶点表示事件 (Event),每个事件表示在其前的所有活动已经完成,其后的活动可以开始;弧表示活动,弧上的权值表示相应活动所需的时间或费用

    与 AOE 有关的研究问题

    • 完成整个工程至少需要多少时间?
    • 哪些活动是影响工程进度 (费用) 的关键?

    工程完成最短时间:从起点到终点的最长路径长度 (路径上各活动持续时间之和)

    长度最长的路径称为关键路径,关键路径上的活动称为关键活动。 关键活动是影响整个工程的关键。

    # 求 AOE 中关键路径和关键活动

    v0v_0 是起点,从v0v_0viv_i 的最长路径长度称为事件viv_i 的最早发生时间,即是 以viv_i 为尾的所有活动的最早发生时间。

    若活动aia_i 是弧<j,k><j,k>,持续时间是dut(<j,k>)dut(<j,k>), 设:

    e(i)e(i): 表示活动aia_i 的最早开始时间;

    l(i)l(i): 在不影响进度的前提下,表示活动aia_i 的最晚开始时间;

    l(i)e(i)l(i)-e(i) 表示活动aia_i 的时间余量,若l(i)e(i)=0l(i)-e(i)=0,表示活动aia_i 是关键活动

    ve(i)v_e(i): 表示事件viv_i 的最早发生时间,即从起点到顶点viv_i 的最长路径长度

    vl(i)v_l(i): 表示事件viv_i 的最晚发生时间

    则有以下关系:

    e(i)=ve(j)l(i)=vl(k)dut(<j,k>)\begin{aligned} e(i) &= v_e(j)\\ l(i) &= v_l(k)-dut(<j,k>) \end{aligned}

    并且:

    ve(j)={0Max{ve(i)+dut(<i,j>)<vi,vj>是网中的弧}j=0,表示vj是起点j0\begin{aligned} v_e(j) = \left\{ \begin{aligned} &0\\ &Max\{ve(i)+dut(<i, j>)|<v_i, v_j>是网中的弧\} \end{aligned} \begin{aligned} j=0,表示vj是起点\\ j\neq 0 \end{aligned} \right. \end{aligned}

    含义是:源点事件的最早发生时间设为 0; 除源点外,只有进入顶点vjv_j 的所有弧所代表的活 动全部结束后,事件vjv_j 才能发生。即只有vjv_j 的所有前驱事件viv_i 的最早发生时间ve(i)v_e(i) 计算出来 后,才能计算ve(j)v_e(j)

    方法是:对所有事件进行拓扑排序,然后依次按拓扑顺序计算每个事件的最早发生时间。

    vl(j)={ve(n1)Min{vl(k)dut(<j,k>)<vj,vk>是网中的弧}j=n1,表示vj是终点jn1\begin{aligned} v_l(j) = \left\{ \begin{aligned} &v_e(n-1) \\ &Min\{v_l(k)-dut(<j, k>)|<v_j, v_k>是网中的弧\} \end{aligned} \begin{aligned} j=n-1,表示v_j是终点\\ j\neq n-1 \end{aligned} \right. \end{aligned}

    含义是:只有vjv_j 的所有后继事件vkv_k 的最晚发生时间vl(k)v_l(k) 计算出来后,才能计算vl(j)v_l(j)

    方法是:按拓扑排序的逆顺序,依次计算每个事件的最晚发生时间。

    # 算法思想

    1. 利用拓扑排序求出 AOE 网的一个拓扑序列;

    2. 从拓扑排序的序列的第一个顶点 (源点) 开始,按拓扑顺序依次计算每个事件的最早发生时间ve(i)v_e(i);

    3. 从拓扑排序的序列的最后一个顶点 (汇点) 开始,按逆拓扑顺序依次计算每个事件的最晚发生时间vl(i)v_l(i);

    # 代码实现

    c++
    void critical_path(ALGraph *G) //Adjacency List Graph
    {
        int j, k, m ; LinkNode *p ;
        if (Topologic_Sort(G)==-1)
            printf("\nAOE网中存在回路,错误!!\n\n");
        else   
        {
            for (j=0; j<G->vexnum; j++)
                ve[j]=0; // 事件最早发生时间初始化 
            for (m=0 ; m<G->vexnum; m++)
            {
                j=topol[m] ;
                p=G->adjlist[i].firstarc ;
                for(; pl=NULL; p=p->nextarc )
                {
                    k=p->adjvex;
                    if (ve[j]+p->weight>ve[k])
                        ve[k]=ve[j]+p->weight ;
                }
            }// 计算每个事件的最早发生时间 ve 值
            for ( j=0; j<G->vexnum; j++)
                vl[j]=ve[j];// 事件最晚发生时间初始化
            for (m=G->vexnum-1; m>=0, m--)
            {
                j=topol[m] ;
                p=G->adjlist[i].firstarc;
                for (; pl=NULL; p=p->nextarc)
                {
                    k=p->adjvex ;
                    if (vl[k]-p->weight<vl[j])
                        vl[i]=$v_i$[k]-p->weight ;
                }
            }// 计算每个事件的最晚发生时间 vl 值
            for (m=0 , m<G->vexnum; m++)
            {
                p=G->adjlist[m].firstarc ;
                for(; p!=NULL; p=p->nextarc )
                {
                    k=p->adjvex;
                    if ( (ve[m]+p->weight)==l[k])
                        printf("<%d, %d>, m, j");
                }
            }// 输出所有的关键活动
        }//end of else
    }

    # 算法分析

    设 AOE 网有 n 个事件,e 个活动,则算法的主要执行是:

    • 进行拓扑排序:时间复杂度是O(n+e)O(n+e)

    • 求每个事件的 ve 值和 vl 值:时间复杂度是O(n+e)O(n+e)

    • 根据 ve 值和 vl 值找关键活动:时间复杂度是O(n+e)O(n+e)

    因此,整个算法的时间复杂度是O(n+e)O(n+e)

    \ No newline at end of file diff --git a/cs/data-structure-and-algorithm/graph/index.html b/cs/data-structure-and-algorithm/graph/index.html index 6af452e2..01b9f3d6 100644 --- a/cs/data-structure-and-algorithm/graph/index.html +++ b/cs/data-structure-and-algorithm/graph/index.html @@ -1 +1 @@ -图 - 数据结构与算法 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 图的存储结构:

    1. AM: Adjacent Matrix 邻接矩阵
    2. AL: Adjacent List 邻接表
    3. RAL: Reverse Adjacent List 逆邻接表
    4. TL: Triple List 三元组顺序表 (往往是行逻辑链接的,性能更好)
    5. CL: Cross List 十字链表

    # 图按存储结构分为:

    1. AMGraph: Adjacent Matrix Graph
    2. ALGraph: Adjacent List Graph
    3. TLGraph: Triple List Graph
    4. RALGraph: Reverse Adjacent List Graph
    5. CLGraph: Cross/Orthogonal List Graph

    # 图的遍历

    # BFS

    时间复杂度取决于图的存储结构:

    • 邻接矩阵:O(n2)O(n^2)
    • 邻接表:O(n+e)O(n+e)

    # DFS

    时间复杂度取决于图的存储结构:

    • 邻接矩阵:O(n2)O(n^2)
    • 邻接表:O(n+e)O(n+e)

    # 有向图的拓扑排序

    一个图可拓扑排序的充要条件是它无环,即它是有向无环图 (DirectedAcyclic Graph,DAG)

    # 算法实现

    采用正邻接链作为 AOV 网的存储结构;

    设立堆栈 (或队列),用来暂存入度为 0 的顶点;

    伪代码
    while(堆栈/队列不为空){
        删除顶点以它为尾的弧:弧头顶点的入度减1,度数为0则入堆栈(或队列)
    }

    # 算法分析

    设 AOV 网有 n 个顶点,e 条边,则算法的主要执行是:

    • 统计各顶点的入度:时间复杂度是O(n+e)O(n+e);

    • 入度为 0 的顶点入栈:时间复杂度是O(n)O(n);

    • 排序过程:顶点入栈和出栈操作执行 n 次,入度减 1 的操作共执行 e
      次,时间复杂度是O(n+e)O(n+e);

    因此,整个算法的时间复杂度是O(n+e)O(n+e)

    # 无向图的生成树

    dfs 稍作修改即可得到生成树算法

    保证每个节点只访问一次就可以避免环

    # 无向图的生成森林

    在生成树的基础上,加上并查集

    # 有向图的强连通分量

    1. 对有向图 G 进行 dfs 生成森林 T
    2. 按中序遍历顺序编号
    3. 从编号最大的节点开始再 dfs 遍历
    4. 按 (2) 中标出的顶点编号,从编号最大的顶点开始对 G’进行深度优先搜索,得到一棵深度优先生成树。若一次完整的搜索过程没有遍历 G’的所有顶点,则从未访问的顶点中选择一个编号最大的顶点,由它开始再进行深度优先搜索,并得到另一棵深度优先生成树。在该步中,每一次深度优先搜索所得到的生成树中的顶点就是 G 的一个强连通分量的所有顶点。
    5. 重复步骤 (4),直到 G’中的所有顶点都被访问。如下图是求一棵有向树的强连通分量过程。
    \ No newline at end of file +图 - 数据结构与算法 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 图的存储结构:

    1. AM: Adjacent Matrix 邻接矩阵
    2. AL: Adjacent List 邻接表
    3. RAL: Reverse Adjacent List 逆邻接表
    4. TL: Triple List 三元组顺序表 (往往是行逻辑链接的,性能更好)
    5. CL: Cross List 十字链表

    # 图按存储结构分为:

    1. AMGraph: Adjacent Matrix Graph
    2. ALGraph: Adjacent List Graph
    3. TLGraph: Triple List Graph
    4. RALGraph: Reverse Adjacent List Graph
    5. CLGraph: Cross/Orthogonal List Graph

    # 图的遍历

    # BFS

    时间复杂度取决于图的存储结构:

    • 邻接矩阵:O(n2)O(n^2)
    • 邻接表:O(n+e)O(n+e)

    # DFS

    时间复杂度取决于图的存储结构:

    • 邻接矩阵:O(n2)O(n^2)
    • 邻接表:O(n+e)O(n+e)

    # 有向图的拓扑排序

    一个图可拓扑排序的充要条件是它无环,即它是有向无环图 (DirectedAcyclic Graph,DAG)

    # 算法实现

    采用正邻接链作为 AOV 网的存储结构;

    设立堆栈 (或队列),用来暂存入度为 0 的顶点;

    伪代码
    while(堆栈/队列不为空){
        删除顶点以它为尾的弧:弧头顶点的入度减1,度数为0则入堆栈(或队列)
    }

    # 算法分析

    设 AOV 网有 n 个顶点,e 条边,则算法的主要执行是:

    • 统计各顶点的入度:时间复杂度是O(n+e)O(n+e);

    • 入度为 0 的顶点入栈:时间复杂度是O(n)O(n);

    • 排序过程:顶点入栈和出栈操作执行 n 次,入度减 1 的操作共执行 e
      次,时间复杂度是O(n+e)O(n+e);

    因此,整个算法的时间复杂度是O(n+e)O(n+e)

    # 无向图的生成树

    dfs 稍作修改即可得到生成树算法

    保证每个节点只访问一次就可以避免环

    # 无向图的生成森林

    在生成树的基础上,加上并查集

    # 有向图的强连通分量

    1. 对有向图 G 进行 dfs 生成森林 T
    2. 按中序遍历顺序编号
    3. 从编号最大的节点开始再 dfs 遍历
    4. 按 (2) 中标出的顶点编号,从编号最大的顶点开始对 G’进行深度优先搜索,得到一棵深度优先生成树。若一次完整的搜索过程没有遍历 G’的所有顶点,则从未访问的顶点中选择一个编号最大的顶点,由它开始再进行深度优先搜索,并得到另一棵深度优先生成树。在该步中,每一次深度优先搜索所得到的生成树中的顶点就是 G 的一个强连通分量的所有顶点。
    5. 重复步骤 (4),直到 G’中的所有顶点都被访问。如下图是求一棵有向树的强连通分量过程。
    \ No newline at end of file diff --git a/cs/data-structure-and-algorithm/hash/index.html b/cs/data-structure-and-algorithm/hash/index.html index 4fab8665..0ae2cffd 100644 --- a/cs/data-structure-and-algorithm/hash/index.html +++ b/cs/data-structure-and-algorithm/hash/index.html @@ -1 +1 @@ -哈希表 - 数据结构与算法 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 哈希表 / 散列表

    应用哈希函数,由记录的关键字确定记录在表中的地址,并将记录放入此地址,这样构成的表叫哈希表

    # 哈希查找 / 散列查找

    利用哈希函数进行查找的过程叫哈希查找

    # 哈希映射函数 / 散列函数

    在记录的关键字与记录的存储地址之间建立的一种对应关系叫哈希函数

    # 原始 hash 函数

    简单一点的 hash 算法就是直接取模 (除留余数法)。

    高级一点的 hash 算法会模乘以一个大素数来将源数据的分布规律变得不那么显著,也就是说,不管输入的是一串分布很紧凑的数字序列还是很稀疏的数字序列,生成的结果都是非常分散的数字序列。

    linux 内核代码里面的这个 hash 要更高一级,包含 2 个特征:
    最接近 32 位或者者 64 位上限黄金分割点的素数,
    只要要最少的移位和求和就能生成。
    这个素数定义如下:

    c
    /* 2^31 + 2^29 - 2^25 + 2^22 - 2^19 - 2^16 + 1 */
    #define GOLDEN_RATIO_PRIME_32 0x9e370001UL

    前者确保即便是 1,2 这样的小数也会被乘溢出,被轻易打散。素数非常大还可以使无论输入如何,生成的哈希值的高位都是足够随机的,有助于改善生成哈希值高位总是相近的问题。

    后者确保任何一个整数乘这个大素数的时候都可以通过最少的移位和求和就可得到乘积,不用傻傻的真的相乘。

    # 直接定址法

    取关键字或关键字的某个线性函数作哈希地址,即H(key)=keyH(key)=keyH(key)=akey+bH(key)=a·key+b (a,b 为常数)

    特点:直接定址法所得地址集合与关键字集合大小相等,不会发生冲突,但实际中很少使用。

    # 数字分析法

    对关键字进行分析,取关键字的若干位或组合作为哈希地址。

    适用于关键字位数比哈希地址位数大,且可能出现的关键字事先知道的情况。

    # 平方取中法

    将关键字平方后取中间几位作为哈希地址。一个数平方后中间几位和数的每一位都有关,则由随机分布的关键字得到的散列地址也是随机的。散列函数所取的位数由散列表的长度决定。

    适于不知道全部关键字情况,是一种较为常用的方法

    # 折叠法

    将关键字分割成位数相同的几部分 (最后一部分可以不同),然后取这几部分的叠加和作为哈希地址。

    数位叠加有移位叠加和间界叠加两种:

    • 移位叠加:将分割后的几部分低位对齐相加。
    • 间界叠加:从一端到另一端沿分割界来回折迭,然后对齐相加。

    适于关键字位数很多,且每一位上数字分布大致均匀情况。

    # 除留余数法

    取关键字被某个不大于哈希表表长 m 的数 p 除后所得余数作哈希地址,即:
    H(key)=key%p(pm)H(key)=key\ \% \ p\ (p\leq m)
    是一种简单、常用的哈希函数构造方法。

    利用这种方法的关键是 p 的选取,p 选的不好,容易产生同义词。p 的选取的分析:

    • 选取p=2i(pm)p=2^i\ (p\leq m): 运算便于用移位来实现,但等于将关键字的高位忽略而仅留下低位二进制数。高位不同而低位相同的 关键字是同义词。
    • 选取p=q×f(qf都是质因数,pm)p=q\times f\ (q、f 都是质因数,p\leq m): 则所有含有 q 或 f 因子的关键字的散列地址均是 q 或 f 的倍数。
    • 选取pp 为素数或p=q×f(qf是质数且均大于20pm)p=q\times f\ (q、f是质数且均大于20,p\leq m): 常用的选取方法,能减少冲突出现的可能性。

    # 随机数法

    取关键字的随机函数值作哈希地址,即H(key)=random(key)H(key) = random(key)

    当散列表中关键字长度不等时,该方法比较合适。

    # 冲突处理

    # 开放定址法

    当冲突发生时,形成某个探测序列;按此序列逐个探测散列表中的其他地址,直到找到给定的关键字或一个空地址 (开放的地址) 为止,将发生冲突的记录放到该地址中。散列地址的计算公式是:

    Hi(key)=(H(key)+di)%m,i=1,2,...,k(km1)H_i(key)=(H(key)+d_i) \% m, i=1, 2, ..., k (k \leq m-1)

    其中:
    H(key)H(key):哈希函数;
    mm:散列表长度;
    did_i:第 i 次探测时的增量序列;
    Hi(key)H_i(key):经第 i 次探测后得到的散列地址。

    核心在于探测序列did_i,即如何形成探测序列。

    # 线性探测法

    将散列表T[0...m1]T[0 ...m-1] 看成循环向量。当发生冲突时,从初次发生冲突的 位置依次向后探测其他的地址。

    增量序列为:di=1,2,3,...,m1d_i=1, 2, 3, ..., m-1

    设初次发生冲突的地址是 h,则依次探测 T[h+1],T[h+2]...,T[h+1],T[h+2]..., 直到T[m1]T[m-1] 之后又循环到表头,再次探测T[0]T[1]...T[0],T[1]...,直到 T [h-1]$。探测过程终止的情况是:

    • 探测到的地址为空:表中没有记录。
      • 若是查找则失败;
      • 若是插入则将记录写入到该地址;
    • 探测到的地址有给定的关键字:
      • 若是查找则成功;
      • 若是插入则失败;
    • 直到T[h1]T[h-1] 之后回到T[h]T[h]:
      • 仍未探测到空地址或给定的关键字,散列表满。

    优点:只要空间未满,总能找到空闲的位置存放新插入的数据,能充分利用整个表的空间。

    问题:例如原本有两个 cluster 数据,随着哈希表不断扩充,两个 cluster 会逐渐聚集为一个更大的 cluster,产生数据聚集的问题。

    循环上限did_iii 最大为nn.

    # 二次探测法 (平方探测法) (变体很多,需要看题)

    原始did_idi=i2d_i = i^2

    原始序列:di=12,22,32...d_i = 1^2, 2^2, 3^2 ...

    改进did_idi=(1)i1i+122d_i = (-1)^{i-1}\cdot\left\lfloor\frac{i+1}{2}\right\rfloor^2

    改进序列:di=12,12,22,22,32,32...di = 1^2, -1^2, 2^2, -2^2, 3^2, -3^2 ...

    循环上限:did_iii 最大为nn,有时可以是\frac{n}

    二次探测针对线性探测法的 “聚集” 问题进行了改进,避免了 “聚集” 现象产生。

    原始did_i 在不断扩充哈希表时会导致一半的空间用不上,改进的did_i 针对这一问题进行了改进。

    优点:探测序列跳跃式地散列到整个表中,不易产生 “冲突聚集” 现象。

    问题:不能保证探测到散列表的所有地址,不能充分利用整个表的空间。(“表满不了”)

    # 伪随机探测法

    增量序列使用一个伪随机函数来生成一个落在闭区间 [1,m-1] 的随机序列。

    # 再哈希法

    这种方法是同时构造多个不同的哈希函数:

    Hi=RHikey),i=12,3,nH_i=RH_i(key),i=1,2,3,…,n

    RHiRH_i:一组不同的哈希函数,第 1 次发生冲突时用RH1RH_1 计算,第 2 次发生冲突时用RH2RH_2 计算,以此类推直到某个HiH_i 不再发生冲突为止。

    优点:不易产生聚集

    问题:增加了计算时间

    # 链定址法 (拉链法)

    将所有关键字为同义词 (散列地址相同) 的记录存储在一个单链表中,并用一维数组存放链表的头指针

    设散列表长为 m,定义一个一维指针数组: RecNode *linkhash[m] ,其中 RecNode 是结点类型,每个分量的初值为空。

    凡是散列地址为 k 的记录都插入到以 linkhash[k] 为头指针的链表中,插入位置可以在表头或表尾或按关键字排序插入。

    注意:若某一位置已存储的冲突元素有 4 个,查找失败需要比较 5 次,包括最后一次空指针。

    # 建立公共溢出区

    这种方法的基本思想是:将哈希表分为基本表和溢出表两部分,凡是和基本表发生冲突的元素,一律填入溢出表.(注意:在这个方法里面是把元素分开两个表来存储)

    # 哈希表的删除操作

    不能抹除该位置的元素,而是需要一个特殊字符 "special key" 取代原来的位置,代表这里原本有元素但是被删除了。

    # 哈希表查找分析

    由于 “冲突” 的存在,哈希表效率的评价仍然需要 ASL。

    哈希表的 填满因子 / 装载因子 λ\lambda ( 或α\alpha ) 定义:

    λ=已经插入的元素数哈希表长度\lambda=\frac{已经插入的元素数}{哈希表长度}

    The expected number U(2)U(2) of probes in an unsuccessful search is:

    U(λ)=k=1kλk1(1λ)=11λU(\lambda)=\sum_{k=1}^{\infty}k\lambda^{k-1}(1-\lambda)=\frac{1}{1-\lambda}

    So the expected number of probes for successful searches (AVL) is:

    S(λ)=1λ0λU(λ)dx=1λln11λS(\lambda)=\frac{1}{\lambda}\int_{0}^{\lambda}{U(\lambda)dx}=\frac{1}{\lambda}ln\frac{1}{1-\lambda}

    线性探测法的平均查找长度是:

    Snl成功12×(1+11λ)Snl失败12×(1+1(1λ)2)S_{nl成功}≈\frac{1}{2}\times(1+\frac{1}{1-\lambda})\\ S_{nl失败}≈\frac{1}{2}\times(1+\frac{1}{(1-\lambda)^2})

    二次 / 平方探测法的平均查找长度是:

    Snl成功1λ×ln(1λ)Snl失败1λS_{nl成功}≈-\frac{1}{\lambda}\times ln(1-\lambda)\\ S_{nl失败}≈\frac{1}{\lambda}

    链地址法解决冲突的平均查找长度是:

    Snl成功1+λ2Snl失败λ+eλS_{nl成功}≈1+\frac{\lambda}{2}\\ S_{nl失败}≈\lambda+e^{-\lambda}

    # 代码例题

    假设构建哈希表使用线性探查法解决冲突,试计算每次查找的关键字比较次数。

    1. 可能的关键字为正整数;
    2. 使用哈希函数 h (k,M)= k % M, M 为哈希表的长度;
    3. 因为 0 不是关键字,所以,假定用 0 表示空单元;
    4. 为了区分空单元和删除后的 “空单元”,给每个单元添加一个布尔型标识,表示该单元是否空;空单元设置为 true,非空设置为 false;
    5. 假定删除关键字时,只将标识置为 true, 关键字域不变。
    c++
    #include <iostream>
    #include <vector>
    using namespace std;
    typedef int Key;
    typedef pair<Key, bool>  Cell;
    //a cell consists of a key and a flag of bool. If the cell is empty, the second component is set to true, otherwise false.
    typedef vector<Cell> Table; //type of a hash table
    //hash function, where M should be the hash table size.
    size_t h(Key k, size_t M = 11){
        return k % M;
    }
    // 它返回线性探查法解决冲突查找是否成功、查找终止的下标以及关键字的比较次数
    pair<pair<bool, size_t>, int> lookup(const Table& t, const Key k){
        size_t M=t.size();
        size_t hi=h(k,M),j,i;
        int count = 0;
        for(j=0;j<M;++j){
            i=(hi+j)%M;
            ++count;
            if(t[i].first==0)break;
            else if(t[i].first==k&&!t[i].second)return make_pair(make_pair(1,i),count);
        }
        return make_pair(make_pair(0,i),count);
    }
    \ No newline at end of file +哈希表 - 数据结构与算法 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 哈希表 / 散列表

    应用哈希函数,由记录的关键字确定记录在表中的地址,并将记录放入此地址,这样构成的表叫哈希表

    # 哈希查找 / 散列查找

    利用哈希函数进行查找的过程叫哈希查找

    # 哈希映射函数 / 散列函数

    在记录的关键字与记录的存储地址之间建立的一种对应关系叫哈希函数

    # 原始 hash 函数

    简单一点的 hash 算法就是直接取模 (除留余数法)。

    高级一点的 hash 算法会模乘以一个大素数来将源数据的分布规律变得不那么显著,也就是说,不管输入的是一串分布很紧凑的数字序列还是很稀疏的数字序列,生成的结果都是非常分散的数字序列。

    linux 内核代码里面的这个 hash 要更高一级,包含 2 个特征:
    最接近 32 位或者者 64 位上限黄金分割点的素数,
    只要要最少的移位和求和就能生成。
    这个素数定义如下:

    c
    /* 2^31 + 2^29 - 2^25 + 2^22 - 2^19 - 2^16 + 1 */
    #define GOLDEN_RATIO_PRIME_32 0x9e370001UL

    前者确保即便是 1,2 这样的小数也会被乘溢出,被轻易打散。素数非常大还可以使无论输入如何,生成的哈希值的高位都是足够随机的,有助于改善生成哈希值高位总是相近的问题。

    后者确保任何一个整数乘这个大素数的时候都可以通过最少的移位和求和就可得到乘积,不用傻傻的真的相乘。

    # 直接定址法

    取关键字或关键字的某个线性函数作哈希地址,即H(key)=keyH(key)=keyH(key)=akey+bH(key)=a·key+b (a,b 为常数)

    特点:直接定址法所得地址集合与关键字集合大小相等,不会发生冲突,但实际中很少使用。

    # 数字分析法

    对关键字进行分析,取关键字的若干位或组合作为哈希地址。

    适用于关键字位数比哈希地址位数大,且可能出现的关键字事先知道的情况。

    # 平方取中法

    将关键字平方后取中间几位作为哈希地址。一个数平方后中间几位和数的每一位都有关,则由随机分布的关键字得到的散列地址也是随机的。散列函数所取的位数由散列表的长度决定。

    适于不知道全部关键字情况,是一种较为常用的方法

    # 折叠法

    将关键字分割成位数相同的几部分 (最后一部分可以不同),然后取这几部分的叠加和作为哈希地址。

    数位叠加有移位叠加和间界叠加两种:

    • 移位叠加:将分割后的几部分低位对齐相加。
    • 间界叠加:从一端到另一端沿分割界来回折迭,然后对齐相加。

    适于关键字位数很多,且每一位上数字分布大致均匀情况。

    # 除留余数法

    取关键字被某个不大于哈希表表长 m 的数 p 除后所得余数作哈希地址,即:
    H(key)=key%p(pm)H(key)=key\ \% \ p\ (p\leq m)
    是一种简单、常用的哈希函数构造方法。

    利用这种方法的关键是 p 的选取,p 选的不好,容易产生同义词。p 的选取的分析:

    • 选取p=2i(pm)p=2^i\ (p\leq m): 运算便于用移位来实现,但等于将关键字的高位忽略而仅留下低位二进制数。高位不同而低位相同的 关键字是同义词。
    • 选取p=q×f(qf都是质因数,pm)p=q\times f\ (q、f 都是质因数,p\leq m): 则所有含有 q 或 f 因子的关键字的散列地址均是 q 或 f 的倍数。
    • 选取pp 为素数或p=q×f(qf是质数且均大于20pm)p=q\times f\ (q、f是质数且均大于20,p\leq m): 常用的选取方法,能减少冲突出现的可能性。

    # 随机数法

    取关键字的随机函数值作哈希地址,即H(key)=random(key)H(key) = random(key)

    当散列表中关键字长度不等时,该方法比较合适。

    # 冲突处理

    # 开放定址法

    当冲突发生时,形成某个探测序列;按此序列逐个探测散列表中的其他地址,直到找到给定的关键字或一个空地址 (开放的地址) 为止,将发生冲突的记录放到该地址中。散列地址的计算公式是:

    Hi(key)=(H(key)+di)%m,i=1,2,...,k(km1)H_i(key)=(H(key)+d_i) \% m, i=1, 2, ..., k (k \leq m-1)

    其中:
    H(key)H(key):哈希函数;
    mm:散列表长度;
    did_i:第 i 次探测时的增量序列;
    Hi(key)H_i(key):经第 i 次探测后得到的散列地址。

    核心在于探测序列did_i,即如何形成探测序列。

    # 线性探测法

    将散列表T[0...m1]T[0 ...m-1] 看成循环向量。当发生冲突时,从初次发生冲突的 位置依次向后探测其他的地址。

    增量序列为:di=1,2,3,...,m1d_i=1, 2, 3, ..., m-1

    设初次发生冲突的地址是 h,则依次探测 T[h+1],T[h+2]...,T[h+1],T[h+2]..., 直到T[m1]T[m-1] 之后又循环到表头,再次探测T[0]T[1]...T[0],T[1]...,直到 T [h-1]$。探测过程终止的情况是:

    • 探测到的地址为空:表中没有记录。
      • 若是查找则失败;
      • 若是插入则将记录写入到该地址;
    • 探测到的地址有给定的关键字:
      • 若是查找则成功;
      • 若是插入则失败;
    • 直到T[h1]T[h-1] 之后回到T[h]T[h]:
      • 仍未探测到空地址或给定的关键字,散列表满。

    优点:只要空间未满,总能找到空闲的位置存放新插入的数据,能充分利用整个表的空间。

    问题:例如原本有两个 cluster 数据,随着哈希表不断扩充,两个 cluster 会逐渐聚集为一个更大的 cluster,产生数据聚集的问题。

    循环上限did_iii 最大为nn.

    # 二次探测法 (平方探测法) (变体很多,需要看题)

    原始did_idi=i2d_i = i^2

    原始序列:di=12,22,32...d_i = 1^2, 2^2, 3^2 ...

    改进did_idi=(1)i1i+122d_i = (-1)^{i-1}\cdot\left\lfloor\frac{i+1}{2}\right\rfloor^2

    改进序列:di=12,12,22,22,32,32...di = 1^2, -1^2, 2^2, -2^2, 3^2, -3^2 ...

    循环上限:did_iii 最大为nn,有时可以是\frac{n}

    二次探测针对线性探测法的 “聚集” 问题进行了改进,避免了 “聚集” 现象产生。

    原始did_i 在不断扩充哈希表时会导致一半的空间用不上,改进的did_i 针对这一问题进行了改进。

    优点:探测序列跳跃式地散列到整个表中,不易产生 “冲突聚集” 现象。

    问题:不能保证探测到散列表的所有地址,不能充分利用整个表的空间。(“表满不了”)

    # 伪随机探测法

    增量序列使用一个伪随机函数来生成一个落在闭区间 [1,m-1] 的随机序列。

    # 再哈希法

    这种方法是同时构造多个不同的哈希函数:

    Hi=RHikey),i=12,3,nH_i=RH_i(key),i=1,2,3,…,n

    RHiRH_i:一组不同的哈希函数,第 1 次发生冲突时用RH1RH_1 计算,第 2 次发生冲突时用RH2RH_2 计算,以此类推直到某个HiH_i 不再发生冲突为止。

    优点:不易产生聚集

    问题:增加了计算时间

    # 链定址法 (拉链法)

    将所有关键字为同义词 (散列地址相同) 的记录存储在一个单链表中,并用一维数组存放链表的头指针

    设散列表长为 m,定义一个一维指针数组: RecNode *linkhash[m] ,其中 RecNode 是结点类型,每个分量的初值为空。

    凡是散列地址为 k 的记录都插入到以 linkhash[k] 为头指针的链表中,插入位置可以在表头或表尾或按关键字排序插入。

    注意:若某一位置已存储的冲突元素有 4 个,查找失败需要比较 5 次,包括最后一次空指针。

    # 建立公共溢出区

    这种方法的基本思想是:将哈希表分为基本表和溢出表两部分,凡是和基本表发生冲突的元素,一律填入溢出表.(注意:在这个方法里面是把元素分开两个表来存储)

    # 哈希表的删除操作

    不能抹除该位置的元素,而是需要一个特殊字符 "special key" 取代原来的位置,代表这里原本有元素但是被删除了。

    # 哈希表查找分析

    由于 “冲突” 的存在,哈希表效率的评价仍然需要 ASL。

    哈希表的 填满因子 / 装载因子 λ\lambda ( 或α\alpha ) 定义:

    λ=已经插入的元素数哈希表长度\lambda=\frac{已经插入的元素数}{哈希表长度}

    The expected number U(2)U(2) of probes in an unsuccessful search is:

    U(λ)=k=1kλk1(1λ)=11λU(\lambda)=\sum_{k=1}^{\infty}k\lambda^{k-1}(1-\lambda)=\frac{1}{1-\lambda}

    So the expected number of probes for successful searches (AVL) is:

    S(λ)=1λ0λU(λ)dx=1λln11λS(\lambda)=\frac{1}{\lambda}\int_{0}^{\lambda}{U(\lambda)dx}=\frac{1}{\lambda}ln\frac{1}{1-\lambda}

    线性探测法的平均查找长度是:

    Snl成功12×(1+11λ)Snl失败12×(1+1(1λ)2)S_{nl成功}≈\frac{1}{2}\times(1+\frac{1}{1-\lambda})\\ S_{nl失败}≈\frac{1}{2}\times(1+\frac{1}{(1-\lambda)^2})

    二次 / 平方探测法的平均查找长度是:

    Snl成功1λ×ln(1λ)Snl失败1λS_{nl成功}≈-\frac{1}{\lambda}\times ln(1-\lambda)\\ S_{nl失败}≈\frac{1}{\lambda}

    链地址法解决冲突的平均查找长度是:

    Snl成功1+λ2Snl失败λ+eλS_{nl成功}≈1+\frac{\lambda}{2}\\ S_{nl失败}≈\lambda+e^{-\lambda}

    # 代码例题

    假设构建哈希表使用线性探查法解决冲突,试计算每次查找的关键字比较次数。

    1. 可能的关键字为正整数;
    2. 使用哈希函数 h (k,M)= k % M, M 为哈希表的长度;
    3. 因为 0 不是关键字,所以,假定用 0 表示空单元;
    4. 为了区分空单元和删除后的 “空单元”,给每个单元添加一个布尔型标识,表示该单元是否空;空单元设置为 true,非空设置为 false;
    5. 假定删除关键字时,只将标识置为 true, 关键字域不变。
    c++
    #include <iostream>
    #include <vector>
    using namespace std;
    typedef int Key;
    typedef pair<Key, bool>  Cell;
    //a cell consists of a key and a flag of bool. If the cell is empty, the second component is set to true, otherwise false.
    typedef vector<Cell> Table; //type of a hash table
    //hash function, where M should be the hash table size.
    size_t h(Key k, size_t M = 11){
        return k % M;
    }
    // 它返回线性探查法解决冲突查找是否成功、查找终止的下标以及关键字的比较次数
    pair<pair<bool, size_t>, int> lookup(const Table& t, const Key k){
        size_t M=t.size();
        size_t hi=h(k,M),j,i;
        int count = 0;
        for(j=0;j<M;++j){
            i=(hi+j)%M;
            ++count;
            if(t[i].first==0)break;
            else if(t[i].first==k&&!t[i].second)return make_pair(make_pair(1,i),count);
        }
        return make_pair(make_pair(0,i),count);
    }
    \ No newline at end of file diff --git a/cs/data-structure-and-algorithm/heap-sort/index.html b/cs/data-structure-and-algorithm/heap-sort/index.html index 60e1a90c..ae056786 100644 --- a/cs/data-structure-and-algorithm/heap-sort/index.html +++ b/cs/data-structure-and-algorithm/heap-sort/index.html @@ -1 +1 @@ -堆排序 - 数据结构与算法 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    区分:堆排序的堆不是堆栈的堆

    # 原理

    • 堆排序可以视为一种聪明的选择排序。
    • 堆排序适合数组这样的数据结构。
    • 选择排序需要不断在未排序部分找最大 / 小值然后放在已排序部分后面,堆排序只是将 “找最值” 的部分用 “调整堆” 代替,调整好的堆的堆顶就是我们想要的最值。
    • 堆排序基于二叉树结构,堆是一种按一定规律存储数据的二叉树。利用堆最值位于堆顶的特点,不断将堆顶的最值和堆最后一个元素交换(堆最后一个元素后面都是已排序部分),然后重新调整未排序部分使其再变回一个堆。

    # 最大堆 / 小根堆 / 小顶堆

    • 最大值在根节点 / 堆顶
    • 结点的键值都小于等于其父结点的键值
    • 排降序建最大堆

    # 最小堆 / 大根堆 / 大顶堆

    • 最小值在根节点 / 堆顶
    • 结点的键值都大于等于其父结点的键值
    • 排升序建最小堆

    # 递归函数实现

    该递归是尾部递归,可以改为循环实现

    c++
    template <class Record>
    void Sortable_list< Record>:: heap_sort ()
    /* Post: The entries of the Sortable_list have been rearranged so that their keys are sorted into nondecreasing order.
    Uses: The contiguous List implementation of Chapter 6, build_heap, and in- sert _heap. */
    {
        Record current;//temporary storage for moving entries
        int last_unsorted; //Entries beyond last _unsorted have been sorted.
        build_heap(); //First phase: Turn the list into a heap.
        for (last_unsorted = count - 1; last unsorted > 0; last_unsorted--) {
            current = entry [last unsorted]; //Extract last entry from list.
            entry [last_unsorted] = entry [0]; //Move top of heap to the end.
            insert_heap(current, O, last _unsorted -1); // Restore the heap.
        }
    }

    # 插入堆:

    c++
    template <class Record>
    void Sortable_list< Record>::insert_heap(const Record &current, int low, int high)
    /* Pre: The entries of the Sortable_list between indices low + 1and high,inclusive, form a heap. The entry in position low wil be discarded.
    Post: The entry current has been inserted into the Sortable list and theentries rearranged so that the entries between indices low and high,inclusive, form a heap.
    Uses: The class Record, and the contiguous List implementation of Chapter 6.*/
    {
        int large; // position of child of entry [low] with the larger key
        large = 2 * low + 1; // large is now the left child of low.
        while (large =< high) {
            if (large < high & entry [large] < entry [large + 1])large ++; //large+1 refers to right child of low, large is now the child of low with the largest key.
            if(current>=entry[large])break; //current belongs ni position low.
            else { //Promote entry [large ] and move down the tree.
                entry [low] = entry [large]; //The larger one is promoted, low becomes the new vacancy and large is the left child.
                low = large;
                large=2*low +1;
            }
        }
        entry [low] = current;
    }

    # 建堆:

    c++
    template <class Record>
    void Sortable_Jist< Record> :: build_heap()
    /* Post:The entries of the Sortable_list have been rearranged so that it becomes a heap.
    Uses: The contiguous List implementation of Chapter 6, and insert_heap. */
    {
        int low; // All entries beyond the position low form a heap.
        for (low = count/2 - 1; low >= 0; --low ){
            Record current = entry [low];
            insert heap(current,low, count一1);
        }
    }

    # 非递归实现

    利用数组实现,常用下标 0 作为根节点,此时孩子节点和父亲节点的计算:

    left_child = parent*2+1

    right_child = parent*2+2

    child-1 = parent*2+0/1

    parent = (child-1)>>1

    # 向上调整

    插入一个节点到堆的末尾

    c++
    void AdjustUp(HPDataType* a, size_t child){
    	size_t parent;
    	while (child > 0){
            parent = (child - 1) / 2;
    		if (a[child] > a[parent]){ // 大堆
    			swap(&a[child], &a[parent]);
    			child = parent;
    		}
    		else{
    			break;
    		}
    	}
    }

    向上调整建堆

    c++
    // 从数组第 2 个数直到最后一个数重复进行向上调整算法
    for(int i=1;i<n;i++){
        AdjustUp(a,i);
    }

    向上调整法建堆时间复杂度: O(nlogn)O(nlogn)

    # 向下调整

    c++
    void AdjustDown(HPDataType* a, size_t size, int parent){
    	size_t child = parent * 2 + 1;
    	while (child < size){
    		if (child + 1 < size && a[child + 1] > a[child]){ // 比较左右孩子
    			++child;
    		}
    		if (a[child] > a[parent]){ // 大堆
    			swap(&a[parent], &a[child]);
    			parent = child;
                size_t child = parent * 2 + 1;
    		}
    		else{
    			break;
    		}
    	}
    }
    c++
    // 从最后一个叶子的父亲开始,不断 --,直到第一个节点 (包括第一个节点),重复进行向下调整算法
    // 最后一个叶子节点的数组下标是 n-1,最后一个叶子节点的父亲的数组下标为 (n-1-1)/2 = n/2-1
    for(int i=n/2-1;i>=0;i--){
        AdjustDown(a,n,i);
    }

    向下调整法建堆时间复杂度: O(n)O(n)

    参考:
    https://blog.csdn.net/zhang_si_hang/article/details/124018889

    # 性能

    • 与快排相比,堆排序的常数部分差一点,比较次数多一些,赋值次数少一些

    • 堆排序是不稳定排序

    # 堆的应用

    堆是一种常用的可以对排序、最值查找…… 优化的数据结构、技巧。

    # 优先级队列

    基于堆实现,c++ 里为 priority_queue

    # 前 k 个最大 / 小的数

    • 经典面试题

      • 题面:给定一个有 N 个 entry 的 list ,设计一个算法, 从 N 个 entry 中找出最大的 k 个。假定 N 很大且 k 很小 (例如 N = 100 亿,k = 30)。

      • 题解:用堆排序,建堆需要时间复杂度:kl˙og2kk\dot log_{2}k;调整需要时间复杂度:Nl˙og2kN\dot log_{2}k;总共为:O((N+k)log2k)O((N+k)log_{2}k)

    # 第 k 个最大 / 小的数

    • 对顶堆
      基于堆维持最大值 / 最小值的特点,用于维持第 k 大 / 第 k 小的数据
    \ No newline at end of file +堆排序 - 数据结构与算法 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    区分:堆排序的堆不是堆栈的堆

    # 原理

    • 堆排序可以视为一种聪明的选择排序。
    • 堆排序适合数组这样的数据结构。
    • 选择排序需要不断在未排序部分找最大 / 小值然后放在已排序部分后面,堆排序只是将 “找最值” 的部分用 “调整堆” 代替,调整好的堆的堆顶就是我们想要的最值。
    • 堆排序基于二叉树结构,堆是一种按一定规律存储数据的二叉树。利用堆最值位于堆顶的特点,不断将堆顶的最值和堆最后一个元素交换(堆最后一个元素后面都是已排序部分),然后重新调整未排序部分使其再变回一个堆。

    # 最大堆 / 小根堆 / 小顶堆

    • 最大值在根节点 / 堆顶
    • 结点的键值都小于等于其父结点的键值
    • 排降序建最大堆

    # 最小堆 / 大根堆 / 大顶堆

    • 最小值在根节点 / 堆顶
    • 结点的键值都大于等于其父结点的键值
    • 排升序建最小堆

    # 递归函数实现

    该递归是尾部递归,可以改为循环实现

    c++
    template <class Record>
    void Sortable_list< Record>:: heap_sort ()
    /* Post: The entries of the Sortable_list have been rearranged so that their keys are sorted into nondecreasing order.
    Uses: The contiguous List implementation of Chapter 6, build_heap, and in- sert _heap. */
    {
        Record current;//temporary storage for moving entries
        int last_unsorted; //Entries beyond last _unsorted have been sorted.
        build_heap(); //First phase: Turn the list into a heap.
        for (last_unsorted = count - 1; last unsorted > 0; last_unsorted--) {
            current = entry [last unsorted]; //Extract last entry from list.
            entry [last_unsorted] = entry [0]; //Move top of heap to the end.
            insert_heap(current, O, last _unsorted -1); // Restore the heap.
        }
    }

    # 插入堆:

    c++
    template <class Record>
    void Sortable_list< Record>::insert_heap(const Record &current, int low, int high)
    /* Pre: The entries of the Sortable_list between indices low + 1and high,inclusive, form a heap. The entry in position low wil be discarded.
    Post: The entry current has been inserted into the Sortable list and theentries rearranged so that the entries between indices low and high,inclusive, form a heap.
    Uses: The class Record, and the contiguous List implementation of Chapter 6.*/
    {
        int large; // position of child of entry [low] with the larger key
        large = 2 * low + 1; // large is now the left child of low.
        while (large =< high) {
            if (large < high & entry [large] < entry [large + 1])large ++; //large+1 refers to right child of low, large is now the child of low with the largest key.
            if(current>=entry[large])break; //current belongs ni position low.
            else { //Promote entry [large ] and move down the tree.
                entry [low] = entry [large]; //The larger one is promoted, low becomes the new vacancy and large is the left child.
                low = large;
                large=2*low +1;
            }
        }
        entry [low] = current;
    }

    # 建堆:

    c++
    template <class Record>
    void Sortable_Jist< Record> :: build_heap()
    /* Post:The entries of the Sortable_list have been rearranged so that it becomes a heap.
    Uses: The contiguous List implementation of Chapter 6, and insert_heap. */
    {
        int low; // All entries beyond the position low form a heap.
        for (low = count/2 - 1; low >= 0; --low ){
            Record current = entry [low];
            insert heap(current,low, count一1);
        }
    }

    # 非递归实现

    利用数组实现,常用下标 0 作为根节点,此时孩子节点和父亲节点的计算:

    left_child = parent*2+1

    right_child = parent*2+2

    child-1 = parent*2+0/1

    parent = (child-1)>>1

    # 向上调整

    插入一个节点到堆的末尾

    c++
    void AdjustUp(HPDataType* a, size_t child){
    	size_t parent;
    	while (child > 0){
            parent = (child - 1) / 2;
    		if (a[child] > a[parent]){ // 大堆
    			swap(&a[child], &a[parent]);
    			child = parent;
    		}
    		else{
    			break;
    		}
    	}
    }

    向上调整建堆

    c++
    // 从数组第 2 个数直到最后一个数重复进行向上调整算法
    for(int i=1;i<n;i++){
        AdjustUp(a,i);
    }

    向上调整法建堆时间复杂度: O(nlogn)O(nlogn)

    # 向下调整

    c++
    void AdjustDown(HPDataType* a, size_t size, int parent){
    	size_t child = parent * 2 + 1;
    	while (child < size){
    		if (child + 1 < size && a[child + 1] > a[child]){ // 比较左右孩子
    			++child;
    		}
    		if (a[child] > a[parent]){ // 大堆
    			swap(&a[parent], &a[child]);
    			parent = child;
                size_t child = parent * 2 + 1;
    		}
    		else{
    			break;
    		}
    	}
    }
    c++
    // 从最后一个叶子的父亲开始,不断 --,直到第一个节点 (包括第一个节点),重复进行向下调整算法
    // 最后一个叶子节点的数组下标是 n-1,最后一个叶子节点的父亲的数组下标为 (n-1-1)/2 = n/2-1
    for(int i=n/2-1;i>=0;i--){
        AdjustDown(a,n,i);
    }

    向下调整法建堆时间复杂度: O(n)O(n)

    参考:
    https://blog.csdn.net/zhang_si_hang/article/details/124018889

    # 性能

    • 与快排相比,堆排序的常数部分差一点,比较次数多一些,赋值次数少一些

    • 堆排序是不稳定排序

    # 堆的应用

    堆是一种常用的可以对排序、最值查找…… 优化的数据结构、技巧。

    # 优先级队列

    基于堆实现,c++ 里为 priority_queue

    # 前 k 个最大 / 小的数

    • 经典面试题

      • 题面:给定一个有 N 个 entry 的 list ,设计一个算法, 从 N 个 entry 中找出最大的 k 个。假定 N 很大且 k 很小 (例如 N = 100 亿,k = 30)。

      • 题解:用堆排序,建堆需要时间复杂度:kl˙og2kk\dot log_{2}k;调整需要时间复杂度:Nl˙og2kN\dot log_{2}k;总共为:O((N+k)log2k)O((N+k)log_{2}k)

    # 第 k 个最大 / 小的数

    • 对顶堆
      基于堆维持最大值 / 最小值的特点,用于维持第 k 大 / 第 k 小的数据
    \ No newline at end of file diff --git a/cs/data-structure-and-algorithm/minimum-spanning-tree/index.html b/cs/data-structure-and-algorithm/minimum-spanning-tree/index.html index d2ff3abe..71312d5a 100644 --- a/cs/data-structure-and-algorithm/minimum-spanning-tree/index.html +++ b/cs/data-structure-and-algorithm/minimum-spanning-tree/index.html @@ -1 +1 @@ -最小生成树 - 数据结构与算法 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 定义

    最小生成树 Minimum Spanning Tree 问题

    等价问题:寻找一个带权无向图中某一点到另一个点的最短路径 / 权最小的路径

    注意:当各边有相同权值时,由于选择的随意性,产生的生成树可能不唯一。

    # Kruskal (克鲁斯卡尔) 算法

    为主导,利用最小堆并查集

    对所有边按权值从小到大排序,逐个添加不形成回路的边,直到包含所有的顶点。

    # Prim (普里姆) 算法

    • 为主导
    • 贪心思想,不断选最小权值的边
    1. 从某一顶点出发,选择与它关联的具有最小权值的边,将其顶点加入到生成树顶点集合 U1U_1 中,剩余顶点集合为 U2U_2
    2. 以后每一步从:
      • 一个顶点在 U1U_1 中,
      • 另一个顶点在 U2U_2 中,
        的所有边中权值最小的边,将 U2U_2 中该顶点加入到 U1U_1
    3. 重复步骤 2 继续 n1n-1 次直到 U2U_2 为空
    c++
    //lowcost 存放生成树顶点集合内顶点到生成树外各顶点的各边上 的当前最小权值
    //nearvex 记录生成树顶点集合外各顶点距离集合内哪个顶点最近 (即权值最小)
    void Prim(Graph<string>& G, MinSpanTree&T) {
        int i, j, n = G.NumberOfVertices( );// 顶点数
        float * lowcost = new float[n];
        int * nearvex = new int[n];
        for(i=1;i<n;i++){
            lowcost[i] = G.GetWeight(0, i); // 顶点 0 到各边代价
            nearvex[i] = 0; // 及最短带权路径
        }
        nearvex[0] = -1;// 加到生成树顶点集合
        MSTEdgeNode e;// 最小生成树结点单元
        for ( i = 1; i < n; i++ ) {// 循环 n-1 次,加入 n-1 条边
            float min = MaxValue;
            int v = 0;
            for ( j = 0; j < n; j++ ){
                if ( nearvex[j] != -1 && lowcost[j] < min ) {
                    v=j;
                    min=lowcost[j];
                }
            }
            // 求生成树外顶点到生成树内顶点具有最小权值的边,v 是当前具最小权值的边
            if ( v ) { //v=0 表示再也找不到要求顶点
                e.tail = nearvex[v];
                e.head = v;
                e.cost = lowcost[v];
                T.Insert (e); // 选出的边加入生成树 // 若是计算 MST 总权值则是加该边的权值
                nearvex[v]=-1; // 该边加入生成树标记
                for ( j = 1; j < n; j++ ){
                    if ( nearvex[j] != -1 && G.GetWeight(v, j) < lowcost[j] ) {
                        lowcost[j] = G.GetWeight(v, j);
                        nearvex[j] = v;
                    }
                }
            }
        }
        free loecodt; free nearex;
    }

    # 两种算法比较

    分析普里姆算法,设连通网络有 n 个顶点,则该算法的时间复杂度为O(n2)O(n^2),它适用于边稠密的网络,与边的数目无关。
    克鲁斯卡尔算法主要针对边展开,边数少时效率会很高,所以对于边稀疏的图有优势。

    \ No newline at end of file +最小生成树 - 数据结构与算法 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 定义

    最小生成树 Minimum Spanning Tree 问题

    等价问题:寻找一个带权无向图中某一点到另一个点的最短路径 / 权最小的路径

    注意:当各边有相同权值时,由于选择的随意性,产生的生成树可能不唯一。

    # Kruskal (克鲁斯卡尔) 算法

    为主导,利用最小堆并查集

    对所有边按权值从小到大排序,逐个添加不形成回路的边,直到包含所有的顶点。

    # Prim (普里姆) 算法

    • 为主导
    • 贪心思想,不断选最小权值的边
    1. 从某一顶点出发,选择与它关联的具有最小权值的边,将其顶点加入到生成树顶点集合 U1U_1 中,剩余顶点集合为 U2U_2
    2. 以后每一步从:
      • 一个顶点在 U1U_1 中,
      • 另一个顶点在 U2U_2 中,
        的所有边中权值最小的边,将 U2U_2 中该顶点加入到 U1U_1
    3. 重复步骤 2 继续 n1n-1 次直到 U2U_2 为空
    c++
    //lowcost 存放生成树顶点集合内顶点到生成树外各顶点的各边上 的当前最小权值
    //nearvex 记录生成树顶点集合外各顶点距离集合内哪个顶点最近 (即权值最小)
    void Prim(Graph<string>& G, MinSpanTree&T) {
        int i, j, n = G.NumberOfVertices( );// 顶点数
        float * lowcost = new float[n];
        int * nearvex = new int[n];
        for(i=1;i<n;i++){
            lowcost[i] = G.GetWeight(0, i); // 顶点 0 到各边代价
            nearvex[i] = 0; // 及最短带权路径
        }
        nearvex[0] = -1;// 加到生成树顶点集合
        MSTEdgeNode e;// 最小生成树结点单元
        for ( i = 1; i < n; i++ ) {// 循环 n-1 次,加入 n-1 条边
            float min = MaxValue;
            int v = 0;
            for ( j = 0; j < n; j++ ){
                if ( nearvex[j] != -1 && lowcost[j] < min ) {
                    v=j;
                    min=lowcost[j];
                }
            }
            // 求生成树外顶点到生成树内顶点具有最小权值的边,v 是当前具最小权值的边
            if ( v ) { //v=0 表示再也找不到要求顶点
                e.tail = nearvex[v];
                e.head = v;
                e.cost = lowcost[v];
                T.Insert (e); // 选出的边加入生成树 // 若是计算 MST 总权值则是加该边的权值
                nearvex[v]=-1; // 该边加入生成树标记
                for ( j = 1; j < n; j++ ){
                    if ( nearvex[j] != -1 && G.GetWeight(v, j) < lowcost[j] ) {
                        lowcost[j] = G.GetWeight(v, j);
                        nearvex[j] = v;
                    }
                }
            }
        }
        free loecodt; free nearex;
    }

    # 两种算法比较

    分析普里姆算法,设连通网络有 n 个顶点,则该算法的时间复杂度为O(n2)O(n^2),它适用于边稠密的网络,与边的数目无关。
    克鲁斯卡尔算法主要针对边展开,边数少时效率会很高,所以对于边稀疏的图有优势。

    \ No newline at end of file diff --git a/cs/data-structure-and-algorithm/pre-in-post-index/index.html b/cs/data-structure-and-algorithm/pre-in-post-index/index.html index 96a43420..97753472 100644 --- a/cs/data-structure-and-algorithm/pre-in-post-index/index.html +++ b/cs/data-structure-and-algorithm/pre-in-post-index/index.html @@ -1 +1 @@ -前缀、中缀、后缀表达式 - 数据结构与算法 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 三种表达式

    • 前缀表达式 / Prefix Notation

      • 波兰式 / Polish Notation
    • 中缀表达式 / Infix Notation (中缀表达式才需要有括号)

    • 后缀表达式 / Postfix Notation

      • 逆波兰式 / Reverse Polish Notation

    # 三种表达式的相互转化

    # 括号法

    通用方法,略

    # 表达式树法

    通用方法,略

    # 栈方法

    专用于: 中缀 -> 前缀 / 后缀

    很像,注意区别,对照记忆

    # 中缀 -> 后缀

    • 扫描
    • 打印
    • 遇到操作数:直接打印
    • 遇到操作符:
      1. 优先级小于等于栈顶:栈顶出栈并打印,继续与下一个栈顶操作符比较
      2. 优先级大于栈顶:入栈
      3. 栈顶为括号 (一定是左括号):入栈
    • 遇到括号:
      1. 左括号:直接入栈
      2. 右括号:不断出栈并打印,直到出栈一个左括号为止

    # 中缀 -> 前缀

    • 扫描
    • 打印
    • 遇到操作数:直接打印
    • 遇到操作符:
      1. 优先级小于栈顶:栈顶出栈并打印,继续与下一个栈顶操作符比较
      2. 优先级大于等于栈顶:入栈
      3. 栈顶为括号 (一定是右括号):入栈
    • 遇到括号:
      1. 左括号:不断出栈并打印,直到出栈一个右括号为止
      2. 右括号:直接入栈
    \ No newline at end of file +前缀、中缀、后缀表达式 - 数据结构与算法 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 三种表达式

    • 前缀表达式 / Prefix Notation

      • 波兰式 / Polish Notation
    • 中缀表达式 / Infix Notation (中缀表达式才需要有括号)

    • 后缀表达式 / Postfix Notation

      • 逆波兰式 / Reverse Polish Notation

    # 三种表达式的相互转化

    # 括号法

    通用方法,略

    # 表达式树法

    通用方法,略

    # 栈方法

    专用于: 中缀 -> 前缀 / 后缀

    很像,注意区别,对照记忆

    # 中缀 -> 后缀

    • 扫描
    • 打印
    • 遇到操作数:直接打印
    • 遇到操作符:
      1. 优先级小于等于栈顶:栈顶出栈并打印,继续与下一个栈顶操作符比较
      2. 优先级大于栈顶:入栈
      3. 栈顶为括号 (一定是左括号):入栈
    • 遇到括号:
      1. 左括号:直接入栈
      2. 右括号:不断出栈并打印,直到出栈一个左括号为止

    # 中缀 -> 前缀

    • 扫描
    • 打印
    • 遇到操作数:直接打印
    • 遇到操作符:
      1. 优先级小于栈顶:栈顶出栈并打印,继续与下一个栈顶操作符比较
      2. 优先级大于等于栈顶:入栈
      3. 栈顶为括号 (一定是右括号):入栈
    • 遇到括号:
      1. 左括号:不断出栈并打印,直到出栈一个右括号为止
      2. 右括号:直接入栈
    \ No newline at end of file diff --git a/cs/data-structure-and-algorithm/radix-sort/index.html b/cs/data-structure-and-algorithm/radix-sort/index.html index bda8781d..b9940daa 100644 --- a/cs/data-structure-and-algorithm/radix-sort/index.html +++ b/cs/data-structure-and-algorithm/radix-sort/index.html @@ -1 +1 @@ -基排序 - 数据结构与算法 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    基排序 / 桶排序 / 数字排序

    # 原理

    需要频繁进行分组重排,(类似于归并)适用于链表

    可分为两种:

    • MSD,最高位优先 (Most Significant Digit first)

    • LSD,最低位优先 (Least Significant Digit first)

    # 代码

    c++
    //Definition for singly-linked list.
    struct ListNode {
        int val;
        ListNode *next;
        ListNode() : val(0), next(nullptr) {}
        ListNode(int x) : val(x), next(nullptr) {}
        ListNode(int x, ListNode *next) : val(x), next(next) {}
    };
    c++
    class Solution {
    public:
        ListNode* sortList(ListNode* head)
        {
            ListNode* dummyhead=new ListNode(0,head);
            ListNode* p=dummyhead;
            while(p->next!=nullptr)
            {
                p->next->val+=100000;
                p=p->next;
            }
            vector<queue<ListNode*>>vec(10); // 按 0~9 分组
            for(int i=1;i<=100000;i*=10) // 这里是 6 位数,共 6 趟
            {
                int flag=0;
                while(dummyhead->next!=nullptr)
                {
                    if(dummyhead->next->val/(i*10)!=0)
                        flag=1;
                    vec[dummyhead->next->val/i%10].push(dummyhead->next);
                    dummyhead->next=dummyhead->next->next;
                }
                ListNode* p=dummyhead;
                for(int j=0;j<10;++j) // 将每个组节点连接起来
                {
                    while(!vec[j].empty())
                    {
                        vec[j].front()->next=p->next;
                        p->next=vec[j].front();
                        p=p->next;
                        vec[j].pop();
                    }
                }
                if(flag==0)
                    break;
            }
            p=dummyhead;
            while(p->next!=nullptr)
            {
                p->next->val-=100000;
                p=p->next;
            }
            ListNode* ret=dummyhead->next;
            delete dummyhead;
            return ret;
        }
    };

    # 复杂度

    设有nn 个待排序记录,关键字位数为dd,每位有rr 种取值。则排序的趟数是dd; 在每一趟中:

    • 链表初始化的时间复杂度:O(r)O(r);
    • 分配的时间复杂度:O(n)O(n);
    • 分配后收集的时间复杂度:O(r)O(r);

    则链式基数排序的时间复杂度为: O(d(n+r))O(d\cdot (n+r))

    在排序过程中使用的辅助空间是:2r2r 个链表指针,nn 个指针域空间,
    实际应用中rr 远小于nn,可以得到时间复杂度为:O(dn)O(dn)

    则空间复杂度为:O(n+r)O(n+r)

    • 当 n 很大,d 比较小小,基数排序可以比归并、快排性能更好

    • 基排序是稳定排序。

    \ No newline at end of file +基排序 - 数据结构与算法 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    基排序 / 桶排序 / 数字排序

    # 原理

    需要频繁进行分组重排,(类似于归并)适用于链表

    可分为两种:

    • MSD,最高位优先 (Most Significant Digit first)

    • LSD,最低位优先 (Least Significant Digit first)

    # 代码

    c++
    //Definition for singly-linked list.
    struct ListNode {
        int val;
        ListNode *next;
        ListNode() : val(0), next(nullptr) {}
        ListNode(int x) : val(x), next(nullptr) {}
        ListNode(int x, ListNode *next) : val(x), next(next) {}
    };
    c++
    class Solution {
    public:
        ListNode* sortList(ListNode* head)
        {
            ListNode* dummyhead=new ListNode(0,head);
            ListNode* p=dummyhead;
            while(p->next!=nullptr)
            {
                p->next->val+=100000;
                p=p->next;
            }
            vector<queue<ListNode*>>vec(10); // 按 0~9 分组
            for(int i=1;i<=100000;i*=10) // 这里是 6 位数,共 6 趟
            {
                int flag=0;
                while(dummyhead->next!=nullptr)
                {
                    if(dummyhead->next->val/(i*10)!=0)
                        flag=1;
                    vec[dummyhead->next->val/i%10].push(dummyhead->next);
                    dummyhead->next=dummyhead->next->next;
                }
                ListNode* p=dummyhead;
                for(int j=0;j<10;++j) // 将每个组节点连接起来
                {
                    while(!vec[j].empty())
                    {
                        vec[j].front()->next=p->next;
                        p->next=vec[j].front();
                        p=p->next;
                        vec[j].pop();
                    }
                }
                if(flag==0)
                    break;
            }
            p=dummyhead;
            while(p->next!=nullptr)
            {
                p->next->val-=100000;
                p=p->next;
            }
            ListNode* ret=dummyhead->next;
            delete dummyhead;
            return ret;
        }
    };

    # 复杂度

    设有nn 个待排序记录,关键字位数为dd,每位有rr 种取值。则排序的趟数是dd; 在每一趟中:

    • 链表初始化的时间复杂度:O(r)O(r);
    • 分配的时间复杂度:O(n)O(n);
    • 分配后收集的时间复杂度:O(r)O(r);

    则链式基数排序的时间复杂度为: O(d(n+r))O(d\cdot (n+r))

    在排序过程中使用的辅助空间是:2r2r 个链表指针,nn 个指针域空间,
    实际应用中rr 远小于nn,可以得到时间复杂度为:O(dn)O(dn)

    则空间复杂度为:O(n+r)O(n+r)

    • 当 n 很大,d 比较小小,基数排序可以比归并、快排性能更好

    • 基排序是稳定排序。

    \ No newline at end of file diff --git a/cs/data-structure-and-algorithm/shortest-path/index.html b/cs/data-structure-and-algorithm/shortest-path/index.html index 996f5e5a..114d1b71 100644 --- a/cs/data-structure-and-algorithm/shortest-path/index.html +++ b/cs/data-structure-and-algorithm/shortest-path/index.html @@ -1 +1 @@ -最短路径问题 - 数据结构与算法 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 定义

    寻找一个带权有向图中某一点到另一个点的最短路径 / 权最小的路径

    当然以下算法也适用于无向图

    # Floyd (弗洛伊德) 算法

    • 可以算出所有点之间的最短距离
    c++
    //dist 初始化为邻接矩阵,若 i 没有指向 j 的边,则应有 dist [i][j]=MAX_INT (表示无穷大的数)
    void floyd(){
        for(int k = 0; k < n; k ++){   // 作为循环中间点的 k 必须放在最外一层循环 
            for(int i = 0; i < n; i ++){
                for(int j = 0; j < n; j ++){
                    dist[i][j] = min(dist[i][j], dist[i][k] + dist[k][j]); 
                }    
            }    
        }    
    }
    //dist [i][j] 得出的是 i 到 j 的最短路径
    • 时间复杂度为:O(n3)O(n^{3})

    # Warshall 算法

    和 Floyd (弗洛伊德) 算法很像,二者常常一起统称为 Warshall-Floyd 算法 ,三重循环一样,只是循环最内层执行的赋值不一样

    • 用于由邻接矩阵计算可达矩阵

      • 由关系矩阵计算传递闭包矩阵

      • 关系矩阵即为邻接矩阵,对应的传递闭包矩阵即为可达矩阵

    c++
    //dist 初始化为邻接矩阵,若存在 i 指向 j 的边,则应有 dist [i][j]=1
    void floyd(){
        for(int k = 0; k < n; k ++){   // 作为循环中间点的 k 必须放在最外一层循环 
            for(int i = 0; i < n; i ++){
                for(int j = 0; j < n; j ++){
                    dist[i][j] = dist[i][j] | (dist[i][k] && dist[k][j]); 
                }    
            }    
        }
    }
    //dist [i][j] 得出的可达矩阵

    或者:

    c++
    void floyd(){
        for(int i = 0;i < n;i++){  // 先列
    		for(int j = 0;j < n;j++){  // 后行
    			if(matrix[j][i] == 1){	//	第 j 行与第 i 行进行或操作,并最终赋值到第 j 行 
    				for(int k = 0;k < n;k++){
    					matrix[j][k] = matrix[j][k] | matrix[i][k];
    				} 
    			}		
    		}	
    	}
    }

    初始化邻接矩阵对角线元素d[i][i]d[i][i] 为 0,得到的可达矩阵若d[i][i]=1d[i][i]=1 则存在包含点 i 的环,从而可以判断图是否有环和找环

    • 时间复杂度为:O(n3)O(n^{3})

    # Dijkstra (迪杰斯特拉) 算法

    可以算出某一个点到其他所有点之间的最短距离,即单源点最短路径

    对 Prim 算法略加改动就成了 Dijkstra 算法,
    将 Prim 算法中求每个顶点VkV_k 的 lowcost 值用 dist [k] 代替即可。

    与 Prim 算法一样用了贪心思想,只有在每条边权值非负的情况下才能保证结果正确

    • dist [i] 是点 s 出发到 i 的最短路径长度,长度为 n
    • 初始需要 dist [i] 赋值为 s 出发的边指向的各个点的权
    • 剩余不相邻的点全部为 MAX_INT (表示无穷大的数)
    • shortest [i] 是标记 s 到点 i 的距离 dist [i] 是否确定为最短路径,长度为 n
    c++
    void dijkstra(int s){
        memset(shortest, false, sizeof(shortest));
        shortest[s] = true;
        for(int i = 0; i < n; i ++){
            dist[i] = graph[s][i]; //graph [s][i] 为 s 到 i 的有向边的权
            // 初始化 dist [i] 赋值为 s 出发的边指向的各个点的权
        }
        int index;
        for(int i = 1; i < n; i ++){ // 一共 n-1 轮后 s 到其余所有点到距离都是最短距离
            int mincost = MAX_INT; // 表示无穷大的数
            for(int j = 0; j < n; j ++){
                if(!shortest[j] && dist[j] < mincost){
                    mincost = dist[j];
                    index = j;
                }    
            } // 此处可以改用堆,存储 dist 中未被确定为最短距离的元素和下标,维护最小值
            shortest[index] = true; // 第 i 轮确定 i 到 index 的距离为最短距离 
            for(int j = 0; j < n; j ++){
                if(!shortest[j] && dist[j] > dist[index] + graph[index][j]){
                    dist[j] = dist[index] + graph[index][j];
                }    
            }    
        }
    }
    • 时间复杂度为:O(n2)O(n^{2})
    \ No newline at end of file +最短路径问题 - 数据结构与算法 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 定义

    寻找一个带权有向图中某一点到另一个点的最短路径 / 权最小的路径

    当然以下算法也适用于无向图

    # Floyd (弗洛伊德) 算法

    • 可以算出所有点之间的最短距离
    c++
    //dist 初始化为邻接矩阵,若 i 没有指向 j 的边,则应有 dist [i][j]=MAX_INT (表示无穷大的数)
    void floyd(){
        for(int k = 0; k < n; k ++){   // 作为循环中间点的 k 必须放在最外一层循环 
            for(int i = 0; i < n; i ++){
                for(int j = 0; j < n; j ++){
                    dist[i][j] = min(dist[i][j], dist[i][k] + dist[k][j]); 
                }    
            }    
        }    
    }
    //dist [i][j] 得出的是 i 到 j 的最短路径
    • 时间复杂度为:O(n3)O(n^{3})

    # Warshall 算法

    和 Floyd (弗洛伊德) 算法很像,二者常常一起统称为 Warshall-Floyd 算法 ,三重循环一样,只是循环最内层执行的赋值不一样

    • 用于由邻接矩阵计算可达矩阵

      • 由关系矩阵计算传递闭包矩阵

      • 关系矩阵即为邻接矩阵,对应的传递闭包矩阵即为可达矩阵

    c++
    //dist 初始化为邻接矩阵,若存在 i 指向 j 的边,则应有 dist [i][j]=1
    void floyd(){
        for(int k = 0; k < n; k ++){   // 作为循环中间点的 k 必须放在最外一层循环 
            for(int i = 0; i < n; i ++){
                for(int j = 0; j < n; j ++){
                    dist[i][j] = dist[i][j] | (dist[i][k] && dist[k][j]); 
                }    
            }    
        }
    }
    //dist [i][j] 得出的可达矩阵

    或者:

    c++
    void floyd(){
        for(int i = 0;i < n;i++){  // 先列
    		for(int j = 0;j < n;j++){  // 后行
    			if(matrix[j][i] == 1){	//	第 j 行与第 i 行进行或操作,并最终赋值到第 j 行 
    				for(int k = 0;k < n;k++){
    					matrix[j][k] = matrix[j][k] | matrix[i][k];
    				} 
    			}		
    		}	
    	}
    }

    初始化邻接矩阵对角线元素d[i][i]d[i][i] 为 0,得到的可达矩阵若d[i][i]=1d[i][i]=1 则存在包含点 i 的环,从而可以判断图是否有环和找环

    • 时间复杂度为:O(n3)O(n^{3})

    # Dijkstra (迪杰斯特拉) 算法

    可以算出某一个点到其他所有点之间的最短距离,即单源点最短路径

    对 Prim 算法略加改动就成了 Dijkstra 算法,
    将 Prim 算法中求每个顶点VkV_k 的 lowcost 值用 dist [k] 代替即可。

    与 Prim 算法一样用了贪心思想,只有在每条边权值非负的情况下才能保证结果正确

    • dist [i] 是点 s 出发到 i 的最短路径长度,长度为 n
    • 初始需要 dist [i] 赋值为 s 出发的边指向的各个点的权
    • 剩余不相邻的点全部为 MAX_INT (表示无穷大的数)
    • shortest [i] 是标记 s 到点 i 的距离 dist [i] 是否确定为最短路径,长度为 n
    c++
    void dijkstra(int s){
        memset(shortest, false, sizeof(shortest));
        shortest[s] = true;
        for(int i = 0; i < n; i ++){
            dist[i] = graph[s][i]; //graph [s][i] 为 s 到 i 的有向边的权
            // 初始化 dist [i] 赋值为 s 出发的边指向的各个点的权
        }
        int index;
        for(int i = 1; i < n; i ++){ // 一共 n-1 轮后 s 到其余所有点到距离都是最短距离
            int mincost = MAX_INT; // 表示无穷大的数
            for(int j = 0; j < n; j ++){
                if(!shortest[j] && dist[j] < mincost){
                    mincost = dist[j];
                    index = j;
                }    
            } // 此处可以改用堆,存储 dist 中未被确定为最短距离的元素和下标,维护最小值
            shortest[index] = true; // 第 i 轮确定 i 到 index 的距离为最短距离 
            for(int j = 0; j < n; j ++){
                if(!shortest[j] && dist[j] > dist[index] + graph[index][j]){
                    dist[j] = dist[index] + graph[index][j];
                }    
            }    
        }
    }
    • 时间复杂度为:O(n2)O(n^{2})
    \ No newline at end of file diff --git a/cs/data-structure-and-algorithm/stack-queue/index.html b/cs/data-structure-and-algorithm/stack-queue/index.html index 2dd75251..47e7c9bc 100644 --- a/cs/data-structure-and-algorithm/stack-queue/index.html +++ b/cs/data-structure-and-algorithm/stack-queue/index.html @@ -1 +1 @@ -栈与队列 - 数据结构与算法 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 循环队列

    • 对于一个容量为 mm 的循环队列

    • 实际存储空间为 m+1m+1

    • 初始状态 / 判空条件: rear==frontrear == front

    • 判满条件:(rear+1)%m==front(rear+1)\%m==front

    • 入队操作:

    伪代码
    新元素加入队尾;
    队尾指针 + 1;
    • 出队操作:
    伪代码
    访问队头元素;
    队头指针 + 1;
    • 任意时刻队列内包含的元素数量为: (m+rearfront)%m(m+rear-front)\%m

    # 共享栈

    • 对于一个容量为mm 的共享栈

    • 初始状态 / 判空条件: top0==1top1==mtop_0==-1\ \land\ top_1==m

    • 判满条件:top1top0==1top_1-top_0==1

    • 入栈操作:

    伪代码
    top0 先 + 1 再赋值;
    top1 先 - 1 再赋值;
    • 出栈操作:
    伪代码
    top0-1;
    top1+1;
    • 任意时刻队列内包含的元素数量为: ...
    \ No newline at end of file +栈与队列 - 数据结构与算法 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 循环队列

    • 对于一个容量为 mm 的循环队列

    • 实际存储空间为 m+1m+1

    • 初始状态 / 判空条件: rear==frontrear == front

    • 判满条件:(rear+1)%m==front(rear+1)\%m==front

    • 入队操作:

    伪代码
    新元素加入队尾;
    队尾指针 + 1;
    • 出队操作:
    伪代码
    访问队头元素;
    队头指针 + 1;
    • 任意时刻队列内包含的元素数量为: (m+rearfront)%m(m+rear-front)\%m

    # 共享栈

    • 对于一个容量为mm 的共享栈

    • 初始状态 / 判空条件: top0==1top1==mtop_0==-1\ \land\ top_1==m

    • 判满条件:top1top0==1top_1-top_0==1

    • 入栈操作:

    伪代码
    top0 先 + 1 再赋值;
    top1 先 - 1 再赋值;
    • 出栈操作:
    伪代码
    top0-1;
    top1+1;
    • 任意时刻队列内包含的元素数量为: ...
    \ No newline at end of file diff --git a/cs/devops/acme/index.html b/cs/devops/acme/index.html index 9cd13e77..26e5f7e0 100644 --- a/cs/devops/acme/index.html +++ b/cs/devops/acme/index.html @@ -1 +1 @@ -acme.sh 插件 - DevOps - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 证书相关文件格式

    • 证书 (Certificate) - .cer.crt
    • 私钥 (Private Key) - .key
    • 证书签名请求 (Certificate signing request) - .csr
    • 证书吊销列表 (Certificate Revocation List) - .crl

    以上除了 .crl 均可以使用这两种编码方式,从而有这样的后缀:

    • base64 编码 - .pem
    • 二进制编码 (少见) - .der

    为了给我们的网页应用使用 https 协议,我们需要得到有效的第三方认证证书以及对应的私钥

    CA,certificate authority。证书局,用于制作、认证证书的第三方机构

    # 安装

    安装 acme.shhttps://github.com/acmesh-official/acme.sh

    shell
    curl https://get.acme.sh | sh -s email=username@example.com

    但是需要🪜,所以国内等效的做法是:

    shell
    git clone --depth 1 https://github.com/acmesh-official/acme.sh.git
    cd acme.sh
    ./acme.sh --install -m username@example.com

    替换 username@example.com 为自己的邮箱,用来接收上游证书的邮件通知

    目前探索了三种方法,最推荐第三种方法,直接接入运营商的 API ,需要 issue 时只需要一行指令实现,非常方便。

    # 方法一

    请求注册,并将返回结果写到 acme-dns.challenges

    shell
    curl -s -X POST https://auth.acme-dns.io/register | python3 -m json.tool > acme-dns.challenges;cat acme-dns.challenges

    acme-dns注册用户

    导出用户名、密码、子域名等结果

    shell
    export ACMEDNS_USERNAME="$(cat acme-dns.challenges | awk -F"\"" '/username/{print $4}')"
    export ACMEDNS_PASSWORD="$(cat acme-dns.challenges | awk -F"\"" '/password/{print $4}')"
    export ACMEDNS_SUBDOMAIN="$(cat acme-dns.challenges | awk -F"\"" '/subdomain/{print $4}')"
    echo "FULLDOMAIN = $(cat acme-dns.challenges | awk -F"\"" '/fulldomain/{print $4}')"

    同时增加 cname 记录来完成挑战:

    cname

    申请 ssl 证书:

    shell
    ~/.acme.sh/acme.sh --issue --dns dns_acmedns -d vm.fuusen.space --dnssleep 300

    结果:

    shell
    root@proxmox:~# ~/.acme.sh/acme.sh --issue --dns dns_acmedns -d vm.fuusen.space --dnssleep 300
    [Sun Mar 10 12:19:48 PM CST 2024] Using CA: https://acme-v02.api.letsencrypt.org/directory
    [Sun Mar 10 12:19:48 PM CST 2024] Creating domain key
    [Sun Mar 10 12:19:48 PM CST 2024] The domain key is here: /root/.acme.sh/vm.fuusen.space_ecc/vm.fuusen.space.key
    [Sun Mar 10 12:19:48 PM CST 2024] Single domain='vm.fuusen.space'
    [Sun Mar 10 12:19:51 PM CST 2024] Getting webroot for domain='vm.fuusen.space'
    [Sun Mar 10 12:19:51 PM CST 2024] Adding txt value: JE6AGlpWVR5S192edUoStf3_OXAG9UR_cwnKIGGIL9U for domain:  _acme-challenge.vm.fuusen.space
    [Sun Mar 10 12:19:51 PM CST 2024] Using acme-dns
    [Sun Mar 10 12:19:52 PM CST 2024] The txt record is added: Success.
    [Sun Mar 10 12:19:52 PM CST 2024] Sleep 300 seconds for the txt records to take effect
    [Sun Mar 10 12:24:53 PM CST 2024] Verifying: vm.fuusen.space
    [Sun Mar 10 12:24:54 PM CST 2024] Pending, The CA is processing your order, please just wait. (1/30)
    [Sun Mar 10 12:24:58 PM CST 2024] Success
    [Sun Mar 10 12:24:58 PM CST 2024] Removing DNS records.
    [Sun Mar 10 12:24:58 PM CST 2024] Removing txt: JE6AGlpWVR5S192edUoStf3_OXAG9UR_cwnKIGGIL9U for domain: _acme-challenge.vm.fuusen.space
    [Sun Mar 10 12:24:58 PM CST 2024] Using acme-dns
    [Sun Mar 10 12:24:58 PM CST 2024] Removed: Success
    [Sun Mar 10 12:24:58 PM CST 2024] Verify finished, start to sign.
    [Sun Mar 10 12:24:58 PM CST 2024] Lets finalize the order.
    [Sun Mar 10 12:24:58 PM CST 2024] Le_OrderFinalize='https://acme-v02.api.letsencrypt.org/acme/finalize/1610913557/251061640687'
    [Sun Mar 10 12:24:59 PM CST 2024] Downloading cert.
    [Sun Mar 10 12:24:59 PM CST 2024] Le_LinkCert='https://acme-v02.api.letsencrypt.org/acme/cert/03567f6bc53bd6e36d65f5b8f094bfb91eaa'
    [Sun Mar 10 12:25:00 PM CST 2024] Cert success.
    -----BEGIN CERTIFICATE-----
    MIIEHzCCAwegAwIBAgISA1Z/a8U71uNtZfW48JS/uR6qMA0GCSqGSIb3DQEBCwUA
    MDIxCzAJBgNVBAYTAlVTMRYwFAYDVQQKEw1MZXQncyBFbmNyeXB0MQswCQYDVQQD
    EwJSMzAeFw0yNDAzMTAwMzI0NTlaFw0yNDA2MDgwMzI0NThaMBoxGDAWBgNVBAMT
    D3ZtLmZ1dXNlbi5zcGFjZTBZMBMGByqGSM49AgEGCCqGSM49AwEHA0IABEbT5BcQ
    x14Rc9EqGps5dTQ9Jl5ywvrWAVJmQFtmbIBAileSa+/KYuYeSmjuDAyjj99to2KM
    R9fVK6j4fYDycCOjggIQMIICDDAOBgNVHQ8BAf8EBAMCB4AwHQYDVR0lBBYwFAYI
    KwYBBQUHAwEGCCsGAQUFBwMCMAwGA1UdEwEB/wQCMAAwHQYDVR0OBBYEFBo67Zs5
    olZHLkCkGTQK/ZFVQJwvMB8GA1UdIwQYMBaAFBQusxe3WFbLrlAJQOYfr52LFMLG
    MFUGCCsGAQUFBwEBBEkwRzAhBggrBgEFBQcwAYYVaHR0cDovL3IzLm8ubGVuY3Iu
    b3JnMCIGCCsGAQUFBzAChhZodHRwOi8vcjMuaS5sZW5jci5vcmcvMBoGA1UdEQQT
    MBGCD3ZtLmZ1dXNlbi5zcGFjZTATBgNVHSAEDDAKMAgGBmeBDAECATCCAQMGCisG
    AQQB1nkCBAIEgfQEgfEA7wB1ADtTd3U+LbmAToswWwb+QDtn2E/D9Me9AA0tcm/h
    +tQXAAABjiabW4kAAAQDAEYwRAIgehF+9/VI/MC8IvkLbFTpa82IP1zXOpZFvW1U
    D77cdCYCIAy+z2hwinQtpc4up40bhNx4+h1JujS7DDgyNwsfdGaCAHYA7s3QZNXb
    Gs7FXLedtM0TojKHRny87N7DUUhZRnEftZsAAAGOJptbiQAABAMARzBFAiAy8u5B
    ZN0V23ZzMJ1dViQ4bfXWJyxs42W93GIpmJRkPAIhAOfoZmJpDW02uZV667vTZWqF
    zmGJfM3njf+XQnnARF3dMA0GCSqGSIb3DQEBCwUAA4IBAQAxiO0OOA/tVqDfoQne
    JrlX1wB+aYmTHeqwfxBE+Q0Mu0irGtX9K4XyhQR06edLKw1fXY2dH+iYKvscsE91
    iGhs0nSrAqKIi3VLLgvEFrTEW7C8Pftgt4Hbx89LvION7MgXXHe62L4JvrTkQ/Y4
    sRa6vt48bDkDspQ1BQrT589SiXMUsNOE947mbaViR3cbg2oQd+XxLJaAb1ZoSNd0
    Lvs/j5u67r30C5QnSEGxChR/6c9V9+aw07yttoD6M8iA7LA736pPjWAM8hvQ+cQa
    wAWyDU1Ub89KJx5Pv+nvbNCRT1TjoqmhPxYSxsDTYAjUTV9sBbxvcUdG7w+6tUwX
    6b9m
    -----END CERTIFICATE-----
    [Sun Mar 10 12:25:00 PM CST 2024] Your cert is in: /root/.acme.sh/vm.fuusen.space_ecc/vm.fuusen.space.cer
    [Sun Mar 10 12:25:00 PM CST 2024] Your cert key is in: /root/.acme.sh/vm.fuusen.space_ecc/vm.fuusen.space.key
    [Sun Mar 10 12:25:00 PM CST 2024] The intermediate CA cert is in: /root/.acme.sh/vm.fuusen.space_ecc/ca.cer
    [Sun Mar 10 12:25:00 PM CST 2024] And the full chain certs is there: /root/.acme.sh/vm.fuusen.space_ecc/fullchain.cer

    # 方法二

    推荐使用 Let's Encrypt 作为上游 CA :

    shell
    acme.sh --set-default-ca --server letsencrypt

    使用 Nginx ,在 /etc/nginx/conf.d/ 中新建一个配置文件 letsencrypt.conf (名字随意),内容如下:

    letsencrypt.conf
    server {
    	listen 80;
    	listen [::]:80;
    	server_name example1.com;
    	server_name example2.com;
    	location /.well-known/acme-challenge {
    		root /var/www/letsencrypt;
    	}
    	location / {
    		rewrite	^/(.*)$ https://$host/$1 permanent;
    	}
    }

    运行:

    shell
    acme.sh --issue -d md.fuusen.space -d tex.fuusen.space -w /var/www/letsencrypt --debug

    reference:

    • https://u.sb/acme-sh-ssl/

    # 方法三

    用运营商的 API ,自动添加 txt 记录完成挑战。参考:https://github.com/acmesh-official/acme.sh/wiki/dnsapi2

    首先第一次需要设置 API ,具体需要看不同的运营商。

    shell
    export ...=...

    然后 issue 之后会自动保存所设置的 API 。

    执行指令:

    shell
    acme.sh --issue --dns [dns_your_server] -d example.com

    似乎也可以一次性批量 issue 多个 cert ,但是我实际测试却只得到了同时 issue 时的第一个域名的证书,可能是还存在 bug 。

    这样方法得到的证书文件在 ~/.acme.sh/example.com_ecc/ 如下:

    shell
    ├── ca.cer
    ├── fullchain.cer
    ├── example.com.cer
    ├── example.com.conf
    ├── example.com.csr
    ├── example.com.csr.conf
    └── example.com.key

    其中:

    • fullchain.cer 是对应的证书文件
    • example.com.key 是对应的密钥文件

    Nginx 配置:

    xxx.conf
    server {
    		......
    		ssl_certificate  /home/lsy/.acme.sh/example.com_ecc/fullchain.cer;
            ssl_certificate_key  /home/lsy/.acme.sh/example.com_ecc/example.com.key;
            ......
    }

    ⚠️ 注意如果 curl 网页时出现如下报错:

    shell
    curl: (35) OpenSSL/3.0.13: error:0A00010B:SSL routines::wrong version number

    说明 http /https 的协议配置有问题,出现了本来该用 https 却用了 http ,或反之的情况。

    Edited on Views times
    \ No newline at end of file +acme.sh 插件 - DevOps - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 证书相关文件格式

    • 证书 (Certificate) - .cer.crt
    • 私钥 (Private Key) - .key
    • 证书签名请求 (Certificate signing request) - .csr
    • 证书吊销列表 (Certificate Revocation List) - .crl

    以上除了 .crl 均可以使用这两种编码方式,从而有这样的后缀:

    • base64 编码 - .pem
    • 二进制编码 (少见) - .der

    为了给我们的网页应用使用 https 协议,我们需要得到有效的第三方认证证书以及对应的私钥

    CA,certificate authority。证书局,用于制作、认证证书的第三方机构

    # 安装

    安装 acme.shhttps://github.com/acmesh-official/acme.sh

    shell
    curl https://get.acme.sh | sh -s email=username@example.com

    但是需要🪜,所以国内等效的做法是:

    shell
    git clone --depth 1 https://github.com/acmesh-official/acme.sh.git
    cd acme.sh
    ./acme.sh --install -m username@example.com

    替换 username@example.com 为自己的邮箱,用来接收上游证书的邮件通知

    目前探索了三种方法,最推荐第三种方法,直接接入运营商的 API ,需要 issue 时只需要一行指令实现,非常方便。

    # 方法一

    请求注册,并将返回结果写到 acme-dns.challenges

    shell
    curl -s -X POST https://auth.acme-dns.io/register | python3 -m json.tool > acme-dns.challenges;cat acme-dns.challenges

    acme-dns注册用户

    导出用户名、密码、子域名等结果

    shell
    export ACMEDNS_USERNAME="$(cat acme-dns.challenges | awk -F"\"" '/username/{print $4}')"
    export ACMEDNS_PASSWORD="$(cat acme-dns.challenges | awk -F"\"" '/password/{print $4}')"
    export ACMEDNS_SUBDOMAIN="$(cat acme-dns.challenges | awk -F"\"" '/subdomain/{print $4}')"
    echo "FULLDOMAIN = $(cat acme-dns.challenges | awk -F"\"" '/fulldomain/{print $4}')"

    同时增加 cname 记录来完成挑战:

    cname

    申请 ssl 证书:

    shell
    ~/.acme.sh/acme.sh --issue --dns dns_acmedns -d vm.fuusen.space --dnssleep 300

    结果:

    shell
    root@proxmox:~# ~/.acme.sh/acme.sh --issue --dns dns_acmedns -d vm.fuusen.space --dnssleep 300
    [Sun Mar 10 12:19:48 PM CST 2024] Using CA: https://acme-v02.api.letsencrypt.org/directory
    [Sun Mar 10 12:19:48 PM CST 2024] Creating domain key
    [Sun Mar 10 12:19:48 PM CST 2024] The domain key is here: /root/.acme.sh/vm.fuusen.space_ecc/vm.fuusen.space.key
    [Sun Mar 10 12:19:48 PM CST 2024] Single domain='vm.fuusen.space'
    [Sun Mar 10 12:19:51 PM CST 2024] Getting webroot for domain='vm.fuusen.space'
    [Sun Mar 10 12:19:51 PM CST 2024] Adding txt value: JE6AGlpWVR5S192edUoStf3_OXAG9UR_cwnKIGGIL9U for domain:  _acme-challenge.vm.fuusen.space
    [Sun Mar 10 12:19:51 PM CST 2024] Using acme-dns
    [Sun Mar 10 12:19:52 PM CST 2024] The txt record is added: Success.
    [Sun Mar 10 12:19:52 PM CST 2024] Sleep 300 seconds for the txt records to take effect
    [Sun Mar 10 12:24:53 PM CST 2024] Verifying: vm.fuusen.space
    [Sun Mar 10 12:24:54 PM CST 2024] Pending, The CA is processing your order, please just wait. (1/30)
    [Sun Mar 10 12:24:58 PM CST 2024] Success
    [Sun Mar 10 12:24:58 PM CST 2024] Removing DNS records.
    [Sun Mar 10 12:24:58 PM CST 2024] Removing txt: JE6AGlpWVR5S192edUoStf3_OXAG9UR_cwnKIGGIL9U for domain: _acme-challenge.vm.fuusen.space
    [Sun Mar 10 12:24:58 PM CST 2024] Using acme-dns
    [Sun Mar 10 12:24:58 PM CST 2024] Removed: Success
    [Sun Mar 10 12:24:58 PM CST 2024] Verify finished, start to sign.
    [Sun Mar 10 12:24:58 PM CST 2024] Lets finalize the order.
    [Sun Mar 10 12:24:58 PM CST 2024] Le_OrderFinalize='https://acme-v02.api.letsencrypt.org/acme/finalize/1610913557/251061640687'
    [Sun Mar 10 12:24:59 PM CST 2024] Downloading cert.
    [Sun Mar 10 12:24:59 PM CST 2024] Le_LinkCert='https://acme-v02.api.letsencrypt.org/acme/cert/03567f6bc53bd6e36d65f5b8f094bfb91eaa'
    [Sun Mar 10 12:25:00 PM CST 2024] Cert success.
    -----BEGIN CERTIFICATE-----
    MIIEHzCCAwegAwIBAgISA1Z/a8U71uNtZfW48JS/uR6qMA0GCSqGSIb3DQEBCwUA
    MDIxCzAJBgNVBAYTAlVTMRYwFAYDVQQKEw1MZXQncyBFbmNyeXB0MQswCQYDVQQD
    EwJSMzAeFw0yNDAzMTAwMzI0NTlaFw0yNDA2MDgwMzI0NThaMBoxGDAWBgNVBAMT
    D3ZtLmZ1dXNlbi5zcGFjZTBZMBMGByqGSM49AgEGCCqGSM49AwEHA0IABEbT5BcQ
    x14Rc9EqGps5dTQ9Jl5ywvrWAVJmQFtmbIBAileSa+/KYuYeSmjuDAyjj99to2KM
    R9fVK6j4fYDycCOjggIQMIICDDAOBgNVHQ8BAf8EBAMCB4AwHQYDVR0lBBYwFAYI
    KwYBBQUHAwEGCCsGAQUFBwMCMAwGA1UdEwEB/wQCMAAwHQYDVR0OBBYEFBo67Zs5
    olZHLkCkGTQK/ZFVQJwvMB8GA1UdIwQYMBaAFBQusxe3WFbLrlAJQOYfr52LFMLG
    MFUGCCsGAQUFBwEBBEkwRzAhBggrBgEFBQcwAYYVaHR0cDovL3IzLm8ubGVuY3Iu
    b3JnMCIGCCsGAQUFBzAChhZodHRwOi8vcjMuaS5sZW5jci5vcmcvMBoGA1UdEQQT
    MBGCD3ZtLmZ1dXNlbi5zcGFjZTATBgNVHSAEDDAKMAgGBmeBDAECATCCAQMGCisG
    AQQB1nkCBAIEgfQEgfEA7wB1ADtTd3U+LbmAToswWwb+QDtn2E/D9Me9AA0tcm/h
    +tQXAAABjiabW4kAAAQDAEYwRAIgehF+9/VI/MC8IvkLbFTpa82IP1zXOpZFvW1U
    D77cdCYCIAy+z2hwinQtpc4up40bhNx4+h1JujS7DDgyNwsfdGaCAHYA7s3QZNXb
    Gs7FXLedtM0TojKHRny87N7DUUhZRnEftZsAAAGOJptbiQAABAMARzBFAiAy8u5B
    ZN0V23ZzMJ1dViQ4bfXWJyxs42W93GIpmJRkPAIhAOfoZmJpDW02uZV667vTZWqF
    zmGJfM3njf+XQnnARF3dMA0GCSqGSIb3DQEBCwUAA4IBAQAxiO0OOA/tVqDfoQne
    JrlX1wB+aYmTHeqwfxBE+Q0Mu0irGtX9K4XyhQR06edLKw1fXY2dH+iYKvscsE91
    iGhs0nSrAqKIi3VLLgvEFrTEW7C8Pftgt4Hbx89LvION7MgXXHe62L4JvrTkQ/Y4
    sRa6vt48bDkDspQ1BQrT589SiXMUsNOE947mbaViR3cbg2oQd+XxLJaAb1ZoSNd0
    Lvs/j5u67r30C5QnSEGxChR/6c9V9+aw07yttoD6M8iA7LA736pPjWAM8hvQ+cQa
    wAWyDU1Ub89KJx5Pv+nvbNCRT1TjoqmhPxYSxsDTYAjUTV9sBbxvcUdG7w+6tUwX
    6b9m
    -----END CERTIFICATE-----
    [Sun Mar 10 12:25:00 PM CST 2024] Your cert is in: /root/.acme.sh/vm.fuusen.space_ecc/vm.fuusen.space.cer
    [Sun Mar 10 12:25:00 PM CST 2024] Your cert key is in: /root/.acme.sh/vm.fuusen.space_ecc/vm.fuusen.space.key
    [Sun Mar 10 12:25:00 PM CST 2024] The intermediate CA cert is in: /root/.acme.sh/vm.fuusen.space_ecc/ca.cer
    [Sun Mar 10 12:25:00 PM CST 2024] And the full chain certs is there: /root/.acme.sh/vm.fuusen.space_ecc/fullchain.cer

    # 方法二

    推荐使用 Let's Encrypt 作为上游 CA :

    shell
    acme.sh --set-default-ca --server letsencrypt

    使用 Nginx ,在 /etc/nginx/conf.d/ 中新建一个配置文件 letsencrypt.conf (名字随意),内容如下:

    letsencrypt.conf
    server {
    	listen 80;
    	listen [::]:80;
    	server_name example1.com;
    	server_name example2.com;
    	location /.well-known/acme-challenge {
    		root /var/www/letsencrypt;
    	}
    	location / {
    		rewrite	^/(.*)$ https://$host/$1 permanent;
    	}
    }

    运行:

    shell
    acme.sh --issue -d md.fuusen.space -d tex.fuusen.space -w /var/www/letsencrypt --debug

    reference:

    • https://u.sb/acme-sh-ssl/

    # 方法三

    用运营商的 API ,自动添加 txt 记录完成挑战。参考:https://github.com/acmesh-official/acme.sh/wiki/dnsapi2

    首先第一次需要设置 API ,具体需要看不同的运营商。

    shell
    export ...=...

    然后 issue 之后会自动保存所设置的 API 。

    执行指令:

    shell
    acme.sh --issue --dns [dns_your_server] -d example.com

    似乎也可以一次性批量 issue 多个 cert ,但是我实际测试却只得到了同时 issue 时的第一个域名的证书,可能是还存在 bug 。

    这样方法得到的证书文件在 ~/.acme.sh/example.com_ecc/ 如下:

    shell
    ├── ca.cer
    ├── fullchain.cer
    ├── example.com.cer
    ├── example.com.conf
    ├── example.com.csr
    ├── example.com.csr.conf
    └── example.com.key

    其中:

    • fullchain.cer 是对应的证书文件
    • example.com.key 是对应的密钥文件

    Nginx 配置:

    xxx.conf
    server {
    		......
    		ssl_certificate  /home/lsy/.acme.sh/example.com_ecc/fullchain.cer;
            ssl_certificate_key  /home/lsy/.acme.sh/example.com_ecc/example.com.key;
            ......
    }

    ⚠️ 注意如果 curl 网页时出现如下报错:

    shell
    curl: (35) OpenSSL/3.0.13: error:0A00010B:SSL routines::wrong version number

    说明 http /https 的协议配置有问题,出现了本来该用 https 却用了 http ,或反之的情况。

    Edited on Views times
    \ No newline at end of file diff --git a/cs/devops/bash/index.html b/cs/devops/bash/index.html index 0a08f0d0..4c838501 100644 --- a/cs/devops/bash/index.html +++ b/cs/devops/bash/index.html @@ -1 +1 @@ -Linux 的 shell 配置原理 - DevOps - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 环境变量文件的加载顺序

    在登录 Linux 系统并启动一个 bash shell 时,默认情况下 bash 会在若干个文件中查找环境变量的设置。这些文件统称为系统环境文件。bash 检查的环境变量文件的顺序情况取决于系统运行的 shell 方式。系统运行 shell 方式一般为 4 种:

    是否交互式是否登陆式哪些情况
    通过系统用户登录默认运行的 shell
    非登录交互式运行 shell
    执行脚本运行非交互式 shell

    shell-config

    # bash 命令启动 shell

    命令 bash 启动 shell 时,可以通过选项改变其行为

    shell
    bash [长选项] [选项] [脚本]
    选项含义
    -ishell 在交互模式下运行
    -lshell 作为登陆 shell
    -r启动受限 shell
    --选项结束标志,后面的内容当做文件名或参数,即使他们以 - 开头
    --login-l
    --noprofile阻止读取初始化文件 /etc/profile~/.bash_profile , ~/.bash_login , ~/profile
    --norc在交互式 shell,阻止读取初始化文件 ~/.bashrc 。如果 shell 以 sh 调用的话,该选项默认是打开的。
    --recfile file在交互式 shell,指定初始化文件是 file 而不是 ~/.bashrc
    --version版本信息

    # 检查当前 shell 状态

    # 查看当前是否登录式

    shell
    shopt login_shell
    # 结果为 on 则是登陆式
    login_shell    	on
    # 结果为 off 不是登陆式
    login_shell    	off

    # 查看当前是否交互式

    shell
    echo $PS1
    # 结果非空则是交互式
    shell
    echo $-
    # 结果包含 i 则是交互式

    References:

    • Reference1
    • Reference2
    Edited on Views times
    \ No newline at end of file +Linux 的 shell 配置原理 - DevOps - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 环境变量文件的加载顺序

    在登录 Linux 系统并启动一个 bash shell 时,默认情况下 bash 会在若干个文件中查找环境变量的设置。这些文件统称为系统环境文件。bash 检查的环境变量文件的顺序情况取决于系统运行的 shell 方式。系统运行 shell 方式一般为 4 种:

    是否交互式是否登陆式哪些情况
    通过系统用户登录默认运行的 shell
    非登录交互式运行 shell
    执行脚本运行非交互式 shell

    shell-config

    # bash 命令启动 shell

    命令 bash 启动 shell 时,可以通过选项改变其行为

    shell
    bash [长选项] [选项] [脚本]
    选项含义
    -ishell 在交互模式下运行
    -lshell 作为登陆 shell
    -r启动受限 shell
    --选项结束标志,后面的内容当做文件名或参数,即使他们以 - 开头
    --login-l
    --noprofile阻止读取初始化文件 /etc/profile~/.bash_profile , ~/.bash_login , ~/profile
    --norc在交互式 shell,阻止读取初始化文件 ~/.bashrc 。如果 shell 以 sh 调用的话,该选项默认是打开的。
    --recfile file在交互式 shell,指定初始化文件是 file 而不是 ~/.bashrc
    --version版本信息

    # 检查当前 shell 状态

    # 查看当前是否登录式

    shell
    shopt login_shell
    # 结果为 on 则是登陆式
    login_shell    	on
    # 结果为 off 不是登陆式
    login_shell    	off

    # 查看当前是否交互式

    shell
    echo $PS1
    # 结果非空则是交互式
    shell
    echo $-
    # 结果包含 i 则是交互式

    References:

    • Reference1
    • Reference2
    Edited on Views times
    \ No newline at end of file diff --git a/cs/devops/clone-vm/index.html b/cs/devops/clone-vm/index.html index 9bd87b7b..1b47964c 100644 --- a/cs/devops/clone-vm/index.html +++ b/cs/devops/clone-vm/index.html @@ -1 +1 @@ -VM 迁移与克隆 - DevOps - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # VMware 或 VirtualBox 导出虚拟机

    导出之前可以参考以下几个因素,避免后续麻烦:

    • 虚拟机由 VMware 或 VirtualBox 导出,磁盘镜像有可能是 vmdk 格式,如果从 KVM 管理器导出,可能是 qcow2 格式。最流行的虚拟机导出格式是 OVF 标准,但实际上由于 OVF 标准本身不完善,以及虚拟机管理器导出的众多非标准扩展信息,跨管理器使用 OVF 往往受很多限制。
    • 除了格式不兼容之外,如果虚拟机管理器之间的虚拟硬件设备差别太大,也可能导致虚拟机镜像导入失败。特别是 Windows 虚拟机,对于硬件变化特别敏感。为解决这一问题,可以在导出 Windows 虚拟机之前安装 MergeIDE.zip,并在导入后启动前将虚拟磁盘改为 IDE 类型。
    • 最后还需要考虑半虚拟化驱动因素。半虚拟化驱动能够改善虚拟硬件性能,但往往针对特定虚拟机管理器。GNU/Linux 和其他开源 Unix 类操作系统默认已经安装所有必要的半虚拟化驱动,可以在导入虚拟机后直接改用半虚拟化驱动。对于 Windows 虚拟机,还需要自行安装 Windows 版本的半虚拟化驱动软件。

    GNU/Linux 和其他开源 Unix 虚拟机通常可以直接导入。但由于以上提到的因素,不能保证所有 Windows 虚拟机均能够顺利导入成功。

    以 VMware 导出举例,点击: 文件 / File -> 导出为OVF / Export as OVF

    vmware

    # 导入虚拟机到 PVE

    执行以下命令可以创建新虚拟机,虚拟机的 CPU、内存和名称沿用 OVF 配置文件中的设置,磁盘镜像将导入存储。网络配置可以手工完成。

    shell
    qm importovf 999 myvm.ovf local-lvm

    其中 999 是虚拟机 id , local-lvm 表示磁盘镜像将导入该节点的 local-lvm 存储

    # 克隆 PVE 虚拟机

    # Full Clone

    可以从 VM 或者 VM 模版 Full Clone 一个 VM ,这样会完全复制一台虚拟机,占用和原来虚拟机一样的磁盘,克隆需要的时间久一点。

    # Linked Clone

    首先需要将一台已有的 VM 关机后转化为模版:

    convert-to-template

    此后该虚拟机无法再被启动,被锁定作为一个共享的模版。希望从该模版创建虚拟机,则右击该 VM 模版选择克隆 / Clone ,此时可以选择 Linked Clone,这样得到的虚拟机会和该模版虚拟机共享磁盘,创建速度更快,而且占用磁盘更少。

    Edited on Views times
    \ No newline at end of file +VM 迁移与克隆 - DevOps - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # VMware 或 VirtualBox 导出虚拟机

    导出之前可以参考以下几个因素,避免后续麻烦:

    • 虚拟机由 VMware 或 VirtualBox 导出,磁盘镜像有可能是 vmdk 格式,如果从 KVM 管理器导出,可能是 qcow2 格式。最流行的虚拟机导出格式是 OVF 标准,但实际上由于 OVF 标准本身不完善,以及虚拟机管理器导出的众多非标准扩展信息,跨管理器使用 OVF 往往受很多限制。
    • 除了格式不兼容之外,如果虚拟机管理器之间的虚拟硬件设备差别太大,也可能导致虚拟机镜像导入失败。特别是 Windows 虚拟机,对于硬件变化特别敏感。为解决这一问题,可以在导出 Windows 虚拟机之前安装 MergeIDE.zip,并在导入后启动前将虚拟磁盘改为 IDE 类型。
    • 最后还需要考虑半虚拟化驱动因素。半虚拟化驱动能够改善虚拟硬件性能,但往往针对特定虚拟机管理器。GNU/Linux 和其他开源 Unix 类操作系统默认已经安装所有必要的半虚拟化驱动,可以在导入虚拟机后直接改用半虚拟化驱动。对于 Windows 虚拟机,还需要自行安装 Windows 版本的半虚拟化驱动软件。

    GNU/Linux 和其他开源 Unix 虚拟机通常可以直接导入。但由于以上提到的因素,不能保证所有 Windows 虚拟机均能够顺利导入成功。

    以 VMware 导出举例,点击: 文件 / File -> 导出为OVF / Export as OVF

    vmware

    # 导入虚拟机到 PVE

    执行以下命令可以创建新虚拟机,虚拟机的 CPU、内存和名称沿用 OVF 配置文件中的设置,磁盘镜像将导入存储。网络配置可以手工完成。

    shell
    qm importovf 999 myvm.ovf local-lvm

    其中 999 是虚拟机 id , local-lvm 表示磁盘镜像将导入该节点的 local-lvm 存储

    # 克隆 PVE 虚拟机

    # Full Clone

    可以从 VM 或者 VM 模版 Full Clone 一个 VM ,这样会完全复制一台虚拟机,占用和原来虚拟机一样的磁盘,克隆需要的时间久一点。

    # Linked Clone

    首先需要将一台已有的 VM 关机后转化为模版:

    convert-to-template

    此后该虚拟机无法再被启动,被锁定作为一个共享的模版。希望从该模版创建虚拟机,则右击该 VM 模版选择克隆 / Clone ,此时可以选择 Linked Clone,这样得到的虚拟机会和该模版虚拟机共享磁盘,创建速度更快,而且占用磁盘更少。

    Edited on Views times
    \ No newline at end of file diff --git a/cs/devops/haproxy-ssh/index.html b/cs/devops/haproxy-ssh/index.html index a064a689..31dfe556 100644 --- a/cs/devops/haproxy-ssh/index.html +++ b/cs/devops/haproxy-ssh/index.html @@ -1 +1 @@ -HAProxy 代理 SSH - DevOps - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 宿主与虚拟机之间的网络问题

    我们的两台宿主居然都 ping 不到虚拟机,导致后续 haproxy 代理转发后出现所有虚拟机 server 都是 down 状态

    原因应该是宿主只有一个校园网网段的 ip,但是虚拟机只有实验室局域网网段的 ip

    这个问题需要配置宿主节点的网络,可以在 PVE 控制面板修改 网络 / Network:

    network

    实际上就是编辑该节点的: /etc/network/interfaces 配置文件

    应该需要配置虚拟网卡才能解决,但是我们的一台服务器有两张网卡(图中 eno1 eno2) 所以我们最后的解决方案给服务器插了多一条网线,配置该网卡 ip 为实验室局域网的 ip,从而该服务器连接上了虚拟机

    # 安装 HAProxy

    shell
    apt isntall haproxy

    或者源码编译安装

    可快速上手学习、搜索问题的官网资料

    # 配置 haproxy.cfg

    编辑 /etc/haproxy/haproxy.cfg 配置文件,所需要的基本指令如下:

    检查并重载配置文件

    shell
    haproxy -f /etc/haproxy/haproxy.cfg

    仅检查:

    shell
    haproxy -f /etc/haproxy/haproxy.cfg -c

    ⚠️ 注意,文件最后需要有换行符结尾,否则会有报错:

    shell
    Missing LF on last line, file might have been truncated at position xx

    重启 haproxy 服务

    shell
    service haproxy restart
    # or
    systemctl restart haproxy

    配置由 5 部分组成,当然并不都是必须的: globaldefaultsfrontendbackendlisten (可能还有更多)

    我们的 SSH 代理转发需求只需配置 frontendbackend 即可,我们的配置如下:

    w raw
    frontend fe_ssh
       bind *:2222 ssl crt /etc/pve/nodes/proxmox2/pveproxy-ssl.pem
       mode tcp
       log-format "%ci:%cp [%t] %ft %b/%s %Tw/%Tc/%Tt %B %ts %ac/%fc/%bc/%sc/%rc %sq/%bq dst:%[var(sess.dst)] "
       tcp-request content set-var(sess.dst) ssl_fc_sni
       use_backend %[ssl_fc_sni]
    backend server1
       mode tcp
       server s1 192.168.134.247:22 check
    backend server2
       mode tcp
       server s2 192.168.134.30:22 check
    backend server3
       mode tcp
       server s2 172.18.198.204:22 check
    • 其中 frontend 后面的 fe_sshbackend 后面的 server1 等都是自定义的名字;
    • 端口 bind *:2222 所指定的端口也是自定义的;跟在其后面的 ssl crt /etc/pve/nodes/proxmox2/pveproxy-ssl.pem 是指定 ssl 证书,该路径是 PVE 的通过上传的或用其提供插件自动申请的 ssl 证书所在路径。
      • haproxy 似乎只认定一个 xxx.pem 证书文件的对应 key 文件是 xxx.pem.key ,所以 key 文件需要以这个形式命名放在 pem 相同路径上

    # SSH 连接

    shell
    ssh -o ProxyCommand="openssl s_client -quiet -connect [服务器ip]:2222 -servername [servername]" -l [登陆虚拟机用的用户名] [虚拟机主机名]

    其中:

    • [服务器 ip] 即一般使用 ssh 登陆时的 @ 后面的 ip,或者主机名

    • [登陆虚拟机用的用户名] 即一般使用 ssh 登陆时的 @ 前面的用户名

    • [servername] 是 backend 指定的对应服务器名字,例如我们配置文件中的 server1 、 server2 、 server3

    • [虚拟机主机名] 可以自定义,但是最后和虚拟机主机名一致,否则以后连接会出现这样的报错:

      shell
      @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
      @    WARNING: REMOTE HOST IDENTIFICATION HAS CHANGED!     @
      @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
      IT IS POSSIBLE THAT SOMEONE IS DOING SOMETHING NASTY!
      Someone could be eavesdropping on you right now (man-in-the-middle attack)!
      It is also possible that a host key has just been changed.
      The fingerprint for the ED25519 key sent by the remote host is
      SHA256:fDBSRwFB552ZiUUmekpCWrbuSdh186h4c12JdrAQxTM.
      Please contact your system administrator.
      Add correct host key in /Users/xxx/.ssh/known_hosts to get rid of this message.
      Offending ED25519 key in /Users/minamoto/.ssh/known_hosts:xx
      Host key for xxx has changed and you have requested strict checking.
      Host key verification failed.

      即连接不同的虚拟机总是需要指定不同的主机名才不会产生冲突

    # Dashboard

    raw
    frontend stats
        mode http
        bind *:9000
        stats enable
        stats uri /stats
        stats refresh 10s
        # stats uri /stats # choose a subpath for the dashboard
        # stats admin if localhost   # enable to restrict to localhost
        # stats admin if 10.0.0.0/8  # enable to restrict to 10.x.x.x network sources
        # stats admin auth <username:password> # enable to setup basic http auth

    目前发现每次修改配置文件后 dashboard 没有同步更新,除非修改 dashboard 配置的端口号,在新端口号才有同步更新了的数据

    配置好后可以打开 http://node_ip:9000/stats

    若想要 https 访问,则需要在 bind *:9000 后面加上 ssl 证书文件路径,否则只能 http 访问

    raw
    bind *:9000 ssl crt /etc/pve/nodes/proxmox2/pveproxy-ssl.pem

    最终效果(图中用的是 9002 端口):

    dashboard

    • reference1
    • reference2
    Edited on Views times
    \ No newline at end of file +HAProxy 代理 SSH - DevOps - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 宿主与虚拟机之间的网络问题

    我们的两台宿主居然都 ping 不到虚拟机,导致后续 haproxy 代理转发后出现所有虚拟机 server 都是 down 状态

    原因应该是宿主只有一个校园网网段的 ip,但是虚拟机只有实验室局域网网段的 ip

    这个问题需要配置宿主节点的网络,可以在 PVE 控制面板修改 网络 / Network:

    network

    实际上就是编辑该节点的: /etc/network/interfaces 配置文件

    应该需要配置虚拟网卡才能解决,但是我们的一台服务器有两张网卡(图中 eno1 eno2) 所以我们最后的解决方案给服务器插了多一条网线,配置该网卡 ip 为实验室局域网的 ip,从而该服务器连接上了虚拟机

    # 安装 HAProxy

    shell
    apt isntall haproxy

    或者源码编译安装

    可快速上手学习、搜索问题的官网资料

    # 配置 haproxy.cfg

    编辑 /etc/haproxy/haproxy.cfg 配置文件,所需要的基本指令如下:

    检查并重载配置文件

    shell
    haproxy -f /etc/haproxy/haproxy.cfg

    仅检查:

    shell
    haproxy -f /etc/haproxy/haproxy.cfg -c

    ⚠️ 注意,文件最后需要有换行符结尾,否则会有报错:

    shell
    Missing LF on last line, file might have been truncated at position xx

    重启 haproxy 服务

    shell
    service haproxy restart
    # or
    systemctl restart haproxy

    配置由 5 部分组成,当然并不都是必须的: globaldefaultsfrontendbackendlisten (可能还有更多)

    我们的 SSH 代理转发需求只需配置 frontendbackend 即可,我们的配置如下:

    w raw
    frontend fe_ssh
       bind *:2222 ssl crt /etc/pve/nodes/proxmox2/pveproxy-ssl.pem
       mode tcp
       log-format "%ci:%cp [%t] %ft %b/%s %Tw/%Tc/%Tt %B %ts %ac/%fc/%bc/%sc/%rc %sq/%bq dst:%[var(sess.dst)] "
       tcp-request content set-var(sess.dst) ssl_fc_sni
       use_backend %[ssl_fc_sni]
    backend server1
       mode tcp
       server s1 192.168.134.247:22 check
    backend server2
       mode tcp
       server s2 192.168.134.30:22 check
    backend server3
       mode tcp
       server s2 172.18.198.204:22 check
    • 其中 frontend 后面的 fe_sshbackend 后面的 server1 等都是自定义的名字;
    • 端口 bind *:2222 所指定的端口也是自定义的;跟在其后面的 ssl crt /etc/pve/nodes/proxmox2/pveproxy-ssl.pem 是指定 ssl 证书,该路径是 PVE 的通过上传的或用其提供插件自动申请的 ssl 证书所在路径。
      • haproxy 似乎只认定一个 xxx.pem 证书文件的对应 key 文件是 xxx.pem.key ,所以 key 文件需要以这个形式命名放在 pem 相同路径上

    # SSH 连接

    shell
    ssh -o ProxyCommand="openssl s_client -quiet -connect [服务器ip]:2222 -servername [servername]" -l [登陆虚拟机用的用户名] [虚拟机主机名]

    其中:

    • [服务器 ip] 即一般使用 ssh 登陆时的 @ 后面的 ip,或者主机名

    • [登陆虚拟机用的用户名] 即一般使用 ssh 登陆时的 @ 前面的用户名

    • [servername] 是 backend 指定的对应服务器名字,例如我们配置文件中的 server1 、 server2 、 server3

    • [虚拟机主机名] 可以自定义,但是最后和虚拟机主机名一致,否则以后连接会出现这样的报错:

      shell
      @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
      @    WARNING: REMOTE HOST IDENTIFICATION HAS CHANGED!     @
      @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
      IT IS POSSIBLE THAT SOMEONE IS DOING SOMETHING NASTY!
      Someone could be eavesdropping on you right now (man-in-the-middle attack)!
      It is also possible that a host key has just been changed.
      The fingerprint for the ED25519 key sent by the remote host is
      SHA256:fDBSRwFB552ZiUUmekpCWrbuSdh186h4c12JdrAQxTM.
      Please contact your system administrator.
      Add correct host key in /Users/xxx/.ssh/known_hosts to get rid of this message.
      Offending ED25519 key in /Users/minamoto/.ssh/known_hosts:xx
      Host key for xxx has changed and you have requested strict checking.
      Host key verification failed.

      即连接不同的虚拟机总是需要指定不同的主机名才不会产生冲突

    # Dashboard

    raw
    frontend stats
        mode http
        bind *:9000
        stats enable
        stats uri /stats
        stats refresh 10s
        # stats uri /stats # choose a subpath for the dashboard
        # stats admin if localhost   # enable to restrict to localhost
        # stats admin if 10.0.0.0/8  # enable to restrict to 10.x.x.x network sources
        # stats admin auth <username:password> # enable to setup basic http auth

    目前发现每次修改配置文件后 dashboard 没有同步更新,除非修改 dashboard 配置的端口号,在新端口号才有同步更新了的数据

    配置好后可以打开 http://node_ip:9000/stats

    若想要 https 访问,则需要在 bind *:9000 后面加上 ssl 证书文件路径,否则只能 http 访问

    raw
    bind *:9000 ssl crt /etc/pve/nodes/proxmox2/pveproxy-ssl.pem

    最终效果(图中用的是 9002 端口):

    dashboard

    • reference1
    • reference2
    Edited on Views times
    \ No newline at end of file diff --git a/cs/devops/nat-traversal/index.html b/cs/devops/nat-traversal/index.html index edde8350..b0ff1dd1 100644 --- a/cs/devops/nat-traversal/index.html +++ b/cs/devops/nat-traversal/index.html @@ -1 +1 @@ -内网穿透 - DevOps - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # frp

    # Tailscale

    # Ngrok

    # 绑定自己的域名(要充钱)

    Ngrok uses the Host header sent by the browser to determine how to redirect your traffic you will need to "reserve" the domain you want to use in the Ngrok dashboard. Note that doing this requires a paid Ngrok plan. After reserving your domain you will be given a CNAME record to point it to.

    生成 reserverd 域名,获取 cname-target ,然后将域名 DNS 解析添加 CNAME 记录目标到 cname-target

    # Tunnelmole

    支持类似 frp 自己搭建内网穿透,同时官方也提供类似 Ngrok 的服务

    \ No newline at end of file +内网穿透 - DevOps - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # frp

    # Tailscale

    # Ngrok

    # 绑定自己的域名(要充钱)

    Ngrok uses the Host header sent by the browser to determine how to redirect your traffic you will need to "reserve" the domain you want to use in the Ngrok dashboard. Note that doing this requires a paid Ngrok plan. After reserving your domain you will be given a CNAME record to point it to.

    生成 reserverd 域名,获取 cname-target ,然后将域名 DNS 解析添加 CNAME 记录目标到 cname-target

    # Tunnelmole

    支持类似 frp 自己搭建内网穿透,同时官方也提供类似 Ngrok 的服务

    \ No newline at end of file diff --git a/cs/devops/nginx/index.html b/cs/devops/nginx/index.html index 41b49234..ced5697a 100644 --- a/cs/devops/nginx/index.html +++ b/cs/devops/nginx/index.html @@ -1 +1 @@ -Nginx - DevOps - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    查看 niginx 进程:

    ps -ef|grep nginx

    进程:

    • master
    • worker

    查看 nginx 的:版本、安装目录 (--prefix)、编译参数

    nginx -V

    重新加载配置文件:

    shell
    nginx -s reload

    查看配置文件 nginx.config 所在目录,验证默认配置文件,常用于检查配置文件有没有写对:

    nginx -t

    常见目录有:

    /etc/nginx/conf
    /usr/local/etc/nginx
    /opt/homebrew/etc/nginx

    停止 nginx 服务:

    nginx -s quit		 	//优雅停止
    nginx -s stop			//立即停止
    nginx -s reload		//重载配置文件
    nginx -s reopen		//重新打开日志文件

    配置文件 nginx.config

    • 全局块
    • events 块
    • http 块
      • include mime.types
      • server 块 // 代表一个虚拟主机,可以放在 server 目录下的 local.config
        • location 块
      • include servers/*

    使用 openssl 生成证书

    • 生成私钥文件(private key)

      openssl genrsa -out private.key 2048
    • 根据私钥生成证书签名请求文件(Certificate Signing Request, 简称 CSR 文件)

      openssl req-new -key private.key -out cert.cr
    • 使用私钥对证书申请进行签名从而生成证书文件(pem 文件)

      openssl x509 -req-in cert.cr -out cacert.pem -signkey private.key
    Edited on Views times
    \ No newline at end of file +Nginx - DevOps - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    查看 niginx 进程:

    ps -ef|grep nginx

    进程:

    • master
    • worker

    查看 nginx 的:版本、安装目录 (--prefix)、编译参数

    nginx -V

    重新加载配置文件:

    shell
    nginx -s reload

    查看配置文件 nginx.config 所在目录,验证默认配置文件,常用于检查配置文件有没有写对:

    nginx -t

    常见目录有:

    /etc/nginx/conf
    /usr/local/etc/nginx
    /opt/homebrew/etc/nginx

    停止 nginx 服务:

    nginx -s quit		 	//优雅停止
    nginx -s stop			//立即停止
    nginx -s reload		//重载配置文件
    nginx -s reopen		//重新打开日志文件

    配置文件 nginx.config

    • 全局块
    • events 块
    • http 块
      • include mime.types
      • server 块 // 代表一个虚拟主机,可以放在 server 目录下的 local.config
        • location 块
      • include servers/*

    使用 openssl 生成证书

    • 生成私钥文件(private key)

      openssl genrsa -out private.key 2048
    • 根据私钥生成证书签名请求文件(Certificate Signing Request, 简称 CSR 文件)

      openssl req-new -key private.key -out cert.cr
    • 使用私钥对证书申请进行签名从而生成证书文件(pem 文件)

      openssl x509 -req-in cert.cr -out cacert.pem -signkey private.key
    Edited on Views times
    \ No newline at end of file diff --git a/cs/devops/overleaf/index.html b/cs/devops/overleaf/index.html index 6af63b90..708f9fa6 100644 --- a/cs/devops/overleaf/index.html +++ b/cs/devops/overleaf/index.html @@ -1 +1 @@ -OverLeaf - DevOps - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    部署了一个 overleaf 玩,目的需求是做一个共享文档,作为腾讯文档、知识库的替代品

    注册管理员账号的时候,出现了报错:

    alert
    Session error. Please check you have cookies enabled. If the problem persists, try clearing your cache and cookies.

    查阅这个 issue 后,得到了解决方案,将 config/variables.env 这里注释掉:

    variables.env
    # SHARELATEX_BEHIND_PROXY=true
    # SHARELATEX_SECURE_COOKIE=true

    无论这里是 true 还是 false 都需要注释掉

    每次 bin/up 都会这样的结果:

    shell
    lsy@lsy:~/data/overleaf-toolkit$ bin/up
    Initiating Mongo replica set...
    WARN[0000] /home/lsy/data/overleaf-toolkit/lib/docker-compose.base.yml: `version` is obsolete
    WARN[0000] /home/lsy/data/overleaf-toolkit/lib/docker-compose.vars.yml: `version` is obsolete
    WARN[0000] /home/lsy/data/overleaf-toolkit/lib/docker-compose.redis.yml: `version` is obsolete
    WARN[0000] /home/lsy/data/overleaf-toolkit/lib/docker-compose.mongo.yml: `version` is obsolete
    WARN[0000] /home/lsy/data/overleaf-toolkit/lib/docker-compose.nginx.yml: `version` is obsolete
    [+] Running 1/0
     ✔ Container mongo  Running                                                                                                               0.0s
    WARN[0000] /home/lsy/data/overleaf-toolkit/lib/docker-compose.base.yml: `version` is obsolete
    WARN[0000] /home/lsy/data/overleaf-toolkit/lib/docker-compose.vars.yml: `version` is obsolete
    WARN[0000] /home/lsy/data/overleaf-toolkit/lib/docker-compose.redis.yml: `version` is obsolete
    WARN[0000] /home/lsy/data/overleaf-toolkit/lib/docker-compose.mongo.yml: `version` is obsolete
    WARN[0000] /home/lsy/data/overleaf-toolkit/lib/docker-compose.nginx.yml: `version` is obsolete
    [+] Running 4/4
     ✔ Container mongo       Running                                                                                                          0.0s
     ✔ Container redis       Running                                                                                                          0.0s
     ✔ Container sharelatex  Recreated                                                                                                       60.5s
     ✔ Container nginx       Recreated                                                                                                        0.1s
    Attaching to mongo, nginx, redis, sharelatex
    sharelatex  | *** Running /etc/my_init.d/000_check_for_old_bind_mounts_5.sh...
    sharelatex  | *** Running /etc/my_init.d/000_check_for_old_env_vars_5.sh...
    sharelatex  | *** Running /etc/my_init.d/00_regen_ssh_host_keys.sh...
    sharelatex  | *** Running /etc/my_init.d/100_generate_secrets.sh...
    sharelatex  | generating random secrets
    sharelatex  | *** Running /etc/my_init.d/100_make_overleaf_data_dirs.sh...
    ......

    显然这个项目使用的是一个需要 nohup 的脚本。

    使用这样的脚本模版方便管理:

    shell
    .
    ├── running.log
    ├── running.pid
    ├── start.sh
    └── stop.sh
    start.sh :
    run=bin/up # 启动脚本的指令
    DIR="$( cd "$( dirname "$0"  )" && pwd  )"
    export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:$DIR/lib
    nohup $DIR/$run > $DIR/running.log 2>&1& echo $! > $DIR/running.pid
    stop.sh :
    kill `cat running.pid`

    使用 logrotate 定期清理日志。在 /etc/logrotate.conf 配置文件中增加:

    /etc/logrotate.conf
    /home/lsy/data/overleaf-toolkit/running.log {
            size 1k
            create 700 lsy lsy
            rotate 4
    }

    当然这里 /home/lsy/data/overleaf-toolkit/ 换成自己对应的 running.log 存放的目录

    重新加载 logrotate 的配置文件:

    shell
    sudo logrotate /etc/logrotate.conf

    后续出现的问题

    没办法使用 markdown 插件,不符合需求,转而寻找支持 typst 、 markdown 的更紧跟时代潮流的方案

    Edited on Views times
    \ No newline at end of file +OverLeaf - DevOps - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    部署了一个 overleaf 玩,目的需求是做一个共享文档,作为腾讯文档、知识库的替代品

    注册管理员账号的时候,出现了报错:

    alert
    Session error. Please check you have cookies enabled. If the problem persists, try clearing your cache and cookies.

    查阅这个 issue 后,得到了解决方案,将 config/variables.env 这里注释掉:

    variables.env
    # SHARELATEX_BEHIND_PROXY=true
    # SHARELATEX_SECURE_COOKIE=true

    无论这里是 true 还是 false 都需要注释掉

    每次 bin/up 都会这样的结果:

    shell
    lsy@lsy:~/data/overleaf-toolkit$ bin/up
    Initiating Mongo replica set...
    WARN[0000] /home/lsy/data/overleaf-toolkit/lib/docker-compose.base.yml: `version` is obsolete
    WARN[0000] /home/lsy/data/overleaf-toolkit/lib/docker-compose.vars.yml: `version` is obsolete
    WARN[0000] /home/lsy/data/overleaf-toolkit/lib/docker-compose.redis.yml: `version` is obsolete
    WARN[0000] /home/lsy/data/overleaf-toolkit/lib/docker-compose.mongo.yml: `version` is obsolete
    WARN[0000] /home/lsy/data/overleaf-toolkit/lib/docker-compose.nginx.yml: `version` is obsolete
    [+] Running 1/0
     ✔ Container mongo  Running                                                                                                               0.0s
    WARN[0000] /home/lsy/data/overleaf-toolkit/lib/docker-compose.base.yml: `version` is obsolete
    WARN[0000] /home/lsy/data/overleaf-toolkit/lib/docker-compose.vars.yml: `version` is obsolete
    WARN[0000] /home/lsy/data/overleaf-toolkit/lib/docker-compose.redis.yml: `version` is obsolete
    WARN[0000] /home/lsy/data/overleaf-toolkit/lib/docker-compose.mongo.yml: `version` is obsolete
    WARN[0000] /home/lsy/data/overleaf-toolkit/lib/docker-compose.nginx.yml: `version` is obsolete
    [+] Running 4/4
     ✔ Container mongo       Running                                                                                                          0.0s
     ✔ Container redis       Running                                                                                                          0.0s
     ✔ Container sharelatex  Recreated                                                                                                       60.5s
     ✔ Container nginx       Recreated                                                                                                        0.1s
    Attaching to mongo, nginx, redis, sharelatex
    sharelatex  | *** Running /etc/my_init.d/000_check_for_old_bind_mounts_5.sh...
    sharelatex  | *** Running /etc/my_init.d/000_check_for_old_env_vars_5.sh...
    sharelatex  | *** Running /etc/my_init.d/00_regen_ssh_host_keys.sh...
    sharelatex  | *** Running /etc/my_init.d/100_generate_secrets.sh...
    sharelatex  | generating random secrets
    sharelatex  | *** Running /etc/my_init.d/100_make_overleaf_data_dirs.sh...
    ......

    显然这个项目使用的是一个需要 nohup 的脚本。

    使用这样的脚本模版方便管理:

    shell
    .
    ├── running.log
    ├── running.pid
    ├── start.sh
    └── stop.sh
    start.sh :
    run=bin/up # 启动脚本的指令
    DIR="$( cd "$( dirname "$0"  )" && pwd  )"
    export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:$DIR/lib
    nohup $DIR/$run > $DIR/running.log 2>&1& echo $! > $DIR/running.pid
    stop.sh :
    kill `cat running.pid`

    使用 logrotate 定期清理日志。在 /etc/logrotate.conf 配置文件中增加:

    /etc/logrotate.conf
    /home/lsy/data/overleaf-toolkit/running.log {
            size 1k
            create 700 lsy lsy
            rotate 4
    }

    当然这里 /home/lsy/data/overleaf-toolkit/ 换成自己对应的 running.log 存放的目录

    重新加载 logrotate 的配置文件:

    shell
    sudo logrotate /etc/logrotate.conf

    后续出现的问题

    没办法使用 markdown 插件,不符合需求,转而寻找支持 typst 、 markdown 的更紧跟时代潮流的方案

    Edited on Views times
    \ No newline at end of file diff --git a/cs/devops/pve-passthrough-gpu/index.html b/cs/devops/pve-passthrough-gpu/index.html index 0f4b9381..f3011c3d 100644 --- a/cs/devops/pve-passthrough-gpu/index.html +++ b/cs/devops/pve-passthrough-gpu/index.html @@ -1 +1 @@ -PVE 直通显卡 - DevOps - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    PVE 版本:8.1.4 ,显卡:NVIDIA GTX 745

    # 宿主机配置

    # IOMMU

    shell
    nano /etc/default/grub

    GRUB_CMDLINE_LINUX_DEFAULT="..." 中的内容修改为如下:

    shell
    GRUB_CMDLINE_LINUX_DEFAULT="quiet intel_iommu=on iommu=pt video=efifb:off"

    准确来说是在 "..." 原来的内容后面增加这些内容,其中:

    • quiet 一般是原来就有的
    • intel_iommu=oniommu=pt 是必须的
    • video=efifb:off 如果你是给 Windows VM 直通显卡串流玩游戏的话是需要的

    然后更新配置文件:

    shell
    update-grub

    检查 IOMMU 是否启动:

    shell
    dmesg | grep -e DMAR -e IOMMU

    看到有 DMAR 或 IOMMU 输出就行,例如我们就只看到了 DMAR 的输出。

    # 配置显卡

    找到自己的显卡的相关信息:

    shell
    lspci -knn | grep VGA

    我们需要如下内容:

    shell
    01:00.0 VGA compatible controller [0300]: NVIDIA Corporation GM107 [GeForce GTX 745] [10de:1382] (rev a2)
            Subsystem: NVIDIA Corporation GM107 [GeForce GTX 745] [10de:1065] # these ids
            Kernel modules: nvidiafb, nouveau # these modules
    • [10de:1065]<vendor id>:<device id> 形式的厂商 id 和设备 id
    • nvidiafbnouveau 是我们的显卡所使用的驱动模块

    将厂商 id 和设备 id 组合添加到 vfio 配置文件中:

    shell
    nano /etc/modprobe.d/vfio.conf

    增加如下内容:

    shell
    options vfio-pci ids=10de:1382 disable_vga=1

    如果你有多张显卡需要直通,那么不同显卡对应的 <vendor id>:<device id> 就以逗号分隔加在 ids= 后面即可,例如:

    shell
    options vfio-pci ids=1145:1400,1919:1800 disable_vga=1

    屏蔽显卡驱动模块:

    shell
    echo "blacklist <some-module>" >> /etc/modprobe.d/blacklist.conf

    对每一个模块都需要加单独一行,例如我们是 nvidiafbnouveau ,则需要:

    shell
    echo "blacklist nvidiafb" >> /etc/modprobe.d/blacklist.conf
    echo "blacklist nouveau" >> /etc/modprobe.d/blacklist.conf

    有一些教程说是修改配置文件 /etc/modprobe.d/pve-blacklist.conf ,但是官方文档(目前)说的是 /etc/modprobe.d/blacklist.conf ,两者不知道有没有区别

    # vfio

    shell
    nano /etc/modules

    增加如下几行:

    shell
    vfio
    vfio_iommu_type1
    vfio_pci
    vfio_virqfd #not needed if on kernel 6.2 or newer

    最后执行如下指令,并且重启:

    shell
    update-initramfs -u -k all

    最后使用如下指令检查配置是否成功:

    shell
    lspci -k

    预期是得到对应的显卡部分有如下输出结果:

    shell
    01:00.0 VGA compatible controller: NVIDIA Corporation GP108 [GeForce GT 1030] (rev a1)
            Subsystem: Micro-Star International Co., Ltd. [MSI] GP108 [GeForce GT 1030]
            Kernel driver in use: <some-module> # 预期是 vfio
            Kernel modules: <some-module>

    # 问题

    执行最后一句指令时,我们却遇到了这个输出:

    shell
    ...
    Re-executing '/etc/kernel/postinst.d/zz-proxmox-boot' in new private mount namespace..
    No /etc/kernel/proxmox-boot-uuids found, skipping ESP sync.

    然后 lspci -k 检查也没有预期的输出,虚拟机添加显卡设备后无法开机。

    我尝试了官方文档中 Host Device Passthrough 部分的方法:

    For both methods you need to update the initramfs again and reboot after that.

    Should this not work, you might need to set a soft dependency to load the gpu modules before loading vfio-pci. This can be done with the softdep flag, see also the manpages on modprobe.d for more information.

    For example, if you are using drivers named <some-module> :

    shell
    # echo "softdep <some-module> pre: vfio-pci" >> /etc/modprobe.d/<some-module>.conf

    但仍然没有效果。

    所幸,似乎有不少人也遇到了这个问题,我找到了这个帖子,似乎说是因为没有同步连接 ESP 导致的,接下来我按照里面一个解决方案走:

    然后我先尝试了重新挂载 /boot/efi ,首先 lsblk -o +FSTYPE 查看 /boot/efi 挂载的磁盘分区,例如我这里是 sda2 ,然后重新挂载一下:

    shell
    umount /boot/efi
    proxmox-boot-tool init /dev/sda2

    然后先前类似 No /etc/kernel/proxmox-boot-uuids found, skipping ESP sync. 的输出就没有了,最后成功了。

    # VM 配置

    选中 VM ,在 == 硬件(Hardware)== 部分中,== 机型(Machine)== 应该(尽量)选择 q35 类型,如果不是的话可以在关机状态下修改。

    然后还是在硬件(Hardware)部分,点击添加(add),选择 PCI 设备,找到自己的显卡添加设备。

    或者 CLI 版本操作:

    shell
    qm set VMID -hostpci0 00:02.0
    # or add these to vm config file:
    # hostpci0: 00:02.0

    启动虚拟机,一切无误的话就能够正常启动,然后在虚拟机中就可以当作真实物理机插着显卡进一步安装驱动操作啦

    \ No newline at end of file +PVE 直通显卡 - DevOps - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    PVE 版本:8.1.4 ,显卡:NVIDIA GTX 745

    # 宿主机配置

    # IOMMU

    shell
    nano /etc/default/grub

    GRUB_CMDLINE_LINUX_DEFAULT="..." 中的内容修改为如下:

    shell
    GRUB_CMDLINE_LINUX_DEFAULT="quiet intel_iommu=on iommu=pt video=efifb:off"

    准确来说是在 "..." 原来的内容后面增加这些内容,其中:

    • quiet 一般是原来就有的
    • intel_iommu=oniommu=pt 是必须的
    • video=efifb:off 如果你是给 Windows VM 直通显卡串流玩游戏的话是需要的

    然后更新配置文件:

    shell
    update-grub

    检查 IOMMU 是否启动:

    shell
    dmesg | grep -e DMAR -e IOMMU

    看到有 DMAR 或 IOMMU 输出就行,例如我们就只看到了 DMAR 的输出。

    # 配置显卡

    找到自己的显卡的相关信息:

    shell
    lspci -knn | grep VGA

    我们需要如下内容:

    shell
    01:00.0 VGA compatible controller [0300]: NVIDIA Corporation GM107 [GeForce GTX 745] [10de:1382] (rev a2)
            Subsystem: NVIDIA Corporation GM107 [GeForce GTX 745] [10de:1065] # these ids
            Kernel modules: nvidiafb, nouveau # these modules
    • [10de:1065]<vendor id>:<device id> 形式的厂商 id 和设备 id
    • nvidiafbnouveau 是我们的显卡所使用的驱动模块

    将厂商 id 和设备 id 组合添加到 vfio 配置文件中:

    shell
    nano /etc/modprobe.d/vfio.conf

    增加如下内容:

    shell
    options vfio-pci ids=10de:1382 disable_vga=1

    如果你有多张显卡需要直通,那么不同显卡对应的 <vendor id>:<device id> 就以逗号分隔加在 ids= 后面即可,例如:

    shell
    options vfio-pci ids=1145:1400,1919:1800 disable_vga=1

    屏蔽显卡驱动模块:

    shell
    echo "blacklist <some-module>" >> /etc/modprobe.d/blacklist.conf

    对每一个模块都需要加单独一行,例如我们是 nvidiafbnouveau ,则需要:

    shell
    echo "blacklist nvidiafb" >> /etc/modprobe.d/blacklist.conf
    echo "blacklist nouveau" >> /etc/modprobe.d/blacklist.conf

    有一些教程说是修改配置文件 /etc/modprobe.d/pve-blacklist.conf ,但是官方文档(目前)说的是 /etc/modprobe.d/blacklist.conf ,两者不知道有没有区别

    # vfio

    shell
    nano /etc/modules

    增加如下几行:

    shell
    vfio
    vfio_iommu_type1
    vfio_pci
    vfio_virqfd #not needed if on kernel 6.2 or newer

    最后执行如下指令,并且重启:

    shell
    update-initramfs -u -k all

    最后使用如下指令检查配置是否成功:

    shell
    lspci -k

    预期是得到对应的显卡部分有如下输出结果:

    shell
    01:00.0 VGA compatible controller: NVIDIA Corporation GP108 [GeForce GT 1030] (rev a1)
            Subsystem: Micro-Star International Co., Ltd. [MSI] GP108 [GeForce GT 1030]
            Kernel driver in use: <some-module> # 预期是 vfio
            Kernel modules: <some-module>

    # 问题

    执行最后一句指令时,我们却遇到了这个输出:

    shell
    ...
    Re-executing '/etc/kernel/postinst.d/zz-proxmox-boot' in new private mount namespace..
    No /etc/kernel/proxmox-boot-uuids found, skipping ESP sync.

    然后 lspci -k 检查也没有预期的输出,虚拟机添加显卡设备后无法开机。

    我尝试了官方文档中 Host Device Passthrough 部分的方法:

    For both methods you need to update the initramfs again and reboot after that.

    Should this not work, you might need to set a soft dependency to load the gpu modules before loading vfio-pci. This can be done with the softdep flag, see also the manpages on modprobe.d for more information.

    For example, if you are using drivers named <some-module> :

    shell
    # echo "softdep <some-module> pre: vfio-pci" >> /etc/modprobe.d/<some-module>.conf

    但仍然没有效果。

    所幸,似乎有不少人也遇到了这个问题,我找到了这个帖子,似乎说是因为没有同步连接 ESP 导致的,接下来我按照里面一个解决方案走:

    然后我先尝试了重新挂载 /boot/efi ,首先 lsblk -o +FSTYPE 查看 /boot/efi 挂载的磁盘分区,例如我这里是 sda2 ,然后重新挂载一下:

    shell
    umount /boot/efi
    proxmox-boot-tool init /dev/sda2

    然后先前类似 No /etc/kernel/proxmox-boot-uuids found, skipping ESP sync. 的输出就没有了,最后成功了。

    # VM 配置

    选中 VM ,在 == 硬件(Hardware)== 部分中,== 机型(Machine)== 应该(尽量)选择 q35 类型,如果不是的话可以在关机状态下修改。

    然后还是在硬件(Hardware)部分,点击添加(add),选择 PCI 设备,找到自己的显卡添加设备。

    或者 CLI 版本操作:

    shell
    qm set VMID -hostpci0 00:02.0
    # or add these to vm config file:
    # hostpci0: 00:02.0

    启动虚拟机,一切无误的话就能够正常启动,然后在虚拟机中就可以当作真实物理机插着显卡进一步安装驱动操作啦

    \ No newline at end of file diff --git a/cs/devops/pve/index.html b/cs/devops/pve/index.html index 54c311b7..5abdbaa8 100644 --- a/cs/devops/pve/index.html +++ b/cs/devops/pve/index.html @@ -1,2 +1,2 @@ -PVE 搭建小集群 - DevOps - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 前言

    在 Matrix 实验室用电子垃圾搭了一个小集群。

    起因是 jygg 拿了一台被淘汰的存储型服务器给 cjl 玩,(据说是以前买来发现性能太差了,于是就把里面的 64T 的硬盘全部拿走了装到别的服务器上了,这台就废弃了),cjl 整了个 500g 的机械硬盘给服务器装上了,同时把不知为啥留在这里的一个台式 PC 给我玩,然后索性就装上了 PVE,在 jygg 的帮助下非常顺利搭了一个两台物理机组成的集群。

    # 安装 PVE

    官网下载 PVE,这是一个只有 1G 大小左右的基于 debian 的装好了 proxmox 相关软件、服务的镜像,然后用 U 盘给物理机安装。

    安装过程可以用 proxmox 提供的图形化安装页面,当然安装好的 debian 本身不会有 GUI。会需要用到键盘,以及鼠标 (除非键盘技巧特别好)。

    安装过程有一些需要额外注意:

    • hostname 会设置采取 FQDN 域名,但是实际上 proxmox 只会检查输入的是不是默认的 .invalid 结尾,只需改掉这个即可
      • jygg 说这个域名最好不要填可以被解析到的域名
    • 不需要设置登陆 debian 的用户名,web 控制台登陆用的超级管理员用户名即 root 即可
    • 安装过程设置的 ip 地址后面改起来也比较麻烦(尤其是组建集群时)最好提前确定好
      • jygg 的帮助下我们安装时就给 2 个物理机设置了校园网内可供访问的 ip(Matrix 从校园网络中心租的网段)
    • DNS Server 在这个界面里只能设置一个,可以安装完后再加
    • 安装结束,及时拔掉 U 盘,然后重启即可
    • 此后不再需要显示器键盘鼠标等外设,物理机插着电源和网线即可

    重启后,用浏览器可以直接访问 https://ip:8006https://域名:8006 ,登陆 proxmox 的控制台,需要注意 PVE 强制只能使用 https 协议

    按照先前 jygg 说的安装过程最好不使用可被解析的域名,此时应该没法使用域名访问,用 ip 访问即可

    此外还需记得配置 dns 服务器:

    dns

    # ssl 证书

    这里介绍两种方法,都可以通过 PVE 控制台操作实现。需要注意,两种方法最后节点网页使用的 ssl 证书的路径在该节点 /etc/pve/nodes/节点名/ 目录下,名字默认是:

    • pveproxy-ssl.pem
    • pveproxy-ssl.key

    证书位置

    # 插件自动申请 ssl 证书

    为了 https 访问能够不被浏览器阻挡,也为了安全,还需给自己的域名申请 ssl 证书,此后就用 https://域名:8006 对此 PVE 提供了基于 acme.sh 的自动化脚本,根据不同运营商给 proxmox 安装对应的插件,就可以实现自动申请 ssl 证书

    ssl

    # 注册 ACM 账号

    数据中心 -> ACME -> 账户

    注册ACM账号

    # 添加 ACME 插件

    数据中心 -> ACME -> 质询插件,或者指令 pvenode acme plugin add

    ACME 插件的配置存放在 /etc/pve/priv/acme/plugins.cfg 中 ,一个插件可供整个集群使用

    我们当然没有公网 ip,只能选择 dns 挑战。按照我们购买域名的运营商选择对应 api 接口,在运营商网页 API 相关部分获取对应的接口所需数据,填入数据,然后创建 acme 插件即可。

    acme-plugin

    如图是以 Porkbun 运营商为例创建 acme 插件

    # 添加节点域名

    每一个物理机节点都可以设置一个节点域名(node domain),但实际上一个集群只需要一台机器的 ip 绑定对应域名即可。选择这个 “主节点”:

    机器 -> Certificate / 凭证 -> ACME -> Add / 添加

    选择刚刚创建好的 acme 插件,填入你想要绑定的域名即可:

    节点域名

    最后点击 Order Certificate Now / 立即预约凭证 ,api 填写无误将顺利完成任务得到 OK 状态,后续不再需要手动申请证书,PVE 将利用 acme 插件定期自动申请凭证。

    # 手动上传 ssl 证书

    另一种方式是使用自己申请的 ssl 证书:

    自己上传ssl证书

    如图上传对应的 pemkey 文件即可。当然这种方式需要记得 ssl 过期前重新申请并上传文件。

    # 更换 apt 软件源

    PVE 实际上是一个 Debian ,按照 debian 换源方式操作即可。

    需要注意的是 PVE 内置了一个 enterprise 软件源,但是这个软件源必须向 Proxmox 订阅(充钱)才能使用。我们当然只想白嫖,也不是商业用途,所以我们要把 /etc/apt/sources.list.d/pve-enterprise.list 里的这个软件源注释掉(当然删掉这个文件也是可以的):

    javascript
    deb https://enterprise.proxmox.com/debian/pve buster pve-enterprise

    或者可以在 web 控制台操作禁用这些包含 enterprise 字眼的软件源

    最后还需创建一个 /etc/apt/sources.list.d/pve-no-subscription.list 文件,(当然其实文件名只是给人看的,名字自定义即可)然后在文件内添加无需订阅的 PVE 软件源:

    javascript
    # PVE pve-no-subscription repository provided by proxmox.com,
    # NOT recommended for production use
    deb http://download.proxmox.com/debian/pve bookworm pve-no-subscription

    注意其中 bookworm 是我们的 debian12 版本名,如果你的版本不一样还需换成对应的名字

    最后结果:

    apt

    # 动态配置网络

    默认在 PVE 里更改宿主机的网络配置是需要重启才能生效的,此时所有虚拟机业务都会中断。所以我们需要让 PVE 支持 Reload 网络配置,方法很简单:

    apt install ifupdown2
    -

    如果安装时提示说需要卸载 proxmox-ve,请确保已经加入了 pve-no-subscription 这个软件源并 apt update

    # 配置防火墙

    PVE 的防火墙层级比较复杂,分为三个部分:

    • 集群
    • 主机
    • VM

    每个部分都由各自的开关,并且在每个虚拟机网卡上也有开关。

    ** 首先要打开集群(Cluster)的防火墙开关。** 然后你可以配置整个集群的防火墙规则,比如限制对主机的 22 端口访问。

    但是你会发现没有起作用。这是因为主机(Host)的防火墙开关没开。所以还需要开一下主机(Host)的防火墙开关

    此时主机的防火墙规则应当已经工作,如果还没有工作,请检查网卡的防火墙开关是否开启

    Please however check that you:

    • Enabled firewalling on Data Centre options

    • Enabled firewalling on Node(s)

    • Enabled firewalling on VM

    • Set individual VM network interfaces to firewall

    https://forum.proxmox.com/threads/pve-firewall-doesnt-have-any-effect.47393/

    为了搭建 Kubernetes 节点,下一步需要创建对应的 VM 节点。但手动创建 VM 节点、安装系统也是痛苦的,所以应该使用 Terraform 来简化这个过程,并自始至终保持可管理性。

    # 参考资料

    参考了 Matrix 的知识库。官方文档是全英文的,非常详细,但是英文不太适合快速上手,可以考虑参考中文翻译版本,这个翻译不一定及时更新到最新 proxmox 版本

    Edited on Views times
    \ No newline at end of file +PVE 搭建小集群 - DevOps - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 前言

    在 Matrix 实验室用电子垃圾搭了一个小集群。

    起因是 jygg 拿了一台被淘汰的存储型服务器给 cjl 玩,(据说是以前买来发现性能太差了,于是就把里面的 64T 的硬盘全部拿走了装到别的服务器上了,这台就废弃了),cjl 整了个 500g 的机械硬盘给服务器装上了,同时把不知为啥留在这里的一个台式 PC 给我玩,然后索性就装上了 PVE,在 jygg 的帮助下非常顺利搭了一个两台物理机组成的集群。

    # 安装 PVE

    官网下载 PVE,这是一个只有 1G 大小左右的基于 debian 的装好了 proxmox 相关软件、服务的镜像,然后用 U 盘给物理机安装。

    安装过程可以用 proxmox 提供的图形化安装页面,当然安装好的 debian 本身不会有 GUI。会需要用到键盘,以及鼠标 (除非键盘技巧特别好)。

    安装过程有一些需要额外注意:

    • hostname 会设置采取 FQDN 域名,但是实际上 proxmox 只会检查输入的是不是默认的 .invalid 结尾,只需改掉这个即可
      • jygg 说这个域名最好不要填可以被解析到的域名
    • 不需要设置登陆 debian 的用户名,web 控制台登陆用的超级管理员用户名即 root 即可
    • 安装过程设置的 ip 地址后面改起来也比较麻烦(尤其是组建集群时)最好提前确定好
      • jygg 的帮助下我们安装时就给 2 个物理机设置了校园网内可供访问的 ip(Matrix 从校园网络中心租的网段)
    • DNS Server 在这个界面里只能设置一个,可以安装完后再加
    • 安装结束,及时拔掉 U 盘,然后重启即可
    • 此后不再需要显示器键盘鼠标等外设,物理机插着电源和网线即可

    重启后,用浏览器可以直接访问 https://ip:8006https://域名:8006 ,登陆 proxmox 的控制台,需要注意 PVE 强制只能使用 https 协议

    按照先前 jygg 说的安装过程最好不使用可被解析的域名,此时应该没法使用域名访问,用 ip 访问即可

    此外还需记得配置 dns 服务器:

    dns

    # ssl 证书

    这里介绍两种方法,都可以通过 PVE 控制台操作实现。需要注意,两种方法最后节点网页使用的 ssl 证书的路径在该节点 /etc/pve/nodes/节点名/ 目录下,名字默认是:

    • pveproxy-ssl.pem
    • pveproxy-ssl.key

    证书位置

    # 插件自动申请 ssl 证书

    为了 https 访问能够不被浏览器阻挡,也为了安全,还需给自己的域名申请 ssl 证书,此后就用 https://域名:8006 对此 PVE 提供了基于 acme.sh 的自动化脚本,根据不同运营商给 proxmox 安装对应的插件,就可以实现自动申请 ssl 证书

    ssl

    # 注册 ACM 账号

    数据中心 -> ACME -> 账户

    注册ACM账号

    # 添加 ACME 插件

    数据中心 -> ACME -> 质询插件,或者指令 pvenode acme plugin add

    ACME 插件的配置存放在 /etc/pve/priv/acme/plugins.cfg 中 ,一个插件可供整个集群使用

    我们当然没有公网 ip,只能选择 dns 挑战。按照我们购买域名的运营商选择对应 api 接口,在运营商网页 API 相关部分获取对应的接口所需数据,填入数据,然后创建 acme 插件即可。

    acme-plugin

    如图是以 Porkbun 运营商为例创建 acme 插件

    # 添加节点域名

    每一个物理机节点都可以设置一个节点域名(node domain),但实际上一个集群只需要一台机器的 ip 绑定对应域名即可。选择这个 “主节点”:

    机器 -> Certificate / 凭证 -> ACME -> Add / 添加

    选择刚刚创建好的 acme 插件,填入你想要绑定的域名即可:

    节点域名

    最后点击 Order Certificate Now / 立即预约凭证 ,api 填写无误将顺利完成任务得到 OK 状态,后续不再需要手动申请证书,PVE 将利用 acme 插件定期自动申请凭证。

    # 手动上传 ssl 证书

    另一种方式是使用自己申请的 ssl 证书:

    自己上传ssl证书

    如图上传对应的 pemkey 文件即可。当然这种方式需要记得 ssl 过期前重新申请并上传文件。

    # 更换 apt 软件源

    PVE 实际上是一个 Debian ,按照 debian 换源方式操作即可。

    需要注意的是 PVE 内置了一个 enterprise 软件源,但是这个软件源必须向 Proxmox 订阅(充钱)才能使用。我们当然只想白嫖,也不是商业用途,所以我们要把 /etc/apt/sources.list.d/pve-enterprise.list 里的这个软件源注释掉(当然删掉这个文件也是可以的):

    javascript
    deb https://enterprise.proxmox.com/debian/pve buster pve-enterprise

    或者可以在 web 控制台操作禁用这些包含 enterprise 字眼的软件源

    最后还需创建一个 /etc/apt/sources.list.d/pve-no-subscription.list 文件,(当然其实文件名只是给人看的,名字自定义即可)然后在文件内添加无需订阅的 PVE 软件源:

    javascript
    # PVE pve-no-subscription repository provided by proxmox.com,
    # NOT recommended for production use
    deb http://download.proxmox.com/debian/pve bookworm pve-no-subscription

    注意其中 bookworm 是我们的 debian12 版本名,如果你的版本不一样还需换成对应的名字

    最后结果:

    apt

    # 动态配置网络

    默认在 PVE 里更改宿主机的网络配置是需要重启才能生效的,此时所有虚拟机业务都会中断。所以我们需要让 PVE 支持 Reload 网络配置,方法很简单:

    apt install ifupdown2
    +

    如果安装时提示说需要卸载 proxmox-ve,请确保已经加入了 pve-no-subscription 这个软件源并 apt update

    # 配置防火墙

    PVE 的防火墙层级比较复杂,分为三个部分:

    • 集群
    • 主机
    • VM

    每个部分都由各自的开关,并且在每个虚拟机网卡上也有开关。

    ** 首先要打开集群(Cluster)的防火墙开关。** 然后你可以配置整个集群的防火墙规则,比如限制对主机的 22 端口访问。

    但是你会发现没有起作用。这是因为主机(Host)的防火墙开关没开。所以还需要开一下主机(Host)的防火墙开关

    此时主机的防火墙规则应当已经工作,如果还没有工作,请检查网卡的防火墙开关是否开启

    Please however check that you:

    • Enabled firewalling on Data Centre options

    • Enabled firewalling on Node(s)

    • Enabled firewalling on VM

    • Set individual VM network interfaces to firewall

    https://forum.proxmox.com/threads/pve-firewall-doesnt-have-any-effect.47393/

    为了搭建 Kubernetes 节点,下一步需要创建对应的 VM 节点。但手动创建 VM 节点、安装系统也是痛苦的,所以应该使用 Terraform 来简化这个过程,并自始至终保持可管理性。

    # 参考资料

    参考了 Matrix 的知识库。官方文档是全英文的,非常详细,但是英文不太适合快速上手,可以考虑参考中文翻译版本,这个翻译不一定及时更新到最新 proxmox 版本

    Edited on Views times
    \ No newline at end of file diff --git a/cs/devops/sql/index.html b/cs/devops/sql/index.html index 5eae2135..05d38e90 100644 --- a/cs/devops/sql/index.html +++ b/cs/devops/sql/index.html @@ -1 +1 @@ -SQL 基础 - DevOps - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # DBMS

    数据库管理系统 DBMS 的分类:

    關聯式資料庫(SQL)非關聯式資料庫 - (nosQL /not just SQL)
    • MySQL• MogoDB
    • Oracle• Redis
    • PostgreSQL• DynamoDB
    • SQL Server• Elaticsearch
    關聯式資料庫管理系統 (RDBMS)非關聯式資料庫管理系統 (NRDBMS)

    SQL - Structured Query Language - 结构性询问语言

    Edited on Views times
    \ No newline at end of file +SQL 基础 - DevOps - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # DBMS

    数据库管理系统 DBMS 的分类:

    關聯式資料庫(SQL)非關聯式資料庫 - (nosQL /not just SQL)
    • MySQL• MogoDB
    • Oracle• Redis
    • PostgreSQL• DynamoDB
    • SQL Server• Elaticsearch
    關聯式資料庫管理系統 (RDBMS)非關聯式資料庫管理系統 (NRDBMS)

    SQL - Structured Query Language - 结构性询问语言

    Edited on Views times
    \ No newline at end of file diff --git a/cs/fronet-end/hexo/index.html b/cs/fronet-end/hexo/index.html index e56139fe..022d4d48 100644 --- a/cs/fronet-end/hexo/index.html +++ b/cs/fronet-end/hexo/index.html @@ -1,3 +1,3 @@ -Hexo - 前端 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    Hexo 是一个基于 Node 框架的博客框架

    pnpm install hexo-cli -g		//安装
    hexo init blog						//初始化
    cd blog; npm install			//安装依赖

    blog 目录下生成静态文件(到 public 目录下,用于发布)

    hexo g						//缩略
    hexo generate			//一样

    blog 目录下清除已生成的静态文件和 cache( 到 public 目录下)

    hexo clean

    blog 目录下本地运行 hexo 服务

    hexo s						//缩略
    hexo server				//一样

    在 blog 目录下将 public 目录下的静态文件复制到 nginx 的对应目录,从而使用 nginx 服务部署:

    sudo cp -r public/* /var/www/html/

    一键部署,参考: https://hexo.io/docs/one-command-deployment

    hexo d					//缩略
    +Hexo - 前端 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    Hexo 是一个基于 Node 框架的博客框架

    pnpm install hexo-cli -g		//安装
    hexo init blog						//初始化
    cd blog; npm install			//安装依赖

    blog 目录下生成静态文件(到 public 目录下,用于发布)

    hexo g						//缩略
    hexo generate			//一样

    blog 目录下清除已生成的静态文件和 cache( 到 public 目录下)

    hexo clean

    blog 目录下本地运行 hexo 服务

    hexo s						//缩略
    hexo server				//一样

    在 blog 目录下将 public 目录下的静态文件复制到 nginx 的对应目录,从而使用 nginx 服务部署:

    sudo cp -r public/* /var/www/html/

    一键部署,参考: https://hexo.io/docs/one-command-deployment

    hexo d					//缩略
     hexo deploy			//一样
    -
    Edited on Views times
    \ No newline at end of file +
    Edited on Views times
    \ No newline at end of file diff --git a/cs/fronet-end/nodejs/index.html b/cs/fronet-end/nodejs/index.html index f26d380c..da0de208 100644 --- a/cs/fronet-end/nodejs/index.html +++ b/cs/fronet-end/nodejs/index.html @@ -1,5 +1,5 @@ -Node.js - 前端 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # require

    require 是 Node.js 中用于加载模块的函数。它的主要作用是将外部模块引入到当前的 JavaScript 文件中,以便在代码中使用被引入的模块的功能、变量或对象。

    require 函数的基本语法是:

    require(moduleName);

    其中, moduleName 是要引入的模块的名称或路径。

    require 函数的工作原理如下:

    1. 首先,Node.js 会检查 moduleName 是否是一个核心模块(Node.js 内置的模块)。如果是核心模块,Node.js 会直接加载并返回该模块的导出内容。
    2. 如果 moduleName 不是核心模块,Node.js 会尝试在当前目录下查找指定名称的文件或文件夹。
      • 如果指定名称是一个文件,Node.js 将尝试加载该文件,并返回文件中导出的内容。
      • 如果指定名称是一个文件夹,Node.js 将尝试查找该文件夹下的 package.json 文件。如果找到了 package.json 文件,并且文件中包含一个 "main" 属性,Node.js 将加载该属性指定的文件,并返回文件中导出的内容。
      • 如果以上两种情况都不满足,Node.js 将尝试查找指定名称的文件夹下的 index.js 文件,并返回文件中导出的内容。
    3. 如果在当前目录下找不到指定的文件或文件夹,Node.js 会递归地向上查找,直到找到指定的模块或到达文件系统的根目录为止。如果在任何一个父级目录中找到了指定的模块,Node.js 将加载该模块并返回导出的内容。
    4. 如果在所有的搜索路径中都找不到指定的模块,Node.js 将抛出一个 Error

    需要注意的是, require 函数是同步的,并且模块在第一次被引入后会被缓存,所以多次调用 require 函数引入同一个模块不会导致模块被重复加载,而是直接返回缓存的内容。如果想要重新加载一个模块,可以使用 delete require.cache[moduleName] 来删除缓存。

    # HTTP 请求

    在现代的 Node.js 环境中,进行 HTTP 请求最广泛使用且比较前沿的模块是 node-fetchaxios 。它们都支持 TypeScript,并提供了简洁的 API 来发送 HTTP 请求。

    1. node-fetchnode-fetch 是一个基于 Fetch API 标准的轻量级 HTTP 客户端。它提供了类似于浏览器中 fetch 函数的接口,可以发送 HTTP 请求并处理响应。 node-fetch 在 Node.js 环境中非常流行,因为它支持 Promise,并且使用起来非常简洁。你可以使用以下命令安装 node-fetch 模块:

      npm install node-fetch
      +Node.js - 前端 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

      # require

      require 是 Node.js 中用于加载模块的函数。它的主要作用是将外部模块引入到当前的 JavaScript 文件中,以便在代码中使用被引入的模块的功能、变量或对象。

      require 函数的基本语法是:

      require(moduleName);

      其中, moduleName 是要引入的模块的名称或路径。

      require 函数的工作原理如下:

      1. 首先,Node.js 会检查 moduleName 是否是一个核心模块(Node.js 内置的模块)。如果是核心模块,Node.js 会直接加载并返回该模块的导出内容。
      2. 如果 moduleName 不是核心模块,Node.js 会尝试在当前目录下查找指定名称的文件或文件夹。
        • 如果指定名称是一个文件,Node.js 将尝试加载该文件,并返回文件中导出的内容。
        • 如果指定名称是一个文件夹,Node.js 将尝试查找该文件夹下的 package.json 文件。如果找到了 package.json 文件,并且文件中包含一个 "main" 属性,Node.js 将加载该属性指定的文件,并返回文件中导出的内容。
        • 如果以上两种情况都不满足,Node.js 将尝试查找指定名称的文件夹下的 index.js 文件,并返回文件中导出的内容。
      3. 如果在当前目录下找不到指定的文件或文件夹,Node.js 会递归地向上查找,直到找到指定的模块或到达文件系统的根目录为止。如果在任何一个父级目录中找到了指定的模块,Node.js 将加载该模块并返回导出的内容。
      4. 如果在所有的搜索路径中都找不到指定的模块,Node.js 将抛出一个 Error

      需要注意的是, require 函数是同步的,并且模块在第一次被引入后会被缓存,所以多次调用 require 函数引入同一个模块不会导致模块被重复加载,而是直接返回缓存的内容。如果想要重新加载一个模块,可以使用 delete require.cache[moduleName] 来删除缓存。

      # HTTP 请求

      在现代的 Node.js 环境中,进行 HTTP 请求最广泛使用且比较前沿的模块是 node-fetchaxios 。它们都支持 TypeScript,并提供了简洁的 API 来发送 HTTP 请求。

      1. node-fetchnode-fetch 是一个基于 Fetch API 标准的轻量级 HTTP 客户端。它提供了类似于浏览器中 fetch 函数的接口,可以发送 HTTP 请求并处理响应。 node-fetch 在 Node.js 环境中非常流行,因为它支持 Promise,并且使用起来非常简洁。你可以使用以下命令安装 node-fetch 模块:

        npm install node-fetch
         

        或者使用 yarn:

        yarn add node-fetch
         

        安装完成后,你可以在 TypeScript 代码中引入并使用 node-fetch 模块来发送 HTTP 请求。

      2. axiosaxios 是一个流行的基于 Promise 的 HTTP 客户端,可以在 Node.js 环境和浏览器中使用。它提供了简洁的 API 和许多功能,如请求拦截器、响应拦截器、取消请求等。 axios 在 Node.js 环境中也非常受欢迎,并且有广泛的社区支持。你可以使用以下命令安装 axios 模块:

        npm install axios
         

        或者使用 yarn:

        yarn add axios
        -

        安装完成后,你可以在 TypeScript 代码中引入并使用 axios 模块来发送 HTTP 请求。

      这两个模块都非常流行和可靠,具体使用哪个取决于你的个人偏好和项目要求。它们都提供了良好的文档和示例,以帮助你开始使用它们进行 HTTP 请求。

      Edited on Views times
      \ No newline at end of file +

      安装完成后,你可以在 TypeScript 代码中引入并使用 axios 模块来发送 HTTP 请求。

    这两个模块都非常流行和可靠,具体使用哪个取决于你的个人偏好和项目要求。它们都提供了良好的文档和示例,以帮助你开始使用它们进行 HTTP 请求。

    Edited on Views times
    \ No newline at end of file diff --git a/cs/languages/js/index.html b/cs/languages/js/index.html index e4892c87..3d58f40e 100644 --- a/cs/languages/js/index.html +++ b/cs/languages/js/index.html @@ -1 +1 @@ -JavaScript 基础 - 编程语言 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # JSON.stringify()

    JSON.stringify() 是 JavaScript 中的一个方法,用于将 JavaScript 对象或值转换为 JSON 字符串。

    具体作用如下:

    • 序列化: JSON.stringify() 方法将 JavaScript 对象或值序列化为一个 JSON 字符串。序列化是指将对象的结构和数据转换为字符串的过程,使其可以在网络传输或存储中进行传递和持久化。
    • JSON 格式:JSON(JavaScript Object Notation)是一种常用的数据交换格式,具有简洁、易读和跨语言的特性。 JSON.stringify() 将 JavaScript 对象或值转换为符合 JSON 格式的字符串,方便与其他系统或服务进行数据交互。

    语法:

    JSON.stringify(value[, replacer[, space]])
    • value :要被序列化为 JSON 字符串的 JavaScript 对象或值。
    • replacer (可选):一个函数或数组,用于控制序列化过程中的属性过滤和转换。可以传递一个函数作为 replacer,该函数将被应用于对象的每个属性,并在序列化时对属性值进行转换。也可以传递一个数组,包含要序列化的属性名列表,仅包含在该列表中的属性将被序列化。
    • space (可选):用于美化输出的空格数。如果是一个数字,则表示使用相应数量的空格进行缩进;如果是一个字符串,则使用该字符串作为缩进符号。

    下面是一个简单的示例:

    const obj = {
      name: 'Alice',
      age: 30,
      isActive: true
    };
    const jsonString = JSON.stringify(obj);
    console.log(jsonString);
    // 输出: '{"name":"Alice","age":30,"isActive":true}'

    在上面的示例中, obj 对象通过 JSON.stringify() 方法被序列化为一个 JSON 字符串。 jsonString 的值为 {"name":"Alice","age":30,"isActive":true} 。该字符串符合 JSON 格式,可以在传输或存储中使用。

    JSON.stringify() 方法用于将 JavaScript 对象或值转换为 JSON 字符串,方便数据的传输和持久化。

    # JSON.parse()

    JSON.parse() 是一个 JavaScript 内置函数,用于将符合 JSON 格式的字符串转换为对应的 JavaScript 对象。

    JSON.parse() 的基本语法如下:

    JSON.parse(jsonString);

    其中, jsonString 是一个符合 JSON 格式的字符串。

    JSON.parse() 函数的工作原理如下:

    1. 首先,它会接收一个 JSON 格式的字符串作为输入。
    2. 然后,它会解析这个字符串,并将其转换为对应的 JavaScript 对象。
    3. 如果解析成功,它将返回转换后的 JavaScript 对象;如果解析失败,它将抛出一个 SyntaxError

    在将字符串转换为 JavaScript 对象时, JSON.parse() 支持的 JSON 数据类型有:

    • 字符串(字符串需要使用双引号)
    • 数字(整数或浮点数)
    • 布尔值( truefalse
    • null
    • 数组(用方括号 [] 表示)
    • 对象(用花括号 {} 表示)

    以下是一些示例,展示了 JSON.parse() 函数的用法:

    const jsonString = '{"name": "John", "age": 30, "isStudent": true, "hobbies": ["reading", "swimming"]}';
    const obj = JSON.parse(jsonString);
    console.log(obj.name); // 输出: John
    console.log(obj.age); // 输出: 30
    console.log(obj.isStudent); // 输出: true
    console.log(obj.hobbies[0]); // 输出: reading
    console.log(obj.hobbies[1]); // 输出: swimming

    在这个示例中,我们有一个符合 JSON 格式的字符串 jsonString ,然后我们使用 JSON.parse() 将其转换为了一个 JavaScript 对象 obj 。接下来,我们可以通过访问 obj 的属性来获取和操作 JSON 数据。

    需要注意的是, JSON.parse() 只能解析合法的 JSON 字符串,因此如果提供给它的字符串不符合 JSON 语法,就会导致解析失败并抛出 SyntaxError 。另外, JSON.parse() 不支持解析带有函数、日期或正则表达式等特殊类型的 JSON 数据。

    # JSON.parse(JSON.stringify( ))

    JSON.parse(JSON.stringify()) 函数组合用法可以将一个 JavaScript 对象进行深度克隆,从而创建一个与原对象完全相同的新对象。这个过程可以确保输入和输出是不变的。

    JSON.stringify() 函数将 JavaScript 对象转换为 JSON 字符串,而 JSON.parse() 函数将 JSON 字符串解析为 JavaScript 对象。通过将一个对象先使用 JSON.stringify() 转换为字符串,然后再使用 JSON.parse() 解析回对象,可以创建一个新的对象,该对象与原对象在值上是相等的,但是在引用上是不同的。

    这个过程的效果是创建了原对象的一个深度副本(deep copy),而不是对原对象进行引用传递。这在需要确保对原对象的修改不会影响到副本时非常有用。

    需要注意的是,这种深度克隆方法只适用于可以被 JSON 序列化和反序列化的数据类型,例如对象、数组、字符串、数字、布尔值等。对于无法被 JSON 序列化的数据类型,比如函数、正则表达式、 undefined 等,这种方法将无法克隆它们。

    下面是一个示例,展示了 JSON.parse(JSON.stringify()) 函数组合的效果:

    const obj = { a: 1, b: { c: 2 } };
    const clone = JSON.parse(JSON.stringify(obj));
    console.log(obj);   // { a: 1, b: { c: 2 } }
    console.log(clone); // { a: 1, b: { c: 2 } }
    console.log(obj === clone);                  // false
    console.log(obj.b === clone.b);              // false
    console.log(obj.b.c === clone.b.c);          // true

    在上述示例中, obj 对象被克隆为 clone 对象,它们的值是相等的,但是它们是两个不同的对象,因此对其中一个对象的修改不会影响到另一个对象。

    # Promise

    Promise 是 JavaScript 中处理异步操作的一种机制,它代表了一个异步操作的最终完成或失败,并可以返回结果或错误。

    在传统的 JavaScript 中,处理异步操作通常使用回调函数。但是,回调函数的嵌套会导致代码变得难以理解和维护,形成所谓的 "回调地狱"。为了解决这个问题,Promise 被引入作为一种更优雅和可读性更高的异步编程模式。

    一个 Promise 对象代表了一个异步操作,它有三种状态:

    1. Pending(进行中):初始状态,表示异步操作还未完成,仍处于进行中的状态。

    2. Fulfilled(已成功):表示异步操作已经成功完成,并返回了一个值。在这种状态下,Promise 对象会调用 then 方法注册的回调函数,并将异步操作的结果传递给这些回调函数。

    3. Rejected(已失败):表示异步操作发生了错误或失败。在这种状态下,Promise 对象会调用 catch 方法注册的回调函数,并将错误信息传递给这些回调函数。

    Promise 对象具有以下特点:

    • Promise 对象是不可变的,一旦进入了 Fulfilled 或 Rejected 状态,就不会再发生状态的改变。
    • 通过调用 then 方法可以注册成功回调函数,通过调用 catch 方法可以注册失败回调函数。
    • Promise 对象可以被链式调用,通过返回新的 Promise 对象来实现链式操作。
    • Promise 提供了一些静态方法,如 Promise.allPromise.race 等,用于处理多个 Promise 对象。

    在现代的 JavaScript 中,Promise 已经成为了标准的异步操作处理方式,被广泛应用于各种场景,如网络请求、文件操作等。它提供了一种更直观、可靠和易于使用的方式来处理异步操作,避免了回调地狱问题,提高了代码的可读性和可维护性。

    # 网络请求处理

    # handler(req, res)

    async function handler(req, res) 是一个异步函数,通常用作处理 HTTP 请求的处理程序。这个函数接受两个参数 reqres ,分别代表请求对象和响应对象。

    具体的函数实现和参数可能会根据具体的需求和开发框架而有所不同。在某些情况下, reqres 参数可能会被命名为其他名称,而函数的实现也可能会有其他逻辑。

    以下是一个示例,展示了一个类似的函数实现,用于处理 HTTP POST 请求,返回一个 JSON 响应:

    // handler.js
    export default async function handler(req, res) {
      try {
        const { body } = req;
        
        // 处理请求逻辑
        const result = await processRequest(body);
        
        // 返回 JSON 响应
        res.status(200).json({ success: true, data: result });
      } catch (error) {
        console.error(error);
        res.status(500).json({ success: false, error: "Internal Server Error" });
      }
    }

    在上述示例中, handler 函数是一个异步函数,它接受 reqres 作为参数。函数内部可以根据具体需求进行逻辑处理,例如从请求体中获取数据、调用其他函数进行处理,并最终返回一个 JSON 响应。

    # axios

    # axios.post

    await axios.post 是使用 axios 发送 POST 请求并等待其响应的语法。使用 await 关键字可以让 JavaScript 在发送请求后暂停执行,直到获得响应为止。

    下面是一个使用 await axios.post 发送 POST 请求的示例:

    import axios from 'axios';
    async function postData() {
      try {
        const response = await axios.post('https://api.example.com/data', {
          name: 'John',
          age: 30
        });
        console.log(response.data);
      } catch (error) {
        console.error(error);
      }
    }
    postData();

    在这个示例中,我们定义了一个名为 postData 的异步函数。在该函数内部,我们使用 await axios.post 发送 POST 请求到指定的 URL,并传递一个包含数据的 JavaScript 对象作为请求体。

    当发送请求时,JavaScript 会等待 axios 返回响应。一旦收到响应,它会将响应存储在 response 变量中。我们可以使用 response.data 来访问响应数据。

    如果请求成功,我们打印出响应数据。如果请求失败,则会捕获错误并打印错误信息。

    需要注意的是,在使用 await 关键字之前,必须将其包裹在 async 函数内部。这是因为 await 只能在异步函数中使用。

    在使用 await 时,必须将其放在 async 函数内部。这样才能正确地使用 await axios.post 并处理响应。

    axios.post 是一个用于发送 HTTP POST 请求的方法,它返回一个 Promise 对象。这个 Promise 对象在请求完成后会被解析,提供了请求的响应数据和其他相关信息。

    await axios.post 的返回值是一个 Promise,它解析为 axios 的响应对象。响应对象具有许多属性,其中最常用的是 data 属性,它包含服务器返回的数据。具体地说, axios.post 方法返回的 Promise 对象具有以下结构:

    interface AxiosResponse<T = any> {
      data: T;
      status: number;
      statusText: string;
      headers: any;
      config: AxiosRequestConfig;
      request?: any;
    }
    interface AxiosError<T = any> extends Error {
      config: AxiosRequestConfig;
      code?: string;
      request?: any;
      response?: AxiosResponse<T>;
      isAxiosError: boolean;
    }
    interface AxiosPromise<T = any> extends Promise<AxiosResponse<T>> {}
    • AxiosResponse 是成功响应的类型,它包含以下属性:

      • data :响应的数据,类型为泛型 T 所指定的类型。
      • status :HTTP 响应状态码。
      • statusText :HTTP 响应状态文本。
      • headers :响应的 HTTP 头信息。
      • config :发起请求时的配置对象。
      • request :发起的请求实例。
    • AxiosError 是请求错误时的类型,它继承自 Error 对象,包含以下属性:

      • config :发起请求时的配置对象。
      • code :请求错误的错误码。
      • request :发起的请求实例。
      • response :包含错误响应的 AxiosResponse 对象。
      • isAxiosError :标识是否为 Axios 错误。
    • AxiosPromiseaxios.post 方法返回的 Promise 对象的类型,它是一个泛型类型,泛型 T 表示响应数据的类型。

    通过对返回的 Promise 对象调用 .then 方法,您可以注册成功回调函数来处理响应数据。如果请求失败,则可以通过 .catch 方法注册错误回调函数进行错误处理。

    示例代码:

    axios.post(url, data)
      .then((response: AxiosResponse) => {
        // 处理成功响应
        console.log(response.data);
      })
      .catch((error: AxiosError) => {
        // 处理错误
        console.error(error.message);
      });

    请注意, axios.post 返回的 Promise 对象根据请求结果可能会被解析为成功响应或错误响应。因此,在处理响应时,建议使用 .then.catch 方法来适当地处理成功和错误的情况。

    # 网络请求

    # fetch( )

    使用 JavaScript 内置的 Fetch API 进行网络请求:

    1. 构建请求参数:创建一个包含请求方法、请求头、请求体等信息的配置对象。

    2. 发起网络请求:使用 fetch() 函数发送请求,并将配置对象作为参数传递给它。 fetch() 函数返回一个 Promise 对象,代表异步操作的结果。

    3. 处理响应:通过 Promise 的解决来获取服务器返回的响应。可以使用响应对象的方法和属性来读取响应的状态、头部和主体数据。

    4. 解析响应数据:根据服务器返回的数据类型,使用合适的方法将响应的主体数据解析为可用的格式,例如 JSON、文本或二进制数据。

    下面是一个使用 Fetch API 进行 GET 请求的示例:

    fetch('https://api.example.com/data')
      .then(response => {
        if (!response.ok) {
          throw new Error('Network response was not ok');
        }
        return response.json();
      })
      .then(data => {
        console.log(data); // 处理响应数据
      })
      .catch(error => {
        console.error('Error:', error);
      });

    上述示例展示了一个简单的 GET 请求,其中:

    • 使用 fetch() 函数发送了一个 GET 请求到 'https://api.example.com/data'
    • fetch() 函数返回的 Promise 对象通过 .then() 方法进行链式处理。在第一个 .then() 中,我们检查响应的状态码是否为成功状态( ok ),如果不是,则抛出一个错误。
    • 如果响应状态码正常,我们使用 response.json() 方法将响应的主体数据解析为 JSON 格式。
    • 最后一个 .then() 用于处理解析后的数据。
    • 如果在请求过程中发生错误,或任何一个 .then() 中抛出了错误,将进入 .catch() ,在控制台输出错误信息。

    类似地,你可以使用 fetch() 函数进行 POST 请求,只需在配置对象中设置请求方法为 'POST' ,并通过 body 属性提供请求体数据。

    请注意,Fetch API 默认不会将错误状态(如 404 或 500)视为网络错误。它会将这些状态码视为请求成功,并通过 response.ok 属性进行判断。如果你希望将这些状态码也视为网络错误,可以在第一个 .then() 中进行额外的检查或自定义处理逻辑。

    这是使用 JavaScript 内置的 Fetch API 进行网络请求的基本流程。根据具体的需求和场景,你可以进一步使用其他方法和属性来扩展和定制请求和响应的处理。

    Edited on Views times
    \ No newline at end of file +JavaScript 基础 - 编程语言 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # JSON.stringify()

    JSON.stringify() 是 JavaScript 中的一个方法,用于将 JavaScript 对象或值转换为 JSON 字符串。

    具体作用如下:

    • 序列化: JSON.stringify() 方法将 JavaScript 对象或值序列化为一个 JSON 字符串。序列化是指将对象的结构和数据转换为字符串的过程,使其可以在网络传输或存储中进行传递和持久化。
    • JSON 格式:JSON(JavaScript Object Notation)是一种常用的数据交换格式,具有简洁、易读和跨语言的特性。 JSON.stringify() 将 JavaScript 对象或值转换为符合 JSON 格式的字符串,方便与其他系统或服务进行数据交互。

    语法:

    JSON.stringify(value[, replacer[, space]])
    • value :要被序列化为 JSON 字符串的 JavaScript 对象或值。
    • replacer (可选):一个函数或数组,用于控制序列化过程中的属性过滤和转换。可以传递一个函数作为 replacer,该函数将被应用于对象的每个属性,并在序列化时对属性值进行转换。也可以传递一个数组,包含要序列化的属性名列表,仅包含在该列表中的属性将被序列化。
    • space (可选):用于美化输出的空格数。如果是一个数字,则表示使用相应数量的空格进行缩进;如果是一个字符串,则使用该字符串作为缩进符号。

    下面是一个简单的示例:

    const obj = {
      name: 'Alice',
      age: 30,
      isActive: true
    };
    const jsonString = JSON.stringify(obj);
    console.log(jsonString);
    // 输出: '{"name":"Alice","age":30,"isActive":true}'

    在上面的示例中, obj 对象通过 JSON.stringify() 方法被序列化为一个 JSON 字符串。 jsonString 的值为 {"name":"Alice","age":30,"isActive":true} 。该字符串符合 JSON 格式,可以在传输或存储中使用。

    JSON.stringify() 方法用于将 JavaScript 对象或值转换为 JSON 字符串,方便数据的传输和持久化。

    # JSON.parse()

    JSON.parse() 是一个 JavaScript 内置函数,用于将符合 JSON 格式的字符串转换为对应的 JavaScript 对象。

    JSON.parse() 的基本语法如下:

    JSON.parse(jsonString);

    其中, jsonString 是一个符合 JSON 格式的字符串。

    JSON.parse() 函数的工作原理如下:

    1. 首先,它会接收一个 JSON 格式的字符串作为输入。
    2. 然后,它会解析这个字符串,并将其转换为对应的 JavaScript 对象。
    3. 如果解析成功,它将返回转换后的 JavaScript 对象;如果解析失败,它将抛出一个 SyntaxError

    在将字符串转换为 JavaScript 对象时, JSON.parse() 支持的 JSON 数据类型有:

    • 字符串(字符串需要使用双引号)
    • 数字(整数或浮点数)
    • 布尔值( truefalse
    • null
    • 数组(用方括号 [] 表示)
    • 对象(用花括号 {} 表示)

    以下是一些示例,展示了 JSON.parse() 函数的用法:

    const jsonString = '{"name": "John", "age": 30, "isStudent": true, "hobbies": ["reading", "swimming"]}';
    const obj = JSON.parse(jsonString);
    console.log(obj.name); // 输出: John
    console.log(obj.age); // 输出: 30
    console.log(obj.isStudent); // 输出: true
    console.log(obj.hobbies[0]); // 输出: reading
    console.log(obj.hobbies[1]); // 输出: swimming

    在这个示例中,我们有一个符合 JSON 格式的字符串 jsonString ,然后我们使用 JSON.parse() 将其转换为了一个 JavaScript 对象 obj 。接下来,我们可以通过访问 obj 的属性来获取和操作 JSON 数据。

    需要注意的是, JSON.parse() 只能解析合法的 JSON 字符串,因此如果提供给它的字符串不符合 JSON 语法,就会导致解析失败并抛出 SyntaxError 。另外, JSON.parse() 不支持解析带有函数、日期或正则表达式等特殊类型的 JSON 数据。

    # JSON.parse(JSON.stringify( ))

    JSON.parse(JSON.stringify()) 函数组合用法可以将一个 JavaScript 对象进行深度克隆,从而创建一个与原对象完全相同的新对象。这个过程可以确保输入和输出是不变的。

    JSON.stringify() 函数将 JavaScript 对象转换为 JSON 字符串,而 JSON.parse() 函数将 JSON 字符串解析为 JavaScript 对象。通过将一个对象先使用 JSON.stringify() 转换为字符串,然后再使用 JSON.parse() 解析回对象,可以创建一个新的对象,该对象与原对象在值上是相等的,但是在引用上是不同的。

    这个过程的效果是创建了原对象的一个深度副本(deep copy),而不是对原对象进行引用传递。这在需要确保对原对象的修改不会影响到副本时非常有用。

    需要注意的是,这种深度克隆方法只适用于可以被 JSON 序列化和反序列化的数据类型,例如对象、数组、字符串、数字、布尔值等。对于无法被 JSON 序列化的数据类型,比如函数、正则表达式、 undefined 等,这种方法将无法克隆它们。

    下面是一个示例,展示了 JSON.parse(JSON.stringify()) 函数组合的效果:

    const obj = { a: 1, b: { c: 2 } };
    const clone = JSON.parse(JSON.stringify(obj));
    console.log(obj);   // { a: 1, b: { c: 2 } }
    console.log(clone); // { a: 1, b: { c: 2 } }
    console.log(obj === clone);                  // false
    console.log(obj.b === clone.b);              // false
    console.log(obj.b.c === clone.b.c);          // true

    在上述示例中, obj 对象被克隆为 clone 对象,它们的值是相等的,但是它们是两个不同的对象,因此对其中一个对象的修改不会影响到另一个对象。

    # Promise

    Promise 是 JavaScript 中处理异步操作的一种机制,它代表了一个异步操作的最终完成或失败,并可以返回结果或错误。

    在传统的 JavaScript 中,处理异步操作通常使用回调函数。但是,回调函数的嵌套会导致代码变得难以理解和维护,形成所谓的 "回调地狱"。为了解决这个问题,Promise 被引入作为一种更优雅和可读性更高的异步编程模式。

    一个 Promise 对象代表了一个异步操作,它有三种状态:

    1. Pending(进行中):初始状态,表示异步操作还未完成,仍处于进行中的状态。

    2. Fulfilled(已成功):表示异步操作已经成功完成,并返回了一个值。在这种状态下,Promise 对象会调用 then 方法注册的回调函数,并将异步操作的结果传递给这些回调函数。

    3. Rejected(已失败):表示异步操作发生了错误或失败。在这种状态下,Promise 对象会调用 catch 方法注册的回调函数,并将错误信息传递给这些回调函数。

    Promise 对象具有以下特点:

    • Promise 对象是不可变的,一旦进入了 Fulfilled 或 Rejected 状态,就不会再发生状态的改变。
    • 通过调用 then 方法可以注册成功回调函数,通过调用 catch 方法可以注册失败回调函数。
    • Promise 对象可以被链式调用,通过返回新的 Promise 对象来实现链式操作。
    • Promise 提供了一些静态方法,如 Promise.allPromise.race 等,用于处理多个 Promise 对象。

    在现代的 JavaScript 中,Promise 已经成为了标准的异步操作处理方式,被广泛应用于各种场景,如网络请求、文件操作等。它提供了一种更直观、可靠和易于使用的方式来处理异步操作,避免了回调地狱问题,提高了代码的可读性和可维护性。

    # 网络请求处理

    # handler(req, res)

    async function handler(req, res) 是一个异步函数,通常用作处理 HTTP 请求的处理程序。这个函数接受两个参数 reqres ,分别代表请求对象和响应对象。

    具体的函数实现和参数可能会根据具体的需求和开发框架而有所不同。在某些情况下, reqres 参数可能会被命名为其他名称,而函数的实现也可能会有其他逻辑。

    以下是一个示例,展示了一个类似的函数实现,用于处理 HTTP POST 请求,返回一个 JSON 响应:

    // handler.js
    export default async function handler(req, res) {
      try {
        const { body } = req;
        
        // 处理请求逻辑
        const result = await processRequest(body);
        
        // 返回 JSON 响应
        res.status(200).json({ success: true, data: result });
      } catch (error) {
        console.error(error);
        res.status(500).json({ success: false, error: "Internal Server Error" });
      }
    }

    在上述示例中, handler 函数是一个异步函数,它接受 reqres 作为参数。函数内部可以根据具体需求进行逻辑处理,例如从请求体中获取数据、调用其他函数进行处理,并最终返回一个 JSON 响应。

    # axios

    # axios.post

    await axios.post 是使用 axios 发送 POST 请求并等待其响应的语法。使用 await 关键字可以让 JavaScript 在发送请求后暂停执行,直到获得响应为止。

    下面是一个使用 await axios.post 发送 POST 请求的示例:

    import axios from 'axios';
    async function postData() {
      try {
        const response = await axios.post('https://api.example.com/data', {
          name: 'John',
          age: 30
        });
        console.log(response.data);
      } catch (error) {
        console.error(error);
      }
    }
    postData();

    在这个示例中,我们定义了一个名为 postData 的异步函数。在该函数内部,我们使用 await axios.post 发送 POST 请求到指定的 URL,并传递一个包含数据的 JavaScript 对象作为请求体。

    当发送请求时,JavaScript 会等待 axios 返回响应。一旦收到响应,它会将响应存储在 response 变量中。我们可以使用 response.data 来访问响应数据。

    如果请求成功,我们打印出响应数据。如果请求失败,则会捕获错误并打印错误信息。

    需要注意的是,在使用 await 关键字之前,必须将其包裹在 async 函数内部。这是因为 await 只能在异步函数中使用。

    在使用 await 时,必须将其放在 async 函数内部。这样才能正确地使用 await axios.post 并处理响应。

    axios.post 是一个用于发送 HTTP POST 请求的方法,它返回一个 Promise 对象。这个 Promise 对象在请求完成后会被解析,提供了请求的响应数据和其他相关信息。

    await axios.post 的返回值是一个 Promise,它解析为 axios 的响应对象。响应对象具有许多属性,其中最常用的是 data 属性,它包含服务器返回的数据。具体地说, axios.post 方法返回的 Promise 对象具有以下结构:

    interface AxiosResponse<T = any> {
      data: T;
      status: number;
      statusText: string;
      headers: any;
      config: AxiosRequestConfig;
      request?: any;
    }
    interface AxiosError<T = any> extends Error {
      config: AxiosRequestConfig;
      code?: string;
      request?: any;
      response?: AxiosResponse<T>;
      isAxiosError: boolean;
    }
    interface AxiosPromise<T = any> extends Promise<AxiosResponse<T>> {}
    • AxiosResponse 是成功响应的类型,它包含以下属性:

      • data :响应的数据,类型为泛型 T 所指定的类型。
      • status :HTTP 响应状态码。
      • statusText :HTTP 响应状态文本。
      • headers :响应的 HTTP 头信息。
      • config :发起请求时的配置对象。
      • request :发起的请求实例。
    • AxiosError 是请求错误时的类型,它继承自 Error 对象,包含以下属性:

      • config :发起请求时的配置对象。
      • code :请求错误的错误码。
      • request :发起的请求实例。
      • response :包含错误响应的 AxiosResponse 对象。
      • isAxiosError :标识是否为 Axios 错误。
    • AxiosPromiseaxios.post 方法返回的 Promise 对象的类型,它是一个泛型类型,泛型 T 表示响应数据的类型。

    通过对返回的 Promise 对象调用 .then 方法,您可以注册成功回调函数来处理响应数据。如果请求失败,则可以通过 .catch 方法注册错误回调函数进行错误处理。

    示例代码:

    axios.post(url, data)
      .then((response: AxiosResponse) => {
        // 处理成功响应
        console.log(response.data);
      })
      .catch((error: AxiosError) => {
        // 处理错误
        console.error(error.message);
      });

    请注意, axios.post 返回的 Promise 对象根据请求结果可能会被解析为成功响应或错误响应。因此,在处理响应时,建议使用 .then.catch 方法来适当地处理成功和错误的情况。

    # 网络请求

    # fetch( )

    使用 JavaScript 内置的 Fetch API 进行网络请求:

    1. 构建请求参数:创建一个包含请求方法、请求头、请求体等信息的配置对象。

    2. 发起网络请求:使用 fetch() 函数发送请求,并将配置对象作为参数传递给它。 fetch() 函数返回一个 Promise 对象,代表异步操作的结果。

    3. 处理响应:通过 Promise 的解决来获取服务器返回的响应。可以使用响应对象的方法和属性来读取响应的状态、头部和主体数据。

    4. 解析响应数据:根据服务器返回的数据类型,使用合适的方法将响应的主体数据解析为可用的格式,例如 JSON、文本或二进制数据。

    下面是一个使用 Fetch API 进行 GET 请求的示例:

    fetch('https://api.example.com/data')
      .then(response => {
        if (!response.ok) {
          throw new Error('Network response was not ok');
        }
        return response.json();
      })
      .then(data => {
        console.log(data); // 处理响应数据
      })
      .catch(error => {
        console.error('Error:', error);
      });

    上述示例展示了一个简单的 GET 请求,其中:

    • 使用 fetch() 函数发送了一个 GET 请求到 'https://api.example.com/data'
    • fetch() 函数返回的 Promise 对象通过 .then() 方法进行链式处理。在第一个 .then() 中,我们检查响应的状态码是否为成功状态( ok ),如果不是,则抛出一个错误。
    • 如果响应状态码正常,我们使用 response.json() 方法将响应的主体数据解析为 JSON 格式。
    • 最后一个 .then() 用于处理解析后的数据。
    • 如果在请求过程中发生错误,或任何一个 .then() 中抛出了错误,将进入 .catch() ,在控制台输出错误信息。

    类似地,你可以使用 fetch() 函数进行 POST 请求,只需在配置对象中设置请求方法为 'POST' ,并通过 body 属性提供请求体数据。

    请注意,Fetch API 默认不会将错误状态(如 404 或 500)视为网络错误。它会将这些状态码视为请求成功,并通过 response.ok 属性进行判断。如果你希望将这些状态码也视为网络错误,可以在第一个 .then() 中进行额外的检查或自定义处理逻辑。

    这是使用 JavaScript 内置的 Fetch API 进行网络请求的基本流程。根据具体的需求和场景,你可以进一步使用其他方法和属性来扩展和定制请求和响应的处理。

    Edited on Views times
    \ No newline at end of file diff --git a/cs/languages/rust/index.html b/cs/languages/rust/index.html index d4298418..47158313 100644 --- a/cs/languages/rust/index.html +++ b/cs/languages/rust/index.html @@ -1 +1 @@ -Rust 基础 - 编程语言 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 参考资料

    • Rust 语言圣经 (自称)
      • 锈书
    • 官网

    # 环境配置

    更新 Rust

    bash
    rustup update

    卸载 Rust

    bash
    rustup self uninstall

    查看版本

    h bash
    rustc --version
    ## 显示格式
    rustc.x.y.z([commit hash] [commit date])

    安装 Rust 同时还会安装文档,查看本地文档

    bash
    rustup doc

    # Rust 工具

    # 编译器

    # rustc

    只适合单文件编译

    shell
    rustc xxx.rs <output>

    # cargo

    Cargo 是 Rust 的构建系统和包管理工具,适用于较大型项目编译、链接

    • 构建代码
    • 下载依赖的库
    • 构建这些库

    创建项目,会创建一个新目录 project 并且初始化一个 git 项目

    shell
    cargo new project

    创建好的项目结构

    shell
    .
    ├── .git
    ├── .gitignore
    ├── Cargo.toml
    └── src
        └── main.rs

    检查代码能否通过编译,并不会编译项目,很常用

    shell
    cargo check

    构建项目

    shell
    cargo build           # 默认为调试模式即 --debug
    cargo build --debug   # 同上
    cargo build --release # 发布模式,编译出的程序性能更高

    运行项目

    shell
    cargo run           # 默认为调试模式即 --debug
    cargo run --debug   # 同上
    cargo run --release # 发布模式,编译出的程序性能更高

    # 安装依赖库换源

    目前 cargo search 无法使用镜像

    • 字节镜像源参考
    • 华科镜像源参考

    # 方法一

    $HOME/.cargo/config.toml 文件 (没有则创建一个) 添加如下内容:

    l toml
    # 华科镜像源
    [registries]
    ustc = { index = "https://mirrors.ustc.edu.cn/crates.io-index/" } 
    # 或者
    [registries.ustc]
    index = "https://mirrors.ustc.edu.cn/crates.io-index/"

    或者稀疏索引的方式,要求 cargo >= 1.68 :

    toml
    [source.ustc-sparse]
    registry = "sparse+https://mirrors.ustc.edu.cn/crates.io-index/"

    同时需要在项目 cargo.toml 使用注册方式引入依赖库:

    toml
    [dependencies]
    <要引入的包> = { registry = "ustc" }
    <要引入的包> = { registry = "rsproxy" }

    # 方法二

    不需要对项目 cargo.toml 添加配置,只需要 $HOME/.cargo/config.toml 文件 (没有则创建一个) 添加如下内容:

    toml
    [source.crates-io]
    replace-with = 'rsproxy' # 字节跳动镜像源
    [source.rsproxy]
    registry = "https://rsproxy.cn/crates.io-index" # 镜像源 url

    稀疏索引,要求 cargo >= 1.68 :

    toml
    [source.rsproxy-sparse]
    registry = "sparse+https://rsproxy.cn/index/"
    [registries.rsproxy]
    index = "https://rsproxy.cn/crates.io-index"
    [net]
    git-fetch-with-cli = true

    # 项目结构

    # 项目结构要求

    • Cargo.toml 在项目顶层下
    • 源代码都应该在 src 目录下
    • 顶层目录可以放置:README、许可信息、配置文件和其它与程序源码无关的文件

    如果创建项目时没有使用 cargo,也可以把项目转化为使用 cargo:

    • 把源代码文件移动到 src 下
    • 创建 Cargo.toml 并填写相应的配置

    # Cargo.toml

    eg:

    toml
    [package]
    name = "my-project"
    version = "0.1.0"
    authors = ["fuuzen <fuuzen.github.io>"]
    edition = "2024"
    [dependencies]
    ...

    # [package]

    • name
    • version
    • authors
    • edition

    # [dependencies]

    • crate 包
    • ...

    三种描述:

    • 基于 Rust 官方仓库 crates.io ,通过版本说明来描述
    • 基于项目源代码的 git 仓库地址,通过 URL 来描述
    • 基于本地项目的绝对路径或者相对路径,通过类 Unix 模式的路径来描述

    eg :

    l toml
    [dependencies]
    rand = "0.3"
    hammer = { version = "0.5.0"}
    color = { git = "https://github.com/bjz/color-rs" }
    geometry = { path = "crates/geometry" }

    # 变量和常量

    # 变量

    # let 变量绑定 (binding)

    rust 中 let 表达式的类似其他语言赋值的作用,实际上不是赋值,而叫做 绑定,这与 rust 所有权 这一核心概念有关

    默认绑定变量类型为 immutable 不可变的,需要可变则须加上 mut 显式声明

    rust
    // 使用 let 来声明变量,进行绑定,a 是不可变的
    // 此处没有指定 a 的类型,编译器会默认根据 a 的值为 a 推断类型:i32,有符号 32 位整数
    // 语句的末尾必须以分号结尾
    let a = 10;
    // 主动指定 b 的类型为 i32
    let b: i32 = 20;
    // 这里有两点值得注意:
    // 1. 可以在数值中带上类型:30i32 表示数值是 30,类型是 i32
    // 2. c 是可变的,mut 是 mutable 的缩写
    let mut c = 30i32;
    // 还能在数值和类型中间添加一个下划线,让可读性更好
    let d = 30_i32;
    // 跟其它语言一样,可以使用一个函数的返回值来作为另一个函数的参数
    let e = add(add(a, b), add(c, d));

    # _ 编译器忽略未使用变量

    告诉编译器不要警告未使用的变量,为此可以用下划线作为变量名的开头

    rsut
    let _a = 5

    # let 变量解构

    从一个复杂的变量中匹配出部分内容

    rust
    let (a, mut b): (bool,bool) = (true, false);
    //a = true, 不可变;b = false,可变
    println!("a = {:?}, b = {:?}", a, b);
    b = true;
    assert_eq!(a, b);

    # 解构式赋值

    Rust 1.59 版本后可以在赋值语句的左式中使用元组、切片和结构体模式,不需要 let

    struct Struct {
        e: i32
    }
    fn main() {
        let (a, b, c, d, e);
        (a, b) = (1, 2);
        //_ 代表匹配一个值,但是我们不关心具体的值是什么,因此没有使用一个变量名而是使用了 _
        [c, .., d, _] = [1, 2, 3, 4, 5];
        Struct { e, .. } = Struct { e: 5 };
        assert_eq!([1, 2, 1, 4, 5], [a, b, c, d, e]);
    }

    这种使用方式跟之前的 let 保持了一致性,但是 let 会重新绑定,而这里仅仅是对之前绑定的变量进行再赋值。

    使用 += 的赋值语句还不支持解构式赋值

    # 变量遮蔽 (shadowing)

    Rust 允许声明相同的变量名,在后面声明的变量会遮蔽掉前面声明的。

    变量遮蔽的用处在于,如果你在某个作用域内无需再使用之前的变量(在被遮蔽后,无法再访问到之前的同名变量),就可以重复的使用变量名字,而不用绞尽脑汁去想更多的名字。

    • 可以使用相同的名字声明新的变量,新的变量就会 shadow(隐藏)之前声明的同名变量

    • 在后续的代码中这个变量名代表的就是新的变量

    shadow 和把变量标记为 mut 是不一样的:

    • 如果不使用 let 关键字,那么重新给非 mut 的变量赋值会导致编译时错误
    • 而使用 let 声明的同名新变量,也是不可变的
    • 使用 let 声明的同名新变量,它的类型可以与之前不同

    # 常量

    常量也是绑定到一个常量名且不允许更改的值

    # 常量和变量之间的差异

    • 常量不允许使用 mut常量不仅仅默认不可变,而且自始至终不可变,因为常量在编译完成后,已经确定它的值。
    • 常量使用 const 关键字而不是 let 关键字来声明,并且值的类型必须标注

    # Rust 常量的命名约定

    • 全部字母都使用大写
    • 使用下划线分隔单词,另外对数字字面量可插入下划线以提高可读性

    # 基本类型 (标量)

    # 原始类型

    # 整数类型

    类型定义的形式统一为: 有无符号 + 类型大小(位数)

    长度有符号类型无符号类型
    8 位i8u8
    16 位i16u16
    32 位i32u32
    64 位i64u64
    128 位i128u128
    视架构而定isizeusize

    # isizeusize 类型

    isizeusize 类型的位数由程序运行的计算机的架构所决定:

    • 如果是 64 位计算机,那就是 64 位的
    • 如果是 32 位计算机,那就是 32 位的
    • ...

    # 整数字面值

    rust 有以下 5 种整数字面类型:

    Number literalsExample
    Decimal98_222
    Hex0xff
    Octal0o77
    Binary0b1111_0000
    Byte (u8 only)b'A'
    • 除了 Byte 类型外,所有的数值字面值都允许使用类型后缀。

      • 例如 57u8
    • 如果不太清楚应该使用那种类型,可以使用 Rust 相应的默认类型

      • 整数的默认类型就是 i32 ,其速度整体上是较快的,即使是在 64 位操作系统中

    # 整数溢出

    • 调试模式 (--debug) 下编译:Rust 会检查整数溢出,如果发生溢出,程序在运行时就会 panic

    • 发布模式下(--release)编译:Rust 不会检查可能导致 panic 的整数溢出

      • 如果溢出发生:Rust 会执行 “环绕” 操作:(取模)

        • 例如: u8 的范围是 0-255,如果把一个 u8 变量的值设为 256,那么 Rust 会将 256 变成 0,257 变成 1 ⋯
      • 但程序不会 panic

    # 浮点类型

    Rust 有两种基础的浮点类型,也就是含有小数部分的类型:

    • f32 ,32 位,单精度
    • f64 ,64 位,双精度

    Rust 的浮点类型使用了 IEEE-754 标准来表述

    f64默认类型,因为在现代 CPU 上 f64f32 的速度差不多,而且精度更高。

    # 布尔类型

    • 符号是 bool
    • Rust 的布尔类型也有两个值:
      • true
      • false
    • 占一个字节大小

    Rust 不允许其他类型直接当作布尔类型使用,和其他语言不太一样!

    # 字符类型

    • 符号是 char
    • Rust 语言中 char 类型被用来描述语言中最基础的单个字符。
    • 字符类型的字面值使用单引号
    • 占用 4 字节大小
    • 是 Unicode 标量值,可以表示比 ASCII 多得多的字符内容,包括:拼音、中日韩文、零长度空白字符、emoji 表情等。

    Unicode 中并没有 “字符” 的概念,所以直觉上认为的字符也许与 Rust 中的概念并不相符

    # 基本类型转换

    # 隐式

    1. 整数类型之间的转换:Rust 允许在整数类型之间进行隐式转换,只要目标类型的表示范围足够容纳源类型的值。例如,可以将 u8 转换为 u16i32 转换为 u64 等。
    2. 浮点数类型之间的转换:Rust 允许在浮点数类型之间进行隐式转换,只要目标类型能够表示源类型的精度和范围。例如,可以将 f32 转换为 f64
    3. 整数类型到浮点数类型的转换:Rust 允许将整数类型隐式转换为相应的浮点数类型。例如,可以将 i32 转换为 f64
    4. 字面量的隐式转换:在字面量的使用中,Rust 会根据上下文自动进行隐式类型转换。例如,可以将整数字面量 42 隐式转换为 u8i32 或其他整数类型,或将浮点数字面量 3.14 隐式转换为 f32f64

    # 显式

    Rust 鼓励显式的类型转换,并提供了一些安全的类型转换方法,以确保程序在类型转换时保持安全和可预测。Rust 有三种变量类型显式转换的方法:

    • as 关键字
    • frominto 这些 trait 进行转换
    • 字符串类型可以使用 parse 方法转换到数字类型

    # as & as_<T>()

    as 运算符转换示例:

    t rust
    let x: i32 = 5;
    let y: u32 = x as u32;

    as 运算符还有一个安全版本 as_<T>() ,它会在溢出时返回一个 Option 值。

    eg:

    rust
    fn main() {
        let num: u32 = 42;
        
        // 将 u32 类型转换为 i16 类型
        let converted: Option<i16> = num.as_();
        
        match converted {
            Some(value) => println!("Converted value: {}", value),
            None => println!("Conversion failed"),
        }
    }

    # From & Into

    在 Rust 中, FromInto 是用于自定义类型转换的 trait。通过实现这些 trait,可以定义自定义类型之间的转换规则。

    From trait 定义了一个类型转换的关联函数 from() ,用于将其他类型转换为当前类型。

    Into trait 则定义了一个类型转换的方法 into() ,用于将当前类型转换为其他类型。

    eg:

    rust
    // 定义一个自定义类型
    struct MyType(u32);
    // 实现 From trait,将 u32 类型转换为 MyType
    impl From<u32> for MyType {
        fn from(value: u32) -> Self {
            MyType(value)
        }
    }
    // 实现 Into trait,将 MyType 转换为 u32 类型
    impl Into<u32> for MyType {
        fn into(self) -> u32 {
            self.0
        }
    }
    fn main() {
        let num: u32 = 42;
        
        // 使用 From trait 进行转换
        let my_type: MyType = MyType::from(num);
        
        // 使用 Into trait 进行转换
        let back_to_num: u32 = my_type.into();
        println!("MyType: {:?}", my_type);
        println!("Back to u32: {}", back_to_num);
    }

    # parse

    使用 parse 方法将字符串类型转换为数字类型,如整数或浮点数。 parse 方法是定义在 str 类型上的一个函数,它返回一个 Result 枚举类型,表示解析操作的结果。

    eg:

    rust
    fn main() {
        let num_str = "42";
        
        // 将字符串转换为整数类型
        let num: Result<i32, _> = num_str.parse();
        
        match num {
            Ok(n) => println!("Parsed number: {}", n),
            Err(e) => println!("Error: {}", e),
        }
    }

    # 复合类型

    • 复合类型可以将多个值放在一个类型里。
    • Rust 提供了两种基础的复合类型:元组(Tuple)、数组

    # Tuple

    • Tuple 可以将多个类型的多个值放在一个类型里
    • Tuple 的长度是固定的:一旦声明就无法改变
    • Tuple 中的每个位置都对应一个类型, Tuple 中各元素的类型不必相同

    # 创建 Tuple

    在小括号里,将值用逗号分开

    rust
    let tup: (132, f64, 48) = (500, 6.4, 1);

    # 获取 Tuple 的元素值

    可以使用模式匹配来解构(destructure)一个 Tuple 来取元素的值

    rust
    let (x, y, z) = tup;

    # 访问 Tuple 的元素

    在 tuple 变量使用点标记法,后接元素的索引号

    rust
    println!("{} {} {}", tup.O, tup.1, tup.2);

    # 数组

    • 数组也可以将多个值放在一个类型里
    • 数组中每个元素的类型必须相同
    • 数组的长度也是固定的

    数组没有 Vector 灵活,如果想让你的数据存放在 stack(栈)上而不是 heap(堆)上,或者想保证有固定数量的元素,这时使用数组更有好处

    # 数组的类型

    数组的类型以这种形式表示: [类型;长度]

    # 声明一个数组

    两种方法:

    • 在中括号里,各值用逗号分开
    • 如果数组的每个元素值都相同,那么可以定义为: [初始值;长度]
    rust
    let a = [1, 2, 3, 4, 51];
    let a =3;5]
    // 它就相当于:let a= [3, 3, 3, 3, 3];

    # 访问数组的元素

    可以使用索引来访问数组的元素。

    如果访问的索引超出了数组的范围,那么:

    • 索引是字面量等,编译器能检查出来,编译不通过
    • 编译器没能检查出来,编译通过
      • 运行会报错(runtime 时会 panic)
      • Rust 不会允许其继续访问相应地址的内存

    # 枚举类型

    enum 关键字表示枚举类型

    # 定义枚举

    例如 IP 地址:IPV4、IPv6

    rust
    enum IpAddrKind {
      V4,
      V6,
    }

    标准库中的 IpAddr

    rust
    struct Ipv4Addr {
      // --snip--
    struct Ipv6Addr {
      // --snip--
    }
    enum IpAddr {
      V4(Ipv4Addr),
      V6(Ipv6Addr),
    }

    # 变体、标识符、枚举值

    • 变体是枚举类型的不同选项或可能的值之一
    • 标识符是用于表示枚举类型的变体的名称
    • 枚举值是通过选择枚举类型的一个变体来实例化的具体值

    枚举的变体都位于标识符的命名空间下,使用两个冒号 :: 进行分隔

    rust
    let four = IpAddrKind::V4;
    let six = IpAddrKind::V6;

    # 将数据附加到枚举的变体中

    rust
    enum IpAddr {
      V4(String),
      V6(String),
    }
    // 每个变体可以拥有不同的类型以及关联的数据量
    enum IpAddr {
      V4(u8, u8, u8, u8),
      V6(String),
    }
    • 优点:
      • 不需要额外使用 struct
      • 每个变体可以拥有不同的类型以及关联的数据量

    # 枚举方法

    struct 类似,也使用 impl 关键字

    # Option 枚举

    • 定义于标准库中
    • 在 Prelude(预导入模块)中
    • 描述了:某个值可能存在(某种类型)或不存在的情况
    # Option<T>

    Rust 没有 Null。其它语言中,Null 是一个值,它表示 “没有值”,一个变量可以处于两种状态:空值(nuI)、非空。

    Null 引用: Billion Dollar Mistake

    Null 的问题在于:当你尝试像使用非 Null 值那样使用 NuIl 值的时候,就会引起某种错误

    Null 的概念还是有用的:因某种原因而变无效或缺失的值

    Rust 中类似 Null 概念的枚举 Option<T> ,定义如下:

    rust
    enum Option<T>{
      Some(T),
      None,
    }

    T 是一个类型参数,代表值的类型。这里出现三个类型都定义在标准库中:

    • Option<T>
    • Some (T)Some 表示存在一个值,并将该值包装在 Some
    • None

    Option<T>T 是不同的类型,不可以把 Option<T> 直接当成 T ,例如以下代码会报错:

    rust
    let x:i8 =5;
    let y: Option<i8> = Some(5);
    let sum = x + y; // 编译报错:error [E0277]: cannot add `Option<i8>` to `i8`

    若想使用 Option<T> 中的 T ,必须将它转换为 T

    # 基础控制流

    # 选择分支

    # if else

    rust
    if ... {
      ...
    } else if ... {
      ...
    } else {
      ...
    }

    如下代码在 rust 编译器环境下会产生报错:

    rust
    let number = if condition { 5 } else { "6" }

    因为返回给 number 的结果类型不确定!

    # match

    允许一个值与一系列模式进行匹配,并执行匹配的模式对应的代码,模式可以是字面值、变量名、通配符...

    match 匹配必须穷举所有的可能,否则会产生报错:

    bash
    error [E0004]: non-exhaustive patterns:...
    • _ 通配符:替代其余没列出的值
    rust
    match ... {
      a => (...),
      b => (...),
      _ => (...),
    }

    # if let

    语法的一般形式如下:

    rust
    if let pattern = expression {
      // 匹配成功的处理逻辑
    } else if let pattern = expression {
      // 匹配成功的处理逻辑
    } else {
      // 匹配失败的处理逻辑
    }

    在上述语法中, pattern 是要匹配的模式, expression 是要进行匹配的表达式。如果表达式与模式匹配成功,则执行匹配成功的处理逻辑。如果匹配失败,则执行匹配失败的处理逻辑。

    下面是一个示例,展示了如何使用 if let 处理 Option 类型:

    rust
    fn main() {
        let some_value: Option<i32> = Some(5);
        if let Some(x) = some_value {
            println!("Got value: {}", x);
        } else {
            println!("Value is None");
        }
    }

    在上述示例中, some_value 是一个 Option<i32> 类型的变量,被赋值为 Some(5) 。通过使用 if let Some(x) = some_value ,我们尝试将 some_value 解构为 Some 并提取其中的值。如果匹配成功,将会执行 println!("Got value: {}", x) ,打印提取到的值。如果匹配失败(即 some_valueNone ),则执行 println!("Value is None")

    使用 if let 处理只关心一种匹配而忽略其它匹配的情况,可以使代码更简洁、清晰地处理特定模式的匹配,减少了手动编写 match 表达式的工作量,

    可以把 iflet 看作是 match 的语法糖

    # 循环

    # loop

    • 没有条件判断
    • 使用 break 跳出循环
    • loop 有返回值,为 break 后面的表达式
    rust
    let result = loop {
      ...
      if ... {
        break (expression) ;
      }
    }

    # while

    rust
    while ... {
      ...
    }

    和其他语言差别不大

    # for

    loopwhile 更适合遍历容器,更安全、高效

    rust
    // eg:
    let a = [10, 20, 30, 40, 50];
    for element in a.iter() {
      println! ("the value is: {}", element);
    }

    # 所有权 Ownership

    # 什么是所有权

    Rust 的核心特性就是所有权

    所有程序在运行时都必须管理它们使用的计算机内存

    • 有些语言有垃圾收集机制 (GC),在程序运行时,它们会不断地寻找不再使用的内存
    • 还有一些语言,程序员必须显式地分配和释放内存 (eg: C)
    • Rust 采用了第三种方式:
      • 内存是通过一个所有权系统来管理的,其中包含一组编译器在编译时检查的规则
      • 当程序运行时,所有权特性不会减慢程序的运行速度,因为 Rust 的所有权系统将将内存的管理提前到了编译阶段

    在像 Rust 这样的系统级编程语言里,一个值是在 stack 上还是在 heap 上对语言的行和你为什么要做某些决定是有更大的影响的。管理 heap 数据就是所有权存在的原因。

    # Stack vs Heap

    在代码运行的时候,Stack 和 Heap 都是你可用的内存,但他们的结构很不相同:

    • Stack

      • Stack 按值的接收顺序来存储,按相反的顺序将它们移除(后进先出,LIFO)

        • 添加数据叫做压入栈 push
        • 移除数据叫做弹出栈 pop
      • 所有存储在 Stack 上的数据必须拥有已知的固定的大小

        • 因为指针是已知固定大小的,可以把指针存放在 stack 上。但如果想要实际数据,你必须使用指针来定位,然后访问 heap
      • 把数据压到 stack 上要比在 heap 上分配快得多,因为操作系统不需要寻找用来存储新数据的空间,那个位置永远都在 stack 的顶端

      • 如果数据存放的距离比较近,那么处理器的处理速度就会更快一些

    • Heap

      • 当你把数据放入 heap 时,你会请求一定数量的空间,这个过程叫做在 heap 上进行分配,有时仅仅称 “分配”。在 heap 上分配空间需要做更多的工作

        • 操作系统在 heap 里找到一块足够大的空间
        • 然后要做好记录方便下次分配,把它标记为在用
        • 并返回一个指针,也就是这个空间的地址
      • 编译时大小未知的数据或运行时大小可能发生变化的数据必须存放在 heap 上

      • 如果数据之间的距离比较远,那么处理速度就会慢一些

    所有权解决的问题:

    • 跟踪代码的哪些部分正在使用 heap 的哪些数据
    • 最小化 heap 上的重复数据量
    • 清理 heap 上未使用的数据以避免空间不足。

    掌握了所有权,那么就不需要经常去想 stack 或 heap 了

    # 所有权规则

    # 三条规则

    • 每个值都有一个变量,这个变量是该值的所有者
    • 每个值同时只能有一个所有者
    • 当所有者超出作用域(scope)时
      • 它拥有它的值的所有权,就会被 drop 函数清理 (heap 上的数据)
      • 它没有它的值的所有权,不会清理

    # 返回值与作用域

    函数在返回值的过程中同样也会发生所有权的转移,返回表达式的值的所有权将转移到函数结果所赋值的变量上

    # 复制 Copy trait

    用于像整数这样完全存放在 stack 上面不包含 heap 上的简单数据类型(标量)

    • 如果一个类型实现了 Copy 这个 trait ,那么旧的变量在赋值后仍然可用
    • 如果一个类型或者该类型的一部分实现了 Drop trait ,那么 Rust 不允许让它再去实现 Copy trait
    rust
    let x = 5;
    let y = x;
    //x 和 y 都是有效的
    println! ("{} {}", x, y);

    # 移动 Move

    多个变量可以与同一个数据使用一种独特的方式 Move 来交互

    String 类型为例

    rust
    let s1 = String::from ("hello"); 
    let s2 = s1;

    当把 s1 赋给 s2String 在 stack 上的数据被复制了一份,heap 内容不变

    string-move

    s1s2 离开作用域时,它们都会尝试释放相同的内存,就会导致二次释放 (double free) bug 。为了避免这个问题,Rust 会让原来的 s1 失效 ,此后当 s1・离开作用域的时候,Rust 不需要释放任何东西。

    s1 此后处于 moved 状态,即失效。

    • 浅拷贝(shallow copy)
      • 只拷贝 stack 上的数据,有可能产生二次释放 (double free) bug
    • 深拷贝(deep copy)
      • 同时拷贝 stack 和 heap 上的数据

    Rust 中采用的是第三种 “拷贝”,也就是 ** 移动 Move **,在浅拷贝的基础上让原来的 stack 数据失效。

    这体现了 Rust 隐含的一个设计原则:Rust 不会自动创建数据的深拷贝。就运行时性能而言,任何自动赋值的操作都是廉价的。

    # 克隆 CLone

    针对包含 heap 数据的矢量数据类型,相当于深拷贝

    rust
    let s1 = String::from("Hello");
    let s2 = s1.clone();

    string-clone

    # 引用 Reference

    & 符号就表示引用,允许引用该值而不取得其所有权;

    对应的解引用符号是 *

    和 C/C++ 一样,Rust 引用是符号是 & ,解引用符号是 *

    string-reference

    Rust 中的引用在任何给定的时刻,引用必须一直有效 (非悬空引用),只能满足下列条件之一:

    • 一个可变的引用
    • 任意数量不可变的引用

    # 可变引用 Mutable Reference

    Rust 支持可变引用 ,用 &mut 符号代替 & 即为可变引用。

    但是可变引用有以下几个限制:

    • 在特定作用域内,对于某一块数据,只能有一个可变引用
    • 不可以同时拥有可变引用和不可变引用
    • (多个不变的引用是可以的)

    第一个限制的好处是可在编译时防止数据竞争。违反将出现如下报错:

    bash
    error[E0499]: cannot borrow 'xxx' as mutable more than once at a time

    第二个限制避免了可变引用对不可变引用的影响。违反将出现如下报错:

    bash
    error[E0502]: cannot borrow 'xxx' as mutable because it is also borrowed as immutable

    ::: infromation

    以下三种行为下会发生数据竞争:

    • 两个或多个指针同时访问同一个数据
    • 至少有一个指针用于写入数据
    • 没有使用任何机制来同步对数据的访问

    数据竞争在运行时是很难发现的(C/C++ 里经常出现),所以 Rust 从根本上避免了这种问题来保证运行时的安全性

    :::

    可以通过创建新的作用域,来允许非同时的创建多个可变引用

    rust
    fn main () {
      let mut s = String:: from "Hello");
    	{
    	  let s1 = &mut s;
    	}
      let s2 = &mut s;
    }

    # 悬空引用 Dangling Reference

    也叫悬垂引用,在 Rust 里,编译器可保证引用永远都不是悬空引用。如果你引用了某些数据,编译器将保证在引用离开作用域之前数据不会离开作用域。对于任何出现的悬空引用, Rust 在编译阶段就会报错:

    bash
    error[E0106]: missing lifetime specifier

    悬空指针(Dangling Pointer):一个指针引用了内存中的某个地址,而这块内存可能已经释放并分配给其它人使用了。

    # 借用 Borrow

    引用作为函数参数这个行为叫做借用 ,本质上就是引用。

    rust
    fn main () {
      let mut s1 = String::from("Hello");
      let len = calculate_length(&mut s1);
      println!("The length of '{}' is {}.", s1, len);
    }
    fn calculate_length(s: &mut String) -> usize {
      s.push_str(", world");
      s.len()
    }

    # 切片类型 slice

    Rust 的另外一种不持有所有权的数据类型

    # 字符串切片

    字符串切片是指向字符串中一部分内容的引用

    形式: [开始索引..结束索引]

    • 开始索引就是切片起始位置的索引值,若不写则默认为 0
    • 结束索引是切片终止位置的下一个索引值,若不写则默认为 String 的 Capacity
    rust
    fn main(){
      let s = String::from("Hello world");
      let hello = &s[0..5]; // 左闭右开
      let world = &s[6..11];// 左闭右开
    }

    string-slice

    字符串切片的范围索引必须发生在有效的 UTF-8 字符边界内

    如果尝试从一个多字节的字符中创建字符串切片,程序会报错并退出

    字符串字面值实际上就是字符串切片 String slice

    rust
    //s 的类型是 &str
    let s = "Hello, World!"

    ⭐️ 将字符串切片作参数传递

    rust
    fn first_words:(&String) -> &str {}

    对于这样接受字符串作为参数的函数,有经验的 Rust 开发者会采用 &str 作为参数类型,因这样就可以同时接收 String
    和 &str 类型的参数了:

    rust
    fn first_words:(&str) -> &str {}
    • 使用字符串切片,直接作为参数调用该函数 first_words(mystr)
    • 使用 String,可以创建一个完整的 String 切片作为参数来调用该函数 firstwords(&mystr[..])

    定义函数时使用字符串切片来代替字符串引用会使我们的 API 更加通用,且不会损失任何功能。

    # 数组切片

    rust
    let a = [1, 2, 3, 4, 5];
    let slice = &a[1..3];

    # 矢量变量

    # String 类型

    字符串字面值是程序里手写的那些字符串值,它们是不可变的、硬编码的。

    Rust 还有第二种字符串类型: Sting

    String 比那些基础标量数据类型更复杂,在 heap 上分配。能够存储在编译时未知数量的文本。

    # 数据结构

    一个 String 由 stack 上的 3 部分组成:

    • 一个指向存放字符串内容的内存的指针
    • 长度 len,就是存放字符串内容所需的字节数
    • 容量 capacity ,是 String 从操作系统总共获得内存的总字节数

    存放字符串内容的部分在 heap 上

    string"hello"

    # 创建 String 类型的值

    可以使用 from 函数从字符串字面值创建出 String 类型

    rust
    let s = String::from("hello");

    :: 表示 fromString 类型下的函数

    这类字符串是可以被修改的

    # Vector 类型

    # struct

    # struct 基础操作

    # 定义 struct

    • 使用 struct 关键字,并內整个 struct 命名
    • 在花括号内,为所有字段(Field)定义名称和类型
    rust
    struct User {
      username: String,
      email: String,
      sign_in_count: u64,
      active: bool,
    }

    # 实例化 struct

    想要使用 struct ,需要创建 struct 的实例:

    • 每个字段指定具体值
    • 无需按声明的顺序进行指定
    rust
    let user1 = User {
      email: String::from("someone@example.com"),
      username: String::from("someusername123"),
      active: true,
      sign_in_count: 1,
    }

    一旦 struct 的实例是可变的,那么实例中所有的字段都是可变的

    # 取得 struct 里面的某个值

    使用点标记法:

    rust
    user1.email = String::from("anotheremail@example.com");

    # 字段初始化简写

    字段名与字段值对应变量名相同时,就可以使用字段初始化简写的方式:

    rust
    fn build_user(email: String, username: String) -> User {
      User {
        email,
        username,
        active: true,
        sign_in_count: 1,
      }
    }

    # struct 更新语法

    当你想基于某个 struct 实例来创建一个新实例的时候,可以使用 struct 更新语法:

    rust
    // 不使用 struct 更新语法
    let user2 = User {
      email: String::from("another@example.com"),
      username: String::from("anotherusername567"),
      active: user1.active,
      sign_in_count: user1.sign_in_count,
    }
    // 使用 struct 更新语法
    let user2 = User {
      email: String::from("another@example.com"),
      username: String: from ("anotherusername567"),
      ..user1
    };

    # debug 阶段打印 struct

    rust
    #[derive(Debug)] // 添加注解,表示派生自 Debug 这个 trait
    ...
    println!("{:?}", mystruct);// 输出为一行,不够清晰直观
    println!("{:#?}", mystruct);// 输出结构化信息,更清晰直观

    # struct 数据的所有权

    rust
    struct User {
      username: String,
      email: String,
      sign_in_count: u64,
      active: bool,
    }

    这里的字段使用了 String 而不是 &str:

    • struct 实例拥有其所有的数据
    • 只要 struct 实例是有效的,那么里面的字段数据也是有效的
    • struct 里也可以存放引用,但这需要使用生命周期

    # struct 的方法

    Rust 是一个多范式的语言,也支持面向对象编程

    使用 impl 关键字:

    rust
    impl mystruct {
      fn ... (&self, ...){
        
      }
      ...
    }

    Rust 方法与函数不同之处:

    • 方法是在 struct (或 enumtrait 对象)的上下文中定义
    • 第一个参数是 self ,表示方法被调用的 struct 实例

    # struct 扩展定义

    # tuple struct

    Rust 可定义类似 tuplestruct ,叫做 tuple struct

    Tuple struct 整体有个名字,但里面的元素没有名字

    适用于想给整个 tuple 起名,并让它不同于其它 tuple ,而且又不需要给每个元素起名

    # 定义 tuple struct

    使用 struct 关键字,后边是名字,以及里面元素的类型

    rust
    struct Color(i32, i32, i32);
    struct Point(i32, i32, i32);
    let black = Color(0, 0, 0);
    let origin = Point(0, 0, 0);
    //black 和 origin 是不同的类型,是不同 tuple struct 的实例

    # Unit-Like Struct

    可以定义没有任何字段的 struct,叫做 Unit-Like struct(因为与空元组 () 也就是单元类型类似)

    适用于需要在某个类型上实现某个 trait,但是在里面又没有想要存储的数据

    # Package, Crate, Module

    Rust 的模块系统:

    • Package(包):Cargo 的特性,让你构建、测试、共享 crate
    • Crate(单元包):一个模块树,它可产生一个 library 或可执行文件
    • Module(模块): use :让你控制代码的组织、作用域、私有路径
    • Path(路径):为 structfunctionmodule 等项命名的方式

    # Package

    一个 Package 包含:

    • 1 个 Cargo.toml ,它描述了如何构建这些 Crates
    • 必须至少包含一个 crate (library 或 binary)
      • 只能包含 0-1 个 library crate
      • 可以包含任意数量的 binary crate

    # Package 项目结构

    这些是 Cargo 的惯例

    • 一个 Package 包含一个 binary crate src/main.rssrc/lib.rs

      • src/main.rs 即为唯一的 binary crate
        • binary crate 的 crate root
    • 一个 Package 可以包含多个 binary crate

      • src/bin 下每个文件是单独的 binary crate
    • 一个 Package 包含 0 或 1 个 library crate

      • src/lib.rs 存在则包含 1 个 library crate ,否则不包含
      • library crate 的 crate root

    crate 名与 package 名 ( name ) 相同

    Cargo 把 crate root 文件交给 rustc 来构建 library 或 binary

    # Crate

    Crate 的作用:将相关功能组合到一个作用域内,便于在项目间进行共享,防止冲突

    Crate 的类型有两类:

    • binary
    • library

    Crate Root 是源代码文件,Rust 编译器从这里开始,组成 Crate 的根 Module

    src/main.rssrc/lib.rs 叫做 crate roots

    # Module

    其作用在于:

    • 在一个 crate 内,将代码进行分组
    • 增加可读性,易于复用
    • 控制项目(item)的私有性(public、private)

    # 建立 module

    使用 mod 关键字

    • 可嵌套
    • 可包含其它项( structenum 、常量、 trait 、函数等)的定义

    # 私有边界(privacy boundary)

    模块不仅可以组织代码,还可以定义私有边界。如果想把 函数 或 struct 等设为私有,可以将它放到某个模块中。只要是在 mod 里,Rust 中所有的条目(函数,方法, structenum ,模块,常量)默认是私有的。

    :::

    这里的条目包括:函数,方法, structenum ,模块,常量等所有可以被引用、调用的东西

    :::

    • 父级模块无法访问子模块中的私有条目
    • 子模块里可以使用所有祖先模块中的条目
    • 文件的根级无论是私有还是公共的都可以互相调用

    # pub 关键字

    使用 pub 关键字来将某些条目标记为公共的

    rust
    mod front_of_house {
      pub mod hosting {
        pub fn add_to_waitlist() {}
      }
    }
    pub fn eat_at_restaurant() {
      crate::front_of_house::hosting::add_to_waitlist(); // 绝对路径调用
      front_of_house::hosting::add_to_waitlist(); // 相对路径调用
    }

    pubstruct 使用, struct 内所有成员仍然默认是私有的!

    pubenum 使用, enum 内所有成员就都是公共的!

    # Path

    为了在 Rust 的模块中找到某个条目,需要使用路径。路径有两种形式:

    • 绝对路径:从 crate root 开始,使用 crate 名 或字面值 crate
    • 相对路径:从当前模块开始,使用 selfsuper 或当前模块的标识符

    路径至少由一个标识符组成,标识符之间使用 ::

    # super 关键字

    类似文件系统中的 .. 表示上一级目录, super 表示用来访问父级模块路径中的内容。

    rust
    fn serve_order() {}
    mod back_of__house {
      fn fix_incorrect_order {
        cook_order();
        super::serve_order();
      }
      fn cook_order() {}
    }

    # use 关键字

    use 关键字将路径导入到作用域内,仍遵循私有性原则,即只有公共部分才能使用

    rust
    mod front_of_house {
      pub mod hosting {
        pub fn add_to_waitlist() {}
      }
    }
    // 绝对路径:
    use crate::front_of_house::hosting; 
    // 相对路径:
    use front_of_house::hosting; 
    // 默认导入到当前文件 (crate) 作用域内
    // 此后 hosting 等效于如下效果:
    mod hosting() {}

    Rust 惯用做法:并不直接 use 导入函数本身,而是导入函数的父模块,通过父模块调用函数,以此来表示这个函数不是本地定义的,而是从其他地方导入的。

    但对于 struct , enum 则是直接导入它们本身,除非出现同名条目,则只好导入它们的父级模块来区分它们,或者用 as 关键字为他们取别名。

    # pub use 重导出

    use 导入的是在本文件作用域内的私有的模块

    该用 pub use 关键字则外部代码导入本 文件时也能使用这些模块

    # 外部包和标准库

    外部包以及标准库 ( std ) 的使用需要:

    1. Cargo.tomldependencies 添加依赖的包;标准库 ( std ) 也被当做外部包但不需要这一步
    2. use 将特定条目引入作用域,外部包和标准库都需要。

    # 嵌套路径

    如果使用同一个包或模块下的多个条目,可以使用嵌套路径清理大量的 use 语句

    rust
    use 路径相同的部分::{self, 到包1路径差异的部分, 到包2 路径差异的部分, ...}
    // 当需要导入路径相同部分所指向的包本身时可以用 self

    eg :

    rust
    use std::io;
    use std::io::Write;
    // 上面代码等效于下面
    use std::io::{self, Write}

    # 通配符 *

    使用 * 可以把路径中所有的公共条目都引入到作用域。 但注意要谨慎使用。

    应用场景:

    • 测试,将所有被测试代码引入到 tests 模块
    • 有时被用于预导入(prelude)模块

    # 将模块内容移动到其它文件

    模块定义时,如果模块名后边是 ; ,而不是代码块:

    • Rust 会从与模块同名的文件中加载内容
    • 模块树的结构不会变化

    这里我详细解释我讲不清楚,也懒得写了🤯

    # 常用集合

    # Vec<T>

    # 创建

    Vec::new() 函数

    vec!

    # 更新

    push 方法,注意是可变引用

    # 删除

    与任何其它 struct v 一样,当 Vec 离开作用域后

    • 它就被清理掉了
    • 它所有的元素也被清理掉了

    # 读取

    下标索引

    get 方法 ,返回 Option<T> 类型,与 match 搭配

    # 遍历

    rust
    for element in my_vector {
      ...element...
    }
    for element in &my_vector {
      ...*element...
    }
    for element in &mut my_vector {
      ...*element...
    }

    # String

    Rust 开发者经常会被字符串困扰的原因

    • Rust 倾向于暴露可能的错误
    • 字符串数据结构复杂
    • UTF-8

    Rust 选择将正确处理 String 数据作所有 Rust 程序的默认行为

    • 缺点:程序员必须在处理 UTF-8 数据之前投入更多的精力

    • 优点:可防止在开发后期处理涉及非 ASCII 字符的错误

    # 区分字符串和 String

    字符串是什么

    • Rust 的核心语言层面,只有一个字符串类型:字符串切片 str(或 &str)
    • 字符串切片:对存储在其它地方、UTF-8 编码的字符串的引用
      • 字符串字面值:存储在二进制文件中,也是字符串切片

    String 类型是什么

    • 来自 标准库 而不是 核心语言
    • 可增长、可修改、可拥有

    二者都采用 UTF-8 編码。

    Rust 的标准库还包含了很多其它的字符串类型,例如: OsStringOsStrGStrngCStr ... 它们可能使用了不同的编码。通常 String 结尾的是可获得所有权的; Str 结尾的是可借用的。

    String 的本质是对 Vec<u8> 的包装,很多 Vec<T> 的操作都可用于 String

    # 创建

    String::new() 函数

    to_string() 方法,可用于实现了 Display trait 的类型,包括字符串字面值

    String::from() 函数

    # 更新

    push_str( xxx : &str ) 方法,把一个字符串切片附加到 String 后面,注意是可变引用,传入的参数是不获取所有权的借用

    push( x : char ) 方法,把单个字符附加到 String 后面,注意是可变引用,传入的参数是不获取所有权的借用

    + 运算符,连接两个字符串 ,注意必须满足: String = String + &strString = String + &String会取得 + 左边字符串的所有权

    这里 + 运算符使用了类似这个签名的方法 fn add(self, s:&str)-> String{..}

    标准库中的 add 方法使用了泛型 ;本来只能 String = String + &str ,但这里出现了解引用强制转換(deref coercion),从而可以 String = String + &String

    format! 宏,连接多个字符串,不会取得任何参数的所有权

    rust
    let s3 = s1 + "-" + &s2 + "_" + &s3;
    // 等效于
    let s3 = format!("{}-{}-{}", s1, s2, s3);

    # 读取

    Rust 的字符串不支持索引语法访问,原因在于采用的是 UTF-8 编码,很多语言的字符并不能只通过一个字节表示,这些字符专业术语叫 Unicode 标量值,例如汉字需要 3 个字节。

    Rust 的字符串不支持索引语法访问的另外一个原因在于索引操作应消耗一个常量时间 O(1) ,而 String 无法保证:需要遍历所有内容,来确定有多少个合法的字符。

    len() 方法,返回字符串长度

    # 删除

    [..] 切片,由于上述 UTF-8 编码的原因,虽然编译通过,但对于很多语言的字符,如果切割时跨越了字符边界就会 出现 panic

    # 遍历

    Rust 有三种看待字符串的方式:

    • Bytes, 字节,可以用 bytes 方法返回迭代器遍历
    • Scalar Values, 标量值,可以用 chars 方法返回迭代器遍历
    • Grapheme Clusters, 字形簇(最接近所谓的 “字母”),比较复杂,标准库未提供

    # HashMap<K,V>

    HashMap 用的较少,不在 Prelude 中,所以要使用它就需要 use std::collections::HashMap ; 标准库对其支持较少,没有内置的宏来创建 HashMap

    注意 HashMap 数据存储在 heap 上。

    HashMap 时同构的,即一个 HashMap 中:

    • 所有的 K 必须是同一种类型
    • 所有的 V 必须是同一种类型

    # 创建

    HashMap::new() 函数

    collect() 方法, collect() 方法可以把数据整合成很多种集合类型,包括 HashMap ,返回值需要显式指明类型

    • 在元素类型 Vec<Tuple> 上使用 collect() 方法,可以组建一个 HashMap ,要求有两个 Tuple :一个作为 K ,一个作为 V

    • eg :

    • rust
      use std::collections::HashMap;
      fn main () {
        let teams = vec![String::from("Blue"), String::from("Yellow")];
        let intial_scores = vec! [10, 50];
      	let scores: HashMap<_, _> = teams.iter().zip(intial_scores.iter()).collect();
      }

    # 更新

    insert() 方法

    • 替换现有的 V

      • 如果向 HashMap 插入一对 KV ,然后再插入同样的 K ,但是不同的 V ,那么原来的 V 会被替换掉
    • 只在 K 不对应任何值的情况下,才插入 V

      • entry() 方法:检查指定的 K 是否对应一个 V
        • 参数为 K`
        • 返回 enum Entry :代表值是否存在
      • or_insert() 方法:
        • 当返回 enum Entry 代表值不存在,即 EntryVacantEntry 变体,将方法参数作为 K 的新值插进去,返回到这个值的可变引用
        • 否则返回到对应的 V 的一个可变引用
    rust
    scores.entry(String::from("Blue")).or_insert(50);

    # 所有权

    • 对于实现了 Copy trait 的类型(例如 i32 ),值会被复制到 HashMap
    • 对于拥有所有权的值(例如 String ),值会被移动,所有权会转移给 HashMap
    • 如果将值的引用插入到 HashMap ,值本身不会移动,在 HashMap 有效的期间,被引 用的值必须保持有效
    rust
    use std:: collections:: HashMap;
    fn main() {
      let field_name = String::from("Favorite color");
      let field_value = String::from("Blue");
      let mut map = HashMap::new();
      map.insert(&field_name, &field_value);
    	println!("{}: {}", field_name, field_value);// 这一句可以通过编译
    	map.insert(field_name, field_value);
    	println!("{}: {}", field_name, field_value);// 这一句无法通过编译

    # 读取

    get() 方法,借用一个参数 K ;返回类型为 Option<&V>

    rust
    match scores.get(&String::from("Blue")) {
      Some(s) => println!("{}", s),
      None => println!("team not exist"),
    }

    # 遍历

    for 循环 + 模式匹配

    rust
    for (k, v) in &scores {
      printin! ("{}: {}", k, v);
    }

    # Hash 函数

    默认情况下, HashMap 使用加密功能强大的 Hash 函数,可以抵抗拒绝服务(DoS)攻击。它具有如下特点:

    • 不是可用的最快的 Hash 算法
    • 但具有更好安全性。

    可以指定不同的 hasher 来切换到另一个函数。

    • hasher 是实现 BuildHasher trait 的类型

    # 错误处理

    错误处理体现了 Rust 的可靠性,大部分情况下在编译时提示错误并处理。

    Rust 没有类似异常的机制。

    错误的分类:

    • 可恢复错误:文件未找到,可再次尝试
      • 返回 Result<T , E>
    • 不可恢复错误,bug,例如访问的索引超出范围
      • 调用 panic!

    # 不可恢复错误

    panic! 宏执行:

    • 程序会打印一个错误信息,可自定义,即传递给 panic!&str 类型字符串
      • 调试 (默认 --debug ) 模式下才会打印
      • --release 模式下不会打印
    • 执行以下两个操作之一
      • (默认) 程序展开调用栈(工作量大)
        • Rust 沿着调用栈往回走
        • 清理每个遇到的函数
      • 立即中止调用栈
        • 不进行清理,直接停止程序
        • 内存需要 OS 进行清理
    • 退出程序

    想让二进制文件更小,把设置从 “展开” 改为 “中止”,只需在 Cargo.toml 文件下增加:

    toml
    [profile.release]
    panic = 'abort'

    将环境变量 RUST_BACKTRACE 设置为

    • 0 ,使 panic! 报错指出源代码出错的地方,即开发者所写的代码中
    • 1 ,使 panic! 报错指出真正出错的地方,即依赖的代码中,它将回溯,打印出包含错误点的所有调用函数的列表
    • full ,比 1 更加详细

    # 可恢复错误

    rust
    enum Result<T, E> {
      Ok(T),
      Err(E),
    }
    • T :操作成功情况下, Ok 变体里返回的数据的类型
    • E :操作失败情况下, Err 变体里返回的错误的类型

    match 表达式处理

    Option 枚举一样,Result 及其变体也是由 prelude 带入作用域

    rust
    use std::fs::File;
    // simple demo
    fn main() {
      let f = File::open("hello.txt");
      let f = match f {
        Ok(file) => file,
        Err(error) => {
          panic!("Error opening file {:?}", error);
        }
      };
    }
    // complex demo
    fn main() {
      let f = File::open( "hello.txt");
      let f = match f {
        Ok(file) => file,
        Err(error) => match error.kind() {
          ErrorKind::NotFound => match File::create("hello.txt") {
            Ok(fc) => fc,
            Err(e) => panic!("Error creating file: {:?}", e),
          }
          other_error => panic!("Error opening the file: {:?}", other_error),
        },
      };
    }

    闭包处理

    rust
    let f = File::open ("hello.txt").unwrap_or_else(|error| {
      if error.kind() == ErrorKind::NotFound {
        File::create( "hello.txt").unwrap_or_else(|error| {
          panic!("Error creating file: {:?}", error);
        })
      } else {
        panic! ("Error opening file: {:?}", error);
      }
    });

    unwrap() 方法,是 match 方法处理 Result<T, E> 的快捷方式,但是无法自定义错误信息

    rust
    let f = File::open("hello.txt");
    let f = match f {
      Ok(file) => file,
      Err(error) => {
        panic!("Error opening file {:?}", error);
      }
    };
    // 等价于
    let f = File::open("hello.txt").unwrap();

    expect() 方法,在 unwrap 的基础上,支持 自定义错误信息

    # 传播错误

    ? 用于快捷地传播错误

    • 如果 Result<T, E>Ok : Ok 中的值就是表达式的结果,然后继续执行程序
    • 如果 Result<T, E>Err : Err 本身 就是整个函数的返回值,就像使用了 return
    rust
    fn read username_from_file() -> Result<String, io::Error> {
      let f = File::open("hello.txt");
      let mut f = match f {
        Ok(file) => file,
        Err(e) => return Err(e),
      };
      let mut s = String::new();
      match f.read_to_string(&mut s) {
        Ok(_) => Ok(s),
        Err(e) => Err(e)
    // 相当于 Err (e) => return Err (e),
      }
    }
    // 等价于
    fn read username_from_file() -> Result<String, io::Error> {
      let f = File::open("hello.txt")?;
      let mut s = String::new();
      f.read_to_string(&mut s)?;
      Ok(s)
    }
    // 链式调用优化
    fn read username_from_file() -> Result<String, io::Error> {
      let mut s = String::new();
      File::open("hello. txt")?.read_to_string(&mut s)?;
      Ok(s)
    }

    ?from 函数

    trait std::convert::From 上的 from 函数用于错误之间的转换

    • ? 所应用的错误,会隐式的被 from 函数处理
    • ? 调用 from 函数时它所接收的错误类型会被转化为当前函数返回类型所定义的错误类型

    适用于要求针对不同错误原因,返回同一种错误类型,需要 每个错误类型实现了转换为所返回的错误类型的 from 函数

    ? 运算符与 main 函数

    ? 返回类型只能是 Result<T, E>Option<T> 或任何实现了的 Try 类型

    main 函数返回类型是: () ,不能直接返回 ? 运算符的结果

    main 函数的返回类型也可以是:Result<T,E> ,可以定义其返回 Box<dyn Error> ,这是 trait 对象,简单理解:“任何可能的错误类型”

    rust
    use std::fs::File;
    use std::error::Error;
    fn main() -> Result<(), Box<dyn Error>> {
      let f = File::open ("hello.txt")?;
      Ok(())
    }

    # 数据类型有效性检查

    panic! 的主要应用场景之一,自定义结构体方法(以下 demo 是 new )来处理数据是否有效

    t rust
    pub struct Guess {
      value: i32,
    }
    impl Guess {
      pub fn new(value: i32) -> Guess {
        if value <1 || value > 100 {
          panic!("Guess value must be between 1 and 100, got {}", value);
        }
        Guess { value }
      }
      pub fn value(&self) -> i32 {
        self.value
      }
    }

    getter :返回字段数据

    # 抽象化

    # 泛型

    泛型类似 C++ 的模版可以提高代码复用能力,处理重复代码的问题

    泛型是具体类型或其它属性的抽象代替,你编写的代码不是最终的代码,而是一种模板,里面有一些 “占位符”,编译器在编译时将 “占位符” 替换具体的类型。

    占位符的声明:

    • 定义函数时,在 fn 函数名 的后面用 <T, K, ...> 尖括号声明

    • 定义结构体时,在 struct 结构体名 的后面用 <T, K, ...> 尖括号声明

    • 定义枚举类型时,在 enum 枚举类型名 的后面用 <T, K, ...> 尖括号声明

    • 定义方法时,若结构体 / 枚举类型是用泛型定义的,则也需要声明占位符或类型

      • rust
        impl<T> Point<T> { // 在类型 T 上实现方法
          fn x(&self) -> &T {
            &self.x
          }
        }
      • rust
        impl Point<i32> { // 只针对具体类型实现方法
          fn x1(&self) -> i32 {
            &self. X
          }
        }
      • struct 里的泛型类型参数可以和方法的泛型类型参数不同

    例如: fn largest<T>(list:&T) -> T{...}T 就是一个泛型占位符

    当出现需要太多类型参数,代码往往 需要重组为多个更小的单元

    单态化(monomorphization)

    Rust 在编译时将泛型替换具体类型,这一过程叫单态化,使泛型代码性能与非泛型代码无异

    # trait

    Rust 中的 trait 是抽象的定义共享行为,与其它语言的接口 (interface) 类似,但有些区别,它告诉 Rust 编译器某种类型具有哪些并且可以与其它类型共享的功能

    trait bounds (约束):泛型类型参数指定为实现了特定行为的类型

    # 定义 trait

    trait 的定义:把方法签名放在一起,来定义实现某种目的所必需的一组行为。

    • 关键字: trait
    • 只有方法签名,没有具体实现
    • trait 可以有多个方法:每个方法签名占一行,以 ; 结尾
    • 实现该 trait 的类型必须提供具体的方法实现
    rust
    pub trait MyTrait {
      fn my_trait_function(&self) -> ... ;
    }

    # 实现 trait

    在类型上实现 trait 与为类型实现方法类似。格式:

    rust
    impl MyTrait for MyStruct1 {
      fn my_trait_function(&self) -> ... {
        ...
      }
    }
    impl MyTrait for MyStruct2 {
      fn my_trait_function(&self) -> ... {
        ...
      }
    }

    这样的实现实际上叫做重写实现

    # 约束

    可以在某个类型上实现某个 trait 的前提条件是:这个类型或这个 trait 是在本地 crate 里定义的。

    也就是无法外部类型来实现外部的 trait ,这个限制是程序属性的一部分(也就是一致性)。

    更具体地说是孤儿规则:之所以这样命名是因为父类型不存在。

    此规则的目的和原因:

    • 确保其他人的代码不能破坏您的代码,反之亦然。
    • 如果没有这个规则,两个 crate 可以为同一类型实现同一个 trait ,Rust 就不知道应该使用哪个实现了

    # 默认实现

    在定义 trait 的时候,不再只是函数签名,而是直接实现该函数,那么这个实现就是所有类型可以调用(虽然不一定正确)的默认实现

    注意:

    • 默认实现的方法可以调用 trait 中其它的方法,即使这些方法没有默认实现
    • 无法从方法的重写实现里面调用默认的实现

    # 项目结构惯例

    trait 的定义和实现常放在 src/lib.rs ,并加上 pub 关键字 ,在其他文件通过 use package_name::trait_name 导入

    # trait 作为参数

    # impl trait

    适用于简单情况,实际上 impl trait 语法是 trait bound 的语法糖

    rust
    pub fn notify(item: impl MyTrait) {
      //notify 这个函数将只能使用实现了 MyTrait 的类型作为参数 item
    }

    # trait bound

    可用于复杂情况

    rust
    pub fn notify<T: MyTrait>(item1: T, item2: T, ... ) {
      //notify 这个函数将只能使用实现了 MyTrait 的类型作为参数 item1,item2...
    }

    # 实用包、crate

    # std::env

    args 函数收集传入程序的参数

    collect 方法可以将参数按空格分割转化为 Vec<String>

    rust
    use std::env;
    fn main () {
      let args: Vec<String> = env::args().collect();
      //env::args_os () //OsString 迭代器
      println! ("{:?}", args);
    }

    # std::fs

    file system 缩写,显然是处理文件相关的模块

    read_to_string(filename) 函数读取 filename 路径的文件

    Edited on Views times
    \ No newline at end of file +Rust 基础 - 编程语言 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 参考资料

    • Rust 语言圣经 (自称)
      • 锈书
    • 官网

    # 环境配置

    更新 Rust

    bash
    rustup update

    卸载 Rust

    bash
    rustup self uninstall

    查看版本

    h bash
    rustc --version
    ## 显示格式
    rustc.x.y.z([commit hash] [commit date])

    安装 Rust 同时还会安装文档,查看本地文档

    bash
    rustup doc

    # Rust 工具

    # 编译器

    # rustc

    只适合单文件编译

    shell
    rustc xxx.rs <output>

    # cargo

    Cargo 是 Rust 的构建系统和包管理工具,适用于较大型项目编译、链接

    • 构建代码
    • 下载依赖的库
    • 构建这些库

    创建项目,会创建一个新目录 project 并且初始化一个 git 项目

    shell
    cargo new project

    创建好的项目结构

    shell
    .
    ├── .git
    ├── .gitignore
    ├── Cargo.toml
    └── src
        └── main.rs

    检查代码能否通过编译,并不会编译项目,很常用

    shell
    cargo check

    构建项目

    shell
    cargo build           # 默认为调试模式即 --debug
    cargo build --debug   # 同上
    cargo build --release # 发布模式,编译出的程序性能更高

    运行项目

    shell
    cargo run           # 默认为调试模式即 --debug
    cargo run --debug   # 同上
    cargo run --release # 发布模式,编译出的程序性能更高

    # 安装依赖库换源

    目前 cargo search 无法使用镜像

    • 字节镜像源参考
    • 华科镜像源参考

    # 方法一

    $HOME/.cargo/config.toml 文件 (没有则创建一个) 添加如下内容:

    l toml
    # 华科镜像源
    [registries]
    ustc = { index = "https://mirrors.ustc.edu.cn/crates.io-index/" } 
    # 或者
    [registries.ustc]
    index = "https://mirrors.ustc.edu.cn/crates.io-index/"

    或者稀疏索引的方式,要求 cargo >= 1.68 :

    toml
    [source.ustc-sparse]
    registry = "sparse+https://mirrors.ustc.edu.cn/crates.io-index/"

    同时需要在项目 cargo.toml 使用注册方式引入依赖库:

    toml
    [dependencies]
    <要引入的包> = { registry = "ustc" }
    <要引入的包> = { registry = "rsproxy" }

    # 方法二

    不需要对项目 cargo.toml 添加配置,只需要 $HOME/.cargo/config.toml 文件 (没有则创建一个) 添加如下内容:

    toml
    [source.crates-io]
    replace-with = 'rsproxy' # 字节跳动镜像源
    [source.rsproxy]
    registry = "https://rsproxy.cn/crates.io-index" # 镜像源 url

    稀疏索引,要求 cargo >= 1.68 :

    toml
    [source.rsproxy-sparse]
    registry = "sparse+https://rsproxy.cn/index/"
    [registries.rsproxy]
    index = "https://rsproxy.cn/crates.io-index"
    [net]
    git-fetch-with-cli = true

    # 项目结构

    # 项目结构要求

    • Cargo.toml 在项目顶层下
    • 源代码都应该在 src 目录下
    • 顶层目录可以放置:README、许可信息、配置文件和其它与程序源码无关的文件

    如果创建项目时没有使用 cargo,也可以把项目转化为使用 cargo:

    • 把源代码文件移动到 src 下
    • 创建 Cargo.toml 并填写相应的配置

    # Cargo.toml

    eg:

    toml
    [package]
    name = "my-project"
    version = "0.1.0"
    authors = ["fuuzen <fuuzen.github.io>"]
    edition = "2024"
    [dependencies]
    ...

    # [package]

    • name
    • version
    • authors
    • edition

    # [dependencies]

    • crate 包
    • ...

    三种描述:

    • 基于 Rust 官方仓库 crates.io ,通过版本说明来描述
    • 基于项目源代码的 git 仓库地址,通过 URL 来描述
    • 基于本地项目的绝对路径或者相对路径,通过类 Unix 模式的路径来描述

    eg :

    l toml
    [dependencies]
    rand = "0.3"
    hammer = { version = "0.5.0"}
    color = { git = "https://github.com/bjz/color-rs" }
    geometry = { path = "crates/geometry" }

    # 变量和常量

    # 变量

    # let 变量绑定 (binding)

    rust 中 let 表达式的类似其他语言赋值的作用,实际上不是赋值,而叫做 绑定,这与 rust 所有权 这一核心概念有关

    默认绑定变量类型为 immutable 不可变的,需要可变则须加上 mut 显式声明

    rust
    // 使用 let 来声明变量,进行绑定,a 是不可变的
    // 此处没有指定 a 的类型,编译器会默认根据 a 的值为 a 推断类型:i32,有符号 32 位整数
    // 语句的末尾必须以分号结尾
    let a = 10;
    // 主动指定 b 的类型为 i32
    let b: i32 = 20;
    // 这里有两点值得注意:
    // 1. 可以在数值中带上类型:30i32 表示数值是 30,类型是 i32
    // 2. c 是可变的,mut 是 mutable 的缩写
    let mut c = 30i32;
    // 还能在数值和类型中间添加一个下划线,让可读性更好
    let d = 30_i32;
    // 跟其它语言一样,可以使用一个函数的返回值来作为另一个函数的参数
    let e = add(add(a, b), add(c, d));

    # _ 编译器忽略未使用变量

    告诉编译器不要警告未使用的变量,为此可以用下划线作为变量名的开头

    rsut
    let _a = 5

    # let 变量解构

    从一个复杂的变量中匹配出部分内容

    rust
    let (a, mut b): (bool,bool) = (true, false);
    //a = true, 不可变;b = false,可变
    println!("a = {:?}, b = {:?}", a, b);
    b = true;
    assert_eq!(a, b);

    # 解构式赋值

    Rust 1.59 版本后可以在赋值语句的左式中使用元组、切片和结构体模式,不需要 let

    struct Struct {
        e: i32
    }
    fn main() {
        let (a, b, c, d, e);
        (a, b) = (1, 2);
        //_ 代表匹配一个值,但是我们不关心具体的值是什么,因此没有使用一个变量名而是使用了 _
        [c, .., d, _] = [1, 2, 3, 4, 5];
        Struct { e, .. } = Struct { e: 5 };
        assert_eq!([1, 2, 1, 4, 5], [a, b, c, d, e]);
    }

    这种使用方式跟之前的 let 保持了一致性,但是 let 会重新绑定,而这里仅仅是对之前绑定的变量进行再赋值。

    使用 += 的赋值语句还不支持解构式赋值

    # 变量遮蔽 (shadowing)

    Rust 允许声明相同的变量名,在后面声明的变量会遮蔽掉前面声明的。

    变量遮蔽的用处在于,如果你在某个作用域内无需再使用之前的变量(在被遮蔽后,无法再访问到之前的同名变量),就可以重复的使用变量名字,而不用绞尽脑汁去想更多的名字。

    • 可以使用相同的名字声明新的变量,新的变量就会 shadow(隐藏)之前声明的同名变量

    • 在后续的代码中这个变量名代表的就是新的变量

    shadow 和把变量标记为 mut 是不一样的:

    • 如果不使用 let 关键字,那么重新给非 mut 的变量赋值会导致编译时错误
    • 而使用 let 声明的同名新变量,也是不可变的
    • 使用 let 声明的同名新变量,它的类型可以与之前不同

    # 常量

    常量也是绑定到一个常量名且不允许更改的值

    # 常量和变量之间的差异

    • 常量不允许使用 mut常量不仅仅默认不可变,而且自始至终不可变,因为常量在编译完成后,已经确定它的值。
    • 常量使用 const 关键字而不是 let 关键字来声明,并且值的类型必须标注

    # Rust 常量的命名约定

    • 全部字母都使用大写
    • 使用下划线分隔单词,另外对数字字面量可插入下划线以提高可读性

    # 基本类型 (标量)

    # 原始类型

    # 整数类型

    类型定义的形式统一为: 有无符号 + 类型大小(位数)

    长度有符号类型无符号类型
    8 位i8u8
    16 位i16u16
    32 位i32u32
    64 位i64u64
    128 位i128u128
    视架构而定isizeusize

    # isizeusize 类型

    isizeusize 类型的位数由程序运行的计算机的架构所决定:

    • 如果是 64 位计算机,那就是 64 位的
    • 如果是 32 位计算机,那就是 32 位的
    • ...

    # 整数字面值

    rust 有以下 5 种整数字面类型:

    Number literalsExample
    Decimal98_222
    Hex0xff
    Octal0o77
    Binary0b1111_0000
    Byte (u8 only)b'A'
    • 除了 Byte 类型外,所有的数值字面值都允许使用类型后缀。

      • 例如 57u8
    • 如果不太清楚应该使用那种类型,可以使用 Rust 相应的默认类型

      • 整数的默认类型就是 i32 ,其速度整体上是较快的,即使是在 64 位操作系统中

    # 整数溢出

    • 调试模式 (--debug) 下编译:Rust 会检查整数溢出,如果发生溢出,程序在运行时就会 panic

    • 发布模式下(--release)编译:Rust 不会检查可能导致 panic 的整数溢出

      • 如果溢出发生:Rust 会执行 “环绕” 操作:(取模)

        • 例如: u8 的范围是 0-255,如果把一个 u8 变量的值设为 256,那么 Rust 会将 256 变成 0,257 变成 1 ⋯
      • 但程序不会 panic

    # 浮点类型

    Rust 有两种基础的浮点类型,也就是含有小数部分的类型:

    • f32 ,32 位,单精度
    • f64 ,64 位,双精度

    Rust 的浮点类型使用了 IEEE-754 标准来表述

    f64默认类型,因为在现代 CPU 上 f64f32 的速度差不多,而且精度更高。

    # 布尔类型

    • 符号是 bool
    • Rust 的布尔类型也有两个值:
      • true
      • false
    • 占一个字节大小

    Rust 不允许其他类型直接当作布尔类型使用,和其他语言不太一样!

    # 字符类型

    • 符号是 char
    • Rust 语言中 char 类型被用来描述语言中最基础的单个字符。
    • 字符类型的字面值使用单引号
    • 占用 4 字节大小
    • 是 Unicode 标量值,可以表示比 ASCII 多得多的字符内容,包括:拼音、中日韩文、零长度空白字符、emoji 表情等。

    Unicode 中并没有 “字符” 的概念,所以直觉上认为的字符也许与 Rust 中的概念并不相符

    # 基本类型转换

    # 隐式

    1. 整数类型之间的转换:Rust 允许在整数类型之间进行隐式转换,只要目标类型的表示范围足够容纳源类型的值。例如,可以将 u8 转换为 u16i32 转换为 u64 等。
    2. 浮点数类型之间的转换:Rust 允许在浮点数类型之间进行隐式转换,只要目标类型能够表示源类型的精度和范围。例如,可以将 f32 转换为 f64
    3. 整数类型到浮点数类型的转换:Rust 允许将整数类型隐式转换为相应的浮点数类型。例如,可以将 i32 转换为 f64
    4. 字面量的隐式转换:在字面量的使用中,Rust 会根据上下文自动进行隐式类型转换。例如,可以将整数字面量 42 隐式转换为 u8i32 或其他整数类型,或将浮点数字面量 3.14 隐式转换为 f32f64

    # 显式

    Rust 鼓励显式的类型转换,并提供了一些安全的类型转换方法,以确保程序在类型转换时保持安全和可预测。Rust 有三种变量类型显式转换的方法:

    • as 关键字
    • frominto 这些 trait 进行转换
    • 字符串类型可以使用 parse 方法转换到数字类型

    # as & as_<T>()

    as 运算符转换示例:

    t rust
    let x: i32 = 5;
    let y: u32 = x as u32;

    as 运算符还有一个安全版本 as_<T>() ,它会在溢出时返回一个 Option 值。

    eg:

    rust
    fn main() {
        let num: u32 = 42;
        
        // 将 u32 类型转换为 i16 类型
        let converted: Option<i16> = num.as_();
        
        match converted {
            Some(value) => println!("Converted value: {}", value),
            None => println!("Conversion failed"),
        }
    }

    # From & Into

    在 Rust 中, FromInto 是用于自定义类型转换的 trait。通过实现这些 trait,可以定义自定义类型之间的转换规则。

    From trait 定义了一个类型转换的关联函数 from() ,用于将其他类型转换为当前类型。

    Into trait 则定义了一个类型转换的方法 into() ,用于将当前类型转换为其他类型。

    eg:

    rust
    // 定义一个自定义类型
    struct MyType(u32);
    // 实现 From trait,将 u32 类型转换为 MyType
    impl From<u32> for MyType {
        fn from(value: u32) -> Self {
            MyType(value)
        }
    }
    // 实现 Into trait,将 MyType 转换为 u32 类型
    impl Into<u32> for MyType {
        fn into(self) -> u32 {
            self.0
        }
    }
    fn main() {
        let num: u32 = 42;
        
        // 使用 From trait 进行转换
        let my_type: MyType = MyType::from(num);
        
        // 使用 Into trait 进行转换
        let back_to_num: u32 = my_type.into();
        println!("MyType: {:?}", my_type);
        println!("Back to u32: {}", back_to_num);
    }

    # parse

    使用 parse 方法将字符串类型转换为数字类型,如整数或浮点数。 parse 方法是定义在 str 类型上的一个函数,它返回一个 Result 枚举类型,表示解析操作的结果。

    eg:

    rust
    fn main() {
        let num_str = "42";
        
        // 将字符串转换为整数类型
        let num: Result<i32, _> = num_str.parse();
        
        match num {
            Ok(n) => println!("Parsed number: {}", n),
            Err(e) => println!("Error: {}", e),
        }
    }

    # 复合类型

    • 复合类型可以将多个值放在一个类型里。
    • Rust 提供了两种基础的复合类型:元组(Tuple)、数组

    # Tuple

    • Tuple 可以将多个类型的多个值放在一个类型里
    • Tuple 的长度是固定的:一旦声明就无法改变
    • Tuple 中的每个位置都对应一个类型, Tuple 中各元素的类型不必相同

    # 创建 Tuple

    在小括号里,将值用逗号分开

    rust
    let tup: (132, f64, 48) = (500, 6.4, 1);

    # 获取 Tuple 的元素值

    可以使用模式匹配来解构(destructure)一个 Tuple 来取元素的值

    rust
    let (x, y, z) = tup;

    # 访问 Tuple 的元素

    在 tuple 变量使用点标记法,后接元素的索引号

    rust
    println!("{} {} {}", tup.O, tup.1, tup.2);

    # 数组

    • 数组也可以将多个值放在一个类型里
    • 数组中每个元素的类型必须相同
    • 数组的长度也是固定的

    数组没有 Vector 灵活,如果想让你的数据存放在 stack(栈)上而不是 heap(堆)上,或者想保证有固定数量的元素,这时使用数组更有好处

    # 数组的类型

    数组的类型以这种形式表示: [类型;长度]

    # 声明一个数组

    两种方法:

    • 在中括号里,各值用逗号分开
    • 如果数组的每个元素值都相同,那么可以定义为: [初始值;长度]
    rust
    let a = [1, 2, 3, 4, 51];
    let a =3;5]
    // 它就相当于:let a= [3, 3, 3, 3, 3];

    # 访问数组的元素

    可以使用索引来访问数组的元素。

    如果访问的索引超出了数组的范围,那么:

    • 索引是字面量等,编译器能检查出来,编译不通过
    • 编译器没能检查出来,编译通过
      • 运行会报错(runtime 时会 panic)
      • Rust 不会允许其继续访问相应地址的内存

    # 枚举类型

    enum 关键字表示枚举类型

    # 定义枚举

    例如 IP 地址:IPV4、IPv6

    rust
    enum IpAddrKind {
      V4,
      V6,
    }

    标准库中的 IpAddr

    rust
    struct Ipv4Addr {
      // --snip--
    struct Ipv6Addr {
      // --snip--
    }
    enum IpAddr {
      V4(Ipv4Addr),
      V6(Ipv6Addr),
    }

    # 变体、标识符、枚举值

    • 变体是枚举类型的不同选项或可能的值之一
    • 标识符是用于表示枚举类型的变体的名称
    • 枚举值是通过选择枚举类型的一个变体来实例化的具体值

    枚举的变体都位于标识符的命名空间下,使用两个冒号 :: 进行分隔

    rust
    let four = IpAddrKind::V4;
    let six = IpAddrKind::V6;

    # 将数据附加到枚举的变体中

    rust
    enum IpAddr {
      V4(String),
      V6(String),
    }
    // 每个变体可以拥有不同的类型以及关联的数据量
    enum IpAddr {
      V4(u8, u8, u8, u8),
      V6(String),
    }
    • 优点:
      • 不需要额外使用 struct
      • 每个变体可以拥有不同的类型以及关联的数据量

    # 枚举方法

    struct 类似,也使用 impl 关键字

    # Option 枚举

    • 定义于标准库中
    • 在 Prelude(预导入模块)中
    • 描述了:某个值可能存在(某种类型)或不存在的情况
    # Option<T>

    Rust 没有 Null。其它语言中,Null 是一个值,它表示 “没有值”,一个变量可以处于两种状态:空值(nuI)、非空。

    Null 引用: Billion Dollar Mistake

    Null 的问题在于:当你尝试像使用非 Null 值那样使用 NuIl 值的时候,就会引起某种错误

    Null 的概念还是有用的:因某种原因而变无效或缺失的值

    Rust 中类似 Null 概念的枚举 Option<T> ,定义如下:

    rust
    enum Option<T>{
      Some(T),
      None,
    }

    T 是一个类型参数,代表值的类型。这里出现三个类型都定义在标准库中:

    • Option<T>
    • Some (T)Some 表示存在一个值,并将该值包装在 Some
    • None

    Option<T>T 是不同的类型,不可以把 Option<T> 直接当成 T ,例如以下代码会报错:

    rust
    let x:i8 =5;
    let y: Option<i8> = Some(5);
    let sum = x + y; // 编译报错:error [E0277]: cannot add `Option<i8>` to `i8`

    若想使用 Option<T> 中的 T ,必须将它转换为 T

    # 基础控制流

    # 选择分支

    # if else

    rust
    if ... {
      ...
    } else if ... {
      ...
    } else {
      ...
    }

    如下代码在 rust 编译器环境下会产生报错:

    rust
    let number = if condition { 5 } else { "6" }

    因为返回给 number 的结果类型不确定!

    # match

    允许一个值与一系列模式进行匹配,并执行匹配的模式对应的代码,模式可以是字面值、变量名、通配符...

    match 匹配必须穷举所有的可能,否则会产生报错:

    bash
    error [E0004]: non-exhaustive patterns:...
    • _ 通配符:替代其余没列出的值
    rust
    match ... {
      a => (...),
      b => (...),
      _ => (...),
    }

    # if let

    语法的一般形式如下:

    rust
    if let pattern = expression {
      // 匹配成功的处理逻辑
    } else if let pattern = expression {
      // 匹配成功的处理逻辑
    } else {
      // 匹配失败的处理逻辑
    }

    在上述语法中, pattern 是要匹配的模式, expression 是要进行匹配的表达式。如果表达式与模式匹配成功,则执行匹配成功的处理逻辑。如果匹配失败,则执行匹配失败的处理逻辑。

    下面是一个示例,展示了如何使用 if let 处理 Option 类型:

    rust
    fn main() {
        let some_value: Option<i32> = Some(5);
        if let Some(x) = some_value {
            println!("Got value: {}", x);
        } else {
            println!("Value is None");
        }
    }

    在上述示例中, some_value 是一个 Option<i32> 类型的变量,被赋值为 Some(5) 。通过使用 if let Some(x) = some_value ,我们尝试将 some_value 解构为 Some 并提取其中的值。如果匹配成功,将会执行 println!("Got value: {}", x) ,打印提取到的值。如果匹配失败(即 some_valueNone ),则执行 println!("Value is None")

    使用 if let 处理只关心一种匹配而忽略其它匹配的情况,可以使代码更简洁、清晰地处理特定模式的匹配,减少了手动编写 match 表达式的工作量,

    可以把 iflet 看作是 match 的语法糖

    # 循环

    # loop

    • 没有条件判断
    • 使用 break 跳出循环
    • loop 有返回值,为 break 后面的表达式
    rust
    let result = loop {
      ...
      if ... {
        break (expression) ;
      }
    }

    # while

    rust
    while ... {
      ...
    }

    和其他语言差别不大

    # for

    loopwhile 更适合遍历容器,更安全、高效

    rust
    // eg:
    let a = [10, 20, 30, 40, 50];
    for element in a.iter() {
      println! ("the value is: {}", element);
    }

    # 所有权 Ownership

    # 什么是所有权

    Rust 的核心特性就是所有权

    所有程序在运行时都必须管理它们使用的计算机内存

    • 有些语言有垃圾收集机制 (GC),在程序运行时,它们会不断地寻找不再使用的内存
    • 还有一些语言,程序员必须显式地分配和释放内存 (eg: C)
    • Rust 采用了第三种方式:
      • 内存是通过一个所有权系统来管理的,其中包含一组编译器在编译时检查的规则
      • 当程序运行时,所有权特性不会减慢程序的运行速度,因为 Rust 的所有权系统将将内存的管理提前到了编译阶段

    在像 Rust 这样的系统级编程语言里,一个值是在 stack 上还是在 heap 上对语言的行和你为什么要做某些决定是有更大的影响的。管理 heap 数据就是所有权存在的原因。

    # Stack vs Heap

    在代码运行的时候,Stack 和 Heap 都是你可用的内存,但他们的结构很不相同:

    • Stack

      • Stack 按值的接收顺序来存储,按相反的顺序将它们移除(后进先出,LIFO)

        • 添加数据叫做压入栈 push
        • 移除数据叫做弹出栈 pop
      • 所有存储在 Stack 上的数据必须拥有已知的固定的大小

        • 因为指针是已知固定大小的,可以把指针存放在 stack 上。但如果想要实际数据,你必须使用指针来定位,然后访问 heap
      • 把数据压到 stack 上要比在 heap 上分配快得多,因为操作系统不需要寻找用来存储新数据的空间,那个位置永远都在 stack 的顶端

      • 如果数据存放的距离比较近,那么处理器的处理速度就会更快一些

    • Heap

      • 当你把数据放入 heap 时,你会请求一定数量的空间,这个过程叫做在 heap 上进行分配,有时仅仅称 “分配”。在 heap 上分配空间需要做更多的工作

        • 操作系统在 heap 里找到一块足够大的空间
        • 然后要做好记录方便下次分配,把它标记为在用
        • 并返回一个指针,也就是这个空间的地址
      • 编译时大小未知的数据或运行时大小可能发生变化的数据必须存放在 heap 上

      • 如果数据之间的距离比较远,那么处理速度就会慢一些

    所有权解决的问题:

    • 跟踪代码的哪些部分正在使用 heap 的哪些数据
    • 最小化 heap 上的重复数据量
    • 清理 heap 上未使用的数据以避免空间不足。

    掌握了所有权,那么就不需要经常去想 stack 或 heap 了

    # 所有权规则

    # 三条规则

    • 每个值都有一个变量,这个变量是该值的所有者
    • 每个值同时只能有一个所有者
    • 当所有者超出作用域(scope)时
      • 它拥有它的值的所有权,就会被 drop 函数清理 (heap 上的数据)
      • 它没有它的值的所有权,不会清理

    # 返回值与作用域

    函数在返回值的过程中同样也会发生所有权的转移,返回表达式的值的所有权将转移到函数结果所赋值的变量上

    # 复制 Copy trait

    用于像整数这样完全存放在 stack 上面不包含 heap 上的简单数据类型(标量)

    • 如果一个类型实现了 Copy 这个 trait ,那么旧的变量在赋值后仍然可用
    • 如果一个类型或者该类型的一部分实现了 Drop trait ,那么 Rust 不允许让它再去实现 Copy trait
    rust
    let x = 5;
    let y = x;
    //x 和 y 都是有效的
    println! ("{} {}", x, y);

    # 移动 Move

    多个变量可以与同一个数据使用一种独特的方式 Move 来交互

    String 类型为例

    rust
    let s1 = String::from ("hello"); 
    let s2 = s1;

    当把 s1 赋给 s2String 在 stack 上的数据被复制了一份,heap 内容不变

    string-move

    s1s2 离开作用域时,它们都会尝试释放相同的内存,就会导致二次释放 (double free) bug 。为了避免这个问题,Rust 会让原来的 s1 失效 ,此后当 s1・离开作用域的时候,Rust 不需要释放任何东西。

    s1 此后处于 moved 状态,即失效。

    • 浅拷贝(shallow copy)
      • 只拷贝 stack 上的数据,有可能产生二次释放 (double free) bug
    • 深拷贝(deep copy)
      • 同时拷贝 stack 和 heap 上的数据

    Rust 中采用的是第三种 “拷贝”,也就是 ** 移动 Move **,在浅拷贝的基础上让原来的 stack 数据失效。

    这体现了 Rust 隐含的一个设计原则:Rust 不会自动创建数据的深拷贝。就运行时性能而言,任何自动赋值的操作都是廉价的。

    # 克隆 CLone

    针对包含 heap 数据的矢量数据类型,相当于深拷贝

    rust
    let s1 = String::from("Hello");
    let s2 = s1.clone();

    string-clone

    # 引用 Reference

    & 符号就表示引用,允许引用该值而不取得其所有权;

    对应的解引用符号是 *

    和 C/C++ 一样,Rust 引用是符号是 & ,解引用符号是 *

    string-reference

    Rust 中的引用在任何给定的时刻,引用必须一直有效 (非悬空引用),只能满足下列条件之一:

    • 一个可变的引用
    • 任意数量不可变的引用

    # 可变引用 Mutable Reference

    Rust 支持可变引用 ,用 &mut 符号代替 & 即为可变引用。

    但是可变引用有以下几个限制:

    • 在特定作用域内,对于某一块数据,只能有一个可变引用
    • 不可以同时拥有可变引用和不可变引用
    • (多个不变的引用是可以的)

    第一个限制的好处是可在编译时防止数据竞争。违反将出现如下报错:

    bash
    error[E0499]: cannot borrow 'xxx' as mutable more than once at a time

    第二个限制避免了可变引用对不可变引用的影响。违反将出现如下报错:

    bash
    error[E0502]: cannot borrow 'xxx' as mutable because it is also borrowed as immutable

    ::: infromation

    以下三种行为下会发生数据竞争:

    • 两个或多个指针同时访问同一个数据
    • 至少有一个指针用于写入数据
    • 没有使用任何机制来同步对数据的访问

    数据竞争在运行时是很难发现的(C/C++ 里经常出现),所以 Rust 从根本上避免了这种问题来保证运行时的安全性

    :::

    可以通过创建新的作用域,来允许非同时的创建多个可变引用

    rust
    fn main () {
      let mut s = String:: from "Hello");
    	{
    	  let s1 = &mut s;
    	}
      let s2 = &mut s;
    }

    # 悬空引用 Dangling Reference

    也叫悬垂引用,在 Rust 里,编译器可保证引用永远都不是悬空引用。如果你引用了某些数据,编译器将保证在引用离开作用域之前数据不会离开作用域。对于任何出现的悬空引用, Rust 在编译阶段就会报错:

    bash
    error[E0106]: missing lifetime specifier

    悬空指针(Dangling Pointer):一个指针引用了内存中的某个地址,而这块内存可能已经释放并分配给其它人使用了。

    # 借用 Borrow

    引用作为函数参数这个行为叫做借用 ,本质上就是引用。

    rust
    fn main () {
      let mut s1 = String::from("Hello");
      let len = calculate_length(&mut s1);
      println!("The length of '{}' is {}.", s1, len);
    }
    fn calculate_length(s: &mut String) -> usize {
      s.push_str(", world");
      s.len()
    }

    # 切片类型 slice

    Rust 的另外一种不持有所有权的数据类型

    # 字符串切片

    字符串切片是指向字符串中一部分内容的引用

    形式: [开始索引..结束索引]

    • 开始索引就是切片起始位置的索引值,若不写则默认为 0
    • 结束索引是切片终止位置的下一个索引值,若不写则默认为 String 的 Capacity
    rust
    fn main(){
      let s = String::from("Hello world");
      let hello = &s[0..5]; // 左闭右开
      let world = &s[6..11];// 左闭右开
    }

    string-slice

    字符串切片的范围索引必须发生在有效的 UTF-8 字符边界内

    如果尝试从一个多字节的字符中创建字符串切片,程序会报错并退出

    字符串字面值实际上就是字符串切片 String slice

    rust
    //s 的类型是 &str
    let s = "Hello, World!"

    ⭐️ 将字符串切片作参数传递

    rust
    fn first_words:(&String) -> &str {}

    对于这样接受字符串作为参数的函数,有经验的 Rust 开发者会采用 &str 作为参数类型,因这样就可以同时接收 String
    和 &str 类型的参数了:

    rust
    fn first_words:(&str) -> &str {}
    • 使用字符串切片,直接作为参数调用该函数 first_words(mystr)
    • 使用 String,可以创建一个完整的 String 切片作为参数来调用该函数 firstwords(&mystr[..])

    定义函数时使用字符串切片来代替字符串引用会使我们的 API 更加通用,且不会损失任何功能。

    # 数组切片

    rust
    let a = [1, 2, 3, 4, 5];
    let slice = &a[1..3];

    # 矢量变量

    # String 类型

    字符串字面值是程序里手写的那些字符串值,它们是不可变的、硬编码的。

    Rust 还有第二种字符串类型: Sting

    String 比那些基础标量数据类型更复杂,在 heap 上分配。能够存储在编译时未知数量的文本。

    # 数据结构

    一个 String 由 stack 上的 3 部分组成:

    • 一个指向存放字符串内容的内存的指针
    • 长度 len,就是存放字符串内容所需的字节数
    • 容量 capacity ,是 String 从操作系统总共获得内存的总字节数

    存放字符串内容的部分在 heap 上

    string"hello"

    # 创建 String 类型的值

    可以使用 from 函数从字符串字面值创建出 String 类型

    rust
    let s = String::from("hello");

    :: 表示 fromString 类型下的函数

    这类字符串是可以被修改的

    # Vector 类型

    # struct

    # struct 基础操作

    # 定义 struct

    • 使用 struct 关键字,并內整个 struct 命名
    • 在花括号内,为所有字段(Field)定义名称和类型
    rust
    struct User {
      username: String,
      email: String,
      sign_in_count: u64,
      active: bool,
    }

    # 实例化 struct

    想要使用 struct ,需要创建 struct 的实例:

    • 每个字段指定具体值
    • 无需按声明的顺序进行指定
    rust
    let user1 = User {
      email: String::from("someone@example.com"),
      username: String::from("someusername123"),
      active: true,
      sign_in_count: 1,
    }

    一旦 struct 的实例是可变的,那么实例中所有的字段都是可变的

    # 取得 struct 里面的某个值

    使用点标记法:

    rust
    user1.email = String::from("anotheremail@example.com");

    # 字段初始化简写

    字段名与字段值对应变量名相同时,就可以使用字段初始化简写的方式:

    rust
    fn build_user(email: String, username: String) -> User {
      User {
        email,
        username,
        active: true,
        sign_in_count: 1,
      }
    }

    # struct 更新语法

    当你想基于某个 struct 实例来创建一个新实例的时候,可以使用 struct 更新语法:

    rust
    // 不使用 struct 更新语法
    let user2 = User {
      email: String::from("another@example.com"),
      username: String::from("anotherusername567"),
      active: user1.active,
      sign_in_count: user1.sign_in_count,
    }
    // 使用 struct 更新语法
    let user2 = User {
      email: String::from("another@example.com"),
      username: String: from ("anotherusername567"),
      ..user1
    };

    # debug 阶段打印 struct

    rust
    #[derive(Debug)] // 添加注解,表示派生自 Debug 这个 trait
    ...
    println!("{:?}", mystruct);// 输出为一行,不够清晰直观
    println!("{:#?}", mystruct);// 输出结构化信息,更清晰直观

    # struct 数据的所有权

    rust
    struct User {
      username: String,
      email: String,
      sign_in_count: u64,
      active: bool,
    }

    这里的字段使用了 String 而不是 &str:

    • struct 实例拥有其所有的数据
    • 只要 struct 实例是有效的,那么里面的字段数据也是有效的
    • struct 里也可以存放引用,但这需要使用生命周期

    # struct 的方法

    Rust 是一个多范式的语言,也支持面向对象编程

    使用 impl 关键字:

    rust
    impl mystruct {
      fn ... (&self, ...){
        
      }
      ...
    }

    Rust 方法与函数不同之处:

    • 方法是在 struct (或 enumtrait 对象)的上下文中定义
    • 第一个参数是 self ,表示方法被调用的 struct 实例

    # struct 扩展定义

    # tuple struct

    Rust 可定义类似 tuplestruct ,叫做 tuple struct

    Tuple struct 整体有个名字,但里面的元素没有名字

    适用于想给整个 tuple 起名,并让它不同于其它 tuple ,而且又不需要给每个元素起名

    # 定义 tuple struct

    使用 struct 关键字,后边是名字,以及里面元素的类型

    rust
    struct Color(i32, i32, i32);
    struct Point(i32, i32, i32);
    let black = Color(0, 0, 0);
    let origin = Point(0, 0, 0);
    //black 和 origin 是不同的类型,是不同 tuple struct 的实例

    # Unit-Like Struct

    可以定义没有任何字段的 struct,叫做 Unit-Like struct(因为与空元组 () 也就是单元类型类似)

    适用于需要在某个类型上实现某个 trait,但是在里面又没有想要存储的数据

    # Package, Crate, Module

    Rust 的模块系统:

    • Package(包):Cargo 的特性,让你构建、测试、共享 crate
    • Crate(单元包):一个模块树,它可产生一个 library 或可执行文件
    • Module(模块): use :让你控制代码的组织、作用域、私有路径
    • Path(路径):为 structfunctionmodule 等项命名的方式

    # Package

    一个 Package 包含:

    • 1 个 Cargo.toml ,它描述了如何构建这些 Crates
    • 必须至少包含一个 crate (library 或 binary)
      • 只能包含 0-1 个 library crate
      • 可以包含任意数量的 binary crate

    # Package 项目结构

    这些是 Cargo 的惯例

    • 一个 Package 包含一个 binary crate src/main.rssrc/lib.rs

      • src/main.rs 即为唯一的 binary crate
        • binary crate 的 crate root
    • 一个 Package 可以包含多个 binary crate

      • src/bin 下每个文件是单独的 binary crate
    • 一个 Package 包含 0 或 1 个 library crate

      • src/lib.rs 存在则包含 1 个 library crate ,否则不包含
      • library crate 的 crate root

    crate 名与 package 名 ( name ) 相同

    Cargo 把 crate root 文件交给 rustc 来构建 library 或 binary

    # Crate

    Crate 的作用:将相关功能组合到一个作用域内,便于在项目间进行共享,防止冲突

    Crate 的类型有两类:

    • binary
    • library

    Crate Root 是源代码文件,Rust 编译器从这里开始,组成 Crate 的根 Module

    src/main.rssrc/lib.rs 叫做 crate roots

    # Module

    其作用在于:

    • 在一个 crate 内,将代码进行分组
    • 增加可读性,易于复用
    • 控制项目(item)的私有性(public、private)

    # 建立 module

    使用 mod 关键字

    • 可嵌套
    • 可包含其它项( structenum 、常量、 trait 、函数等)的定义

    # 私有边界(privacy boundary)

    模块不仅可以组织代码,还可以定义私有边界。如果想把 函数 或 struct 等设为私有,可以将它放到某个模块中。只要是在 mod 里,Rust 中所有的条目(函数,方法, structenum ,模块,常量)默认是私有的。

    :::

    这里的条目包括:函数,方法, structenum ,模块,常量等所有可以被引用、调用的东西

    :::

    • 父级模块无法访问子模块中的私有条目
    • 子模块里可以使用所有祖先模块中的条目
    • 文件的根级无论是私有还是公共的都可以互相调用

    # pub 关键字

    使用 pub 关键字来将某些条目标记为公共的

    rust
    mod front_of_house {
      pub mod hosting {
        pub fn add_to_waitlist() {}
      }
    }
    pub fn eat_at_restaurant() {
      crate::front_of_house::hosting::add_to_waitlist(); // 绝对路径调用
      front_of_house::hosting::add_to_waitlist(); // 相对路径调用
    }

    pubstruct 使用, struct 内所有成员仍然默认是私有的!

    pubenum 使用, enum 内所有成员就都是公共的!

    # Path

    为了在 Rust 的模块中找到某个条目,需要使用路径。路径有两种形式:

    • 绝对路径:从 crate root 开始,使用 crate 名 或字面值 crate
    • 相对路径:从当前模块开始,使用 selfsuper 或当前模块的标识符

    路径至少由一个标识符组成,标识符之间使用 ::

    # super 关键字

    类似文件系统中的 .. 表示上一级目录, super 表示用来访问父级模块路径中的内容。

    rust
    fn serve_order() {}
    mod back_of__house {
      fn fix_incorrect_order {
        cook_order();
        super::serve_order();
      }
      fn cook_order() {}
    }

    # use 关键字

    use 关键字将路径导入到作用域内,仍遵循私有性原则,即只有公共部分才能使用

    rust
    mod front_of_house {
      pub mod hosting {
        pub fn add_to_waitlist() {}
      }
    }
    // 绝对路径:
    use crate::front_of_house::hosting; 
    // 相对路径:
    use front_of_house::hosting; 
    // 默认导入到当前文件 (crate) 作用域内
    // 此后 hosting 等效于如下效果:
    mod hosting() {}

    Rust 惯用做法:并不直接 use 导入函数本身,而是导入函数的父模块,通过父模块调用函数,以此来表示这个函数不是本地定义的,而是从其他地方导入的。

    但对于 struct , enum 则是直接导入它们本身,除非出现同名条目,则只好导入它们的父级模块来区分它们,或者用 as 关键字为他们取别名。

    # pub use 重导出

    use 导入的是在本文件作用域内的私有的模块

    该用 pub use 关键字则外部代码导入本 文件时也能使用这些模块

    # 外部包和标准库

    外部包以及标准库 ( std ) 的使用需要:

    1. Cargo.tomldependencies 添加依赖的包;标准库 ( std ) 也被当做外部包但不需要这一步
    2. use 将特定条目引入作用域,外部包和标准库都需要。

    # 嵌套路径

    如果使用同一个包或模块下的多个条目,可以使用嵌套路径清理大量的 use 语句

    rust
    use 路径相同的部分::{self, 到包1路径差异的部分, 到包2 路径差异的部分, ...}
    // 当需要导入路径相同部分所指向的包本身时可以用 self

    eg :

    rust
    use std::io;
    use std::io::Write;
    // 上面代码等效于下面
    use std::io::{self, Write}

    # 通配符 *

    使用 * 可以把路径中所有的公共条目都引入到作用域。 但注意要谨慎使用。

    应用场景:

    • 测试,将所有被测试代码引入到 tests 模块
    • 有时被用于预导入(prelude)模块

    # 将模块内容移动到其它文件

    模块定义时,如果模块名后边是 ; ,而不是代码块:

    • Rust 会从与模块同名的文件中加载内容
    • 模块树的结构不会变化

    这里我详细解释我讲不清楚,也懒得写了🤯

    # 常用集合

    # Vec<T>

    # 创建

    Vec::new() 函数

    vec!

    # 更新

    push 方法,注意是可变引用

    # 删除

    与任何其它 struct v 一样,当 Vec 离开作用域后

    • 它就被清理掉了
    • 它所有的元素也被清理掉了

    # 读取

    下标索引

    get 方法 ,返回 Option<T> 类型,与 match 搭配

    # 遍历

    rust
    for element in my_vector {
      ...element...
    }
    for element in &my_vector {
      ...*element...
    }
    for element in &mut my_vector {
      ...*element...
    }

    # String

    Rust 开发者经常会被字符串困扰的原因

    • Rust 倾向于暴露可能的错误
    • 字符串数据结构复杂
    • UTF-8

    Rust 选择将正确处理 String 数据作所有 Rust 程序的默认行为

    • 缺点:程序员必须在处理 UTF-8 数据之前投入更多的精力

    • 优点:可防止在开发后期处理涉及非 ASCII 字符的错误

    # 区分字符串和 String

    字符串是什么

    • Rust 的核心语言层面,只有一个字符串类型:字符串切片 str(或 &str)
    • 字符串切片:对存储在其它地方、UTF-8 编码的字符串的引用
      • 字符串字面值:存储在二进制文件中,也是字符串切片

    String 类型是什么

    • 来自 标准库 而不是 核心语言
    • 可增长、可修改、可拥有

    二者都采用 UTF-8 編码。

    Rust 的标准库还包含了很多其它的字符串类型,例如: OsStringOsStrGStrngCStr ... 它们可能使用了不同的编码。通常 String 结尾的是可获得所有权的; Str 结尾的是可借用的。

    String 的本质是对 Vec<u8> 的包装,很多 Vec<T> 的操作都可用于 String

    # 创建

    String::new() 函数

    to_string() 方法,可用于实现了 Display trait 的类型,包括字符串字面值

    String::from() 函数

    # 更新

    push_str( xxx : &str ) 方法,把一个字符串切片附加到 String 后面,注意是可变引用,传入的参数是不获取所有权的借用

    push( x : char ) 方法,把单个字符附加到 String 后面,注意是可变引用,传入的参数是不获取所有权的借用

    + 运算符,连接两个字符串 ,注意必须满足: String = String + &strString = String + &String会取得 + 左边字符串的所有权

    这里 + 运算符使用了类似这个签名的方法 fn add(self, s:&str)-> String{..}

    标准库中的 add 方法使用了泛型 ;本来只能 String = String + &str ,但这里出现了解引用强制转換(deref coercion),从而可以 String = String + &String

    format! 宏,连接多个字符串,不会取得任何参数的所有权

    rust
    let s3 = s1 + "-" + &s2 + "_" + &s3;
    // 等效于
    let s3 = format!("{}-{}-{}", s1, s2, s3);

    # 读取

    Rust 的字符串不支持索引语法访问,原因在于采用的是 UTF-8 编码,很多语言的字符并不能只通过一个字节表示,这些字符专业术语叫 Unicode 标量值,例如汉字需要 3 个字节。

    Rust 的字符串不支持索引语法访问的另外一个原因在于索引操作应消耗一个常量时间 O(1) ,而 String 无法保证:需要遍历所有内容,来确定有多少个合法的字符。

    len() 方法,返回字符串长度

    # 删除

    [..] 切片,由于上述 UTF-8 编码的原因,虽然编译通过,但对于很多语言的字符,如果切割时跨越了字符边界就会 出现 panic

    # 遍历

    Rust 有三种看待字符串的方式:

    • Bytes, 字节,可以用 bytes 方法返回迭代器遍历
    • Scalar Values, 标量值,可以用 chars 方法返回迭代器遍历
    • Grapheme Clusters, 字形簇(最接近所谓的 “字母”),比较复杂,标准库未提供

    # HashMap<K,V>

    HashMap 用的较少,不在 Prelude 中,所以要使用它就需要 use std::collections::HashMap ; 标准库对其支持较少,没有内置的宏来创建 HashMap

    注意 HashMap 数据存储在 heap 上。

    HashMap 时同构的,即一个 HashMap 中:

    • 所有的 K 必须是同一种类型
    • 所有的 V 必须是同一种类型

    # 创建

    HashMap::new() 函数

    collect() 方法, collect() 方法可以把数据整合成很多种集合类型,包括 HashMap ,返回值需要显式指明类型

    • 在元素类型 Vec<Tuple> 上使用 collect() 方法,可以组建一个 HashMap ,要求有两个 Tuple :一个作为 K ,一个作为 V

    • eg :

    • rust
      use std::collections::HashMap;
      fn main () {
        let teams = vec![String::from("Blue"), String::from("Yellow")];
        let intial_scores = vec! [10, 50];
      	let scores: HashMap<_, _> = teams.iter().zip(intial_scores.iter()).collect();
      }

    # 更新

    insert() 方法

    • 替换现有的 V

      • 如果向 HashMap 插入一对 KV ,然后再插入同样的 K ,但是不同的 V ,那么原来的 V 会被替换掉
    • 只在 K 不对应任何值的情况下,才插入 V

      • entry() 方法:检查指定的 K 是否对应一个 V
        • 参数为 K`
        • 返回 enum Entry :代表值是否存在
      • or_insert() 方法:
        • 当返回 enum Entry 代表值不存在,即 EntryVacantEntry 变体,将方法参数作为 K 的新值插进去,返回到这个值的可变引用
        • 否则返回到对应的 V 的一个可变引用
    rust
    scores.entry(String::from("Blue")).or_insert(50);

    # 所有权

    • 对于实现了 Copy trait 的类型(例如 i32 ),值会被复制到 HashMap
    • 对于拥有所有权的值(例如 String ),值会被移动,所有权会转移给 HashMap
    • 如果将值的引用插入到 HashMap ,值本身不会移动,在 HashMap 有效的期间,被引 用的值必须保持有效
    rust
    use std:: collections:: HashMap;
    fn main() {
      let field_name = String::from("Favorite color");
      let field_value = String::from("Blue");
      let mut map = HashMap::new();
      map.insert(&field_name, &field_value);
    	println!("{}: {}", field_name, field_value);// 这一句可以通过编译
    	map.insert(field_name, field_value);
    	println!("{}: {}", field_name, field_value);// 这一句无法通过编译

    # 读取

    get() 方法,借用一个参数 K ;返回类型为 Option<&V>

    rust
    match scores.get(&String::from("Blue")) {
      Some(s) => println!("{}", s),
      None => println!("team not exist"),
    }

    # 遍历

    for 循环 + 模式匹配

    rust
    for (k, v) in &scores {
      printin! ("{}: {}", k, v);
    }

    # Hash 函数

    默认情况下, HashMap 使用加密功能强大的 Hash 函数,可以抵抗拒绝服务(DoS)攻击。它具有如下特点:

    • 不是可用的最快的 Hash 算法
    • 但具有更好安全性。

    可以指定不同的 hasher 来切换到另一个函数。

    • hasher 是实现 BuildHasher trait 的类型

    # 错误处理

    错误处理体现了 Rust 的可靠性,大部分情况下在编译时提示错误并处理。

    Rust 没有类似异常的机制。

    错误的分类:

    • 可恢复错误:文件未找到,可再次尝试
      • 返回 Result<T , E>
    • 不可恢复错误,bug,例如访问的索引超出范围
      • 调用 panic!

    # 不可恢复错误

    panic! 宏执行:

    • 程序会打印一个错误信息,可自定义,即传递给 panic!&str 类型字符串
      • 调试 (默认 --debug ) 模式下才会打印
      • --release 模式下不会打印
    • 执行以下两个操作之一
      • (默认) 程序展开调用栈(工作量大)
        • Rust 沿着调用栈往回走
        • 清理每个遇到的函数
      • 立即中止调用栈
        • 不进行清理,直接停止程序
        • 内存需要 OS 进行清理
    • 退出程序

    想让二进制文件更小,把设置从 “展开” 改为 “中止”,只需在 Cargo.toml 文件下增加:

    toml
    [profile.release]
    panic = 'abort'

    将环境变量 RUST_BACKTRACE 设置为

    • 0 ,使 panic! 报错指出源代码出错的地方,即开发者所写的代码中
    • 1 ,使 panic! 报错指出真正出错的地方,即依赖的代码中,它将回溯,打印出包含错误点的所有调用函数的列表
    • full ,比 1 更加详细

    # 可恢复错误

    rust
    enum Result<T, E> {
      Ok(T),
      Err(E),
    }
    • T :操作成功情况下, Ok 变体里返回的数据的类型
    • E :操作失败情况下, Err 变体里返回的错误的类型

    match 表达式处理

    Option 枚举一样,Result 及其变体也是由 prelude 带入作用域

    rust
    use std::fs::File;
    // simple demo
    fn main() {
      let f = File::open("hello.txt");
      let f = match f {
        Ok(file) => file,
        Err(error) => {
          panic!("Error opening file {:?}", error);
        }
      };
    }
    // complex demo
    fn main() {
      let f = File::open( "hello.txt");
      let f = match f {
        Ok(file) => file,
        Err(error) => match error.kind() {
          ErrorKind::NotFound => match File::create("hello.txt") {
            Ok(fc) => fc,
            Err(e) => panic!("Error creating file: {:?}", e),
          }
          other_error => panic!("Error opening the file: {:?}", other_error),
        },
      };
    }

    闭包处理

    rust
    let f = File::open ("hello.txt").unwrap_or_else(|error| {
      if error.kind() == ErrorKind::NotFound {
        File::create( "hello.txt").unwrap_or_else(|error| {
          panic!("Error creating file: {:?}", error);
        })
      } else {
        panic! ("Error opening file: {:?}", error);
      }
    });

    unwrap() 方法,是 match 方法处理 Result<T, E> 的快捷方式,但是无法自定义错误信息

    rust
    let f = File::open("hello.txt");
    let f = match f {
      Ok(file) => file,
      Err(error) => {
        panic!("Error opening file {:?}", error);
      }
    };
    // 等价于
    let f = File::open("hello.txt").unwrap();

    expect() 方法,在 unwrap 的基础上,支持 自定义错误信息

    # 传播错误

    ? 用于快捷地传播错误

    • 如果 Result<T, E>Ok : Ok 中的值就是表达式的结果,然后继续执行程序
    • 如果 Result<T, E>Err : Err 本身 就是整个函数的返回值,就像使用了 return
    rust
    fn read username_from_file() -> Result<String, io::Error> {
      let f = File::open("hello.txt");
      let mut f = match f {
        Ok(file) => file,
        Err(e) => return Err(e),
      };
      let mut s = String::new();
      match f.read_to_string(&mut s) {
        Ok(_) => Ok(s),
        Err(e) => Err(e)
    // 相当于 Err (e) => return Err (e),
      }
    }
    // 等价于
    fn read username_from_file() -> Result<String, io::Error> {
      let f = File::open("hello.txt")?;
      let mut s = String::new();
      f.read_to_string(&mut s)?;
      Ok(s)
    }
    // 链式调用优化
    fn read username_from_file() -> Result<String, io::Error> {
      let mut s = String::new();
      File::open("hello. txt")?.read_to_string(&mut s)?;
      Ok(s)
    }

    ?from 函数

    trait std::convert::From 上的 from 函数用于错误之间的转换

    • ? 所应用的错误,会隐式的被 from 函数处理
    • ? 调用 from 函数时它所接收的错误类型会被转化为当前函数返回类型所定义的错误类型

    适用于要求针对不同错误原因,返回同一种错误类型,需要 每个错误类型实现了转换为所返回的错误类型的 from 函数

    ? 运算符与 main 函数

    ? 返回类型只能是 Result<T, E>Option<T> 或任何实现了的 Try 类型

    main 函数返回类型是: () ,不能直接返回 ? 运算符的结果

    main 函数的返回类型也可以是:Result<T,E> ,可以定义其返回 Box<dyn Error> ,这是 trait 对象,简单理解:“任何可能的错误类型”

    rust
    use std::fs::File;
    use std::error::Error;
    fn main() -> Result<(), Box<dyn Error>> {
      let f = File::open ("hello.txt")?;
      Ok(())
    }

    # 数据类型有效性检查

    panic! 的主要应用场景之一,自定义结构体方法(以下 demo 是 new )来处理数据是否有效

    t rust
    pub struct Guess {
      value: i32,
    }
    impl Guess {
      pub fn new(value: i32) -> Guess {
        if value <1 || value > 100 {
          panic!("Guess value must be between 1 and 100, got {}", value);
        }
        Guess { value }
      }
      pub fn value(&self) -> i32 {
        self.value
      }
    }

    getter :返回字段数据

    # 抽象化

    # 泛型

    泛型类似 C++ 的模版可以提高代码复用能力,处理重复代码的问题

    泛型是具体类型或其它属性的抽象代替,你编写的代码不是最终的代码,而是一种模板,里面有一些 “占位符”,编译器在编译时将 “占位符” 替换具体的类型。

    占位符的声明:

    • 定义函数时,在 fn 函数名 的后面用 <T, K, ...> 尖括号声明

    • 定义结构体时,在 struct 结构体名 的后面用 <T, K, ...> 尖括号声明

    • 定义枚举类型时,在 enum 枚举类型名 的后面用 <T, K, ...> 尖括号声明

    • 定义方法时,若结构体 / 枚举类型是用泛型定义的,则也需要声明占位符或类型

      • rust
        impl<T> Point<T> { // 在类型 T 上实现方法
          fn x(&self) -> &T {
            &self.x
          }
        }
      • rust
        impl Point<i32> { // 只针对具体类型实现方法
          fn x1(&self) -> i32 {
            &self. X
          }
        }
      • struct 里的泛型类型参数可以和方法的泛型类型参数不同

    例如: fn largest<T>(list:&T) -> T{...}T 就是一个泛型占位符

    当出现需要太多类型参数,代码往往 需要重组为多个更小的单元

    单态化(monomorphization)

    Rust 在编译时将泛型替换具体类型,这一过程叫单态化,使泛型代码性能与非泛型代码无异

    # trait

    Rust 中的 trait 是抽象的定义共享行为,与其它语言的接口 (interface) 类似,但有些区别,它告诉 Rust 编译器某种类型具有哪些并且可以与其它类型共享的功能

    trait bounds (约束):泛型类型参数指定为实现了特定行为的类型

    # 定义 trait

    trait 的定义:把方法签名放在一起,来定义实现某种目的所必需的一组行为。

    • 关键字: trait
    • 只有方法签名,没有具体实现
    • trait 可以有多个方法:每个方法签名占一行,以 ; 结尾
    • 实现该 trait 的类型必须提供具体的方法实现
    rust
    pub trait MyTrait {
      fn my_trait_function(&self) -> ... ;
    }

    # 实现 trait

    在类型上实现 trait 与为类型实现方法类似。格式:

    rust
    impl MyTrait for MyStruct1 {
      fn my_trait_function(&self) -> ... {
        ...
      }
    }
    impl MyTrait for MyStruct2 {
      fn my_trait_function(&self) -> ... {
        ...
      }
    }

    这样的实现实际上叫做重写实现

    # 约束

    可以在某个类型上实现某个 trait 的前提条件是:这个类型或这个 trait 是在本地 crate 里定义的。

    也就是无法外部类型来实现外部的 trait ,这个限制是程序属性的一部分(也就是一致性)。

    更具体地说是孤儿规则:之所以这样命名是因为父类型不存在。

    此规则的目的和原因:

    • 确保其他人的代码不能破坏您的代码,反之亦然。
    • 如果没有这个规则,两个 crate 可以为同一类型实现同一个 trait ,Rust 就不知道应该使用哪个实现了

    # 默认实现

    在定义 trait 的时候,不再只是函数签名,而是直接实现该函数,那么这个实现就是所有类型可以调用(虽然不一定正确)的默认实现

    注意:

    • 默认实现的方法可以调用 trait 中其它的方法,即使这些方法没有默认实现
    • 无法从方法的重写实现里面调用默认的实现

    # 项目结构惯例

    trait 的定义和实现常放在 src/lib.rs ,并加上 pub 关键字 ,在其他文件通过 use package_name::trait_name 导入

    # trait 作为参数

    # impl trait

    适用于简单情况,实际上 impl trait 语法是 trait bound 的语法糖

    rust
    pub fn notify(item: impl MyTrait) {
      //notify 这个函数将只能使用实现了 MyTrait 的类型作为参数 item
    }

    # trait bound

    可用于复杂情况

    rust
    pub fn notify<T: MyTrait>(item1: T, item2: T, ... ) {
      //notify 这个函数将只能使用实现了 MyTrait 的类型作为参数 item1,item2...
    }

    # 实用包、crate

    # std::env

    args 函数收集传入程序的参数

    collect 方法可以将参数按空格分割转化为 Vec<String>

    rust
    use std::env;
    fn main () {
      let args: Vec<String> = env::args().collect();
      //env::args_os () //OsString 迭代器
      println! ("{:?}", args);
    }

    # std::fs

    file system 缩写,显然是处理文件相关的模块

    read_to_string(filename) 函数读取 filename 路径的文件

    Edited on Views times
    \ No newline at end of file diff --git a/cs/os/concepts/index.html b/cs/os/concepts/index.html index de44b26e..eabca79b 100644 --- a/cs/os/concepts/index.html +++ b/cs/os/concepts/index.html @@ -1 +1 @@ -操作系统绪论 - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 操作系统的定义

    实际上操作系统没有普遍接受的定义

    操作系统 (Operating System,OS)是指控制管理整个计算机系统的硬件和软件资源,并合理地组织调度计算机的工作资源分配;以提供给用户其他软件方便的接口环境;它是计算机系统中最基本的系统软件

    # 操作系统的功能和目标

    # 作为系统资源的管理者

    # 提供的功能

    • 处理机管理

    • 存储器管理

    • 文件管理

    • 设备管理

    # 目标

    安全、高效

    # 向上层提供方便易用的服务

    # GUI

    (Graphical User Interface) 图形化用户接口

    # CLI

    (Command Line Interface) 命令行接口

    • 联机命令接口 = 交互式命令接口
    • 脱机命令接口 = 批处理命令接口

    # 程序接口

    可以在程序中进行系统调用来使用程序接口。普通用户不能直接使用程序接口,只能通过程序代码间接使用。

    软件和程序员才能使用程序接口

    封装思想:操作系统把一些丑陋的硬件功能封装成简单易用的服务,使用户能更方便地使用计算机,用户无需关心底层硬件的原理,只需要对操作系统发出命令即可

    # 作为最接近硬件的层次

    目标:实现对硬件机器的拓展

    没有任何软件支持的计算机称为裸机。在裸机上安装的操作系统,可以提供资源管理功能和方便用户的服务功能,将裸机改造成功能更强、使用更方便的机器

    通常把覆盖了软件的机器成为扩充机器,又称之虚拟机

    # 操作系统的特征

    # 并发

    并发:指两个或多个事件在同一时间间隔内发生。这些事件宏观上是同时发生的,但微观上是交替发生的

    操作系统的并发性:计算机系统中 “同时” 运行着多个程序,这些程序宏观上看是同时运行着的,而微观上看是交替运行的。

    操作系统就是伴随着 “多道程序技术” 而出现的。因此,操作系统和程序并发是一起诞生的。

    注意(重要考点):单核 CPU 同一时刻只能执行一个程序,各个程序只能并发地执行多核 CPU 同一时刻可以同时执行多个程序,多个程序可以并行地执行

    # 共享

    共享即资源共享,是指系统中的资源可供内存中多个并发执行的进程共同使用。

    # 互斥共享方式

    系统中的某些资源,虽然可以提供给多个进程使用,但一个时间段内只允许一个进程访问该资源

    # 同时共享方式

    系统中的某些资源,允许一个时间段内由多个进程 “同时” 对它们进行访问

    所谓的 “同时” 往往是宏观上的,而在微观上,这些进程可能是交替地对该资源进行访问的(即分时共享)

    并发共享 的关系是相互依存,不可分割

    如果失去并发性,则系统中只有一个程序正在运行,则共享性失去存在的意义;

    如果失去共享性,则不同的程序不能同时访问硬盘资源,就无法实现同时发送文件,也就无法并发

    # 虚拟

    虚拟是指把一个物理上的实体变为若干个逻辑上的对应物。物理实体(前者)是实际存在的,而逻辑上对应物(后者)是用户感受到的。

    # 空分复用技术

    如虚拟存储器技术

    # 时分复用技术

    如虚拟处理器

    并发共享 的关系是相互依存,不可分割

    如果失去了并发性,则一个时间段内系统中只需运行一道程序,那么就失去了实现虚拟性的意义了。因此,没有并发性,就谈不上虚拟性

    # 异步

    异步指,在多道程序环境下,允许多个程序并发执行,但由于资源有限,进程的执行不是一贯到底的,而是走走停停,以不可预知的速度向前推进,这就是进程的异步性。

    # 操作系统的发展和分类

    # 手工操作阶段

    主要缺点:

    用户独占全机、人机速度矛盾导致资源利用率极低

    # 批处理阶段

    引入脱机输入 / 输出技术(用外围机 + 磁带完成),并由监督程序负责控制作业的输入、输出,通过外围机把程序提前存到磁带里

    目标是追求吞吐量,提高资源利用率

    # 脱机

    在主机以外的设备上进行输入 / 输出操作,需要时再送主机处理。

    # 批处理 bash

    后端运行的作业,开始运行之后就不再需要任何交互。由于无法交互控制程序,所以还不算真正意义上的操作系统。

    # 单道批处理系统

    也叫简单批处理系统。已经出现操作系统的一些特征:

    • 用户态、内核态(monitor)
    • 时钟

    主要优点:

    缓解了一定程度的人机速度矛盾,资源利用率有所提升。


    主要缺点:

    内存中仅能有一道程序运行,只有该程序运行结束之后才能调入下一道程序。

    CPU 有大量的时间是在空闲等待 1/0 完成。资源利用率依然很低

    没有人机交互功能。

    # 多道程序设计

    多道程序的运行环境比单道程序等更加复杂。

    特点:

    • 多道 —— 计算机内存中同时存放多道相互独立的程序
    • 宏观上并行
    • 微观上串行
    • 非封闭、非顺序 —— 引入多道程序后,程序的执行就失去了封闭性和顺序性
    • 相互制约 —— 程序执行因为共享资源以及相互协同的原因相互制约
    • 断续性 —— 考虑到竞争的公平性,程序的执行是断续的

    多道程序设计的实现需要解决如下问题:

    • 处理机的分配
    • 多道程序的内存分配
    • I/O 设备的分配
    • 如何组织和存放大量的程序和数据,以方便用户使用并保证一致性安全性
      • 一定需要实现虚拟存储吗?—— 不一定,早期多道批处理操作系统将所有进程数据全部调入主存

    这些问题的处理也会增加系统开销。

    多道程序设计开始引入中断,目的是使 CPU 和 I/O 设备并行运行,提高资源利用率。

    # 多道批处理系统(操作系统开始出现)

    在批处理系统中采用多道程序设计就形成了多道批处理系统。

    每次往内存中读入多道程序,操作系统正式诞生,用于支持多道程序并发运行。


    主要优点:

    多道程序并发执行,共享计算机资源。资源利用率大幅提升,CPU 和其他资源更能保持 “忙碌” 状态,系统吞吐量增大。


    主要缺点:

    用户响应时间长,仍然没有人机交互功能(用户提交自己的作业之后就只能等待计算机处理完成,中间不能控制自己的作业执行。eg:无法调试程序 / 无法在程序运行过程中输入一些参数)

    # 分时操作系统

    计算机以时间片为单位轮流为各个用户 / 作业服务,各个用户可通过终端与计算机进行交互。

    目标是追求时延(latency)


    主要特征:

    • 同时性、也叫多路性
    • 交互性
    • 独立性
    • 及时性

    主要优点:

    用户请求可以被即时响应,解决了人机交互问题。允许多个用户同时使用一台计算机,并且用户对计算机的操作相互独立,感受不到别人的存在。


    主要缺点:

    不能优先处理一些紧急任务。操作系统对各个用户 / 作业都是完全公平的,循环地为每个用户 / 作业服务一个时间片,不区分任务的紧急性。

    # 实时操作系统

    主要优点:

    能够优先响应一些紧急任务,某些紧急任务不需时间片排队。

    在实时操作系统的控制下,计算机系统接收到外部信号后及时进行处理,并且要在严格的时限内处理完事件。实时操作系统的主要特点是及时性可靠性

    # 硬实时系统

    必须在绝对严格的规定时间内完成处理

    eg: 导弹控制系统、自动驾驶系统

    # 软实时系统

    能接受偶尔违反时间规定

    如:12306 火车订票系统

    # 网络操作系统

    是伴随着计算机网络的发展而诞生的,能把网络中各个计算机有机地结合起来,实现数据传送等功能,实现网络中各种资源的共享(如文件共享)和各台计算机之间的通信。(如:Windows NT 就是一种典型的网络操作系统,网站服务器就可以使用)

    # 分布式操作系统

    主要特点是分布性和并行性。系统中的各台计算机地位相同,任何工作都可以分布在这些计算机上,由它们并行、协同完成这个任务。

    # 个人计算机操作系统

    eg: windows、macos

    # 操作系统服务

    • 资源分配
    • 记账
    • 日志服务

    # 内核

    内核是操作系统最重要最核心的部分,也是最接近硬件的部分

    甚至可以説,一个操作系統只要有内核就够了(eg:Docker-> 仅需 Linux 内核)

    操作系统的功能未必都在内核中,如图形化用户界面 GUI

    由很多内核程序组成了 “操作系统内核”,或简称 “内核、(Kernel)”

    # 内核程序 —— 运行在内核态

    微软、苹果有一帮人负责实现操作系统,他们写的是 “内核程序”。操作系统内核作为 “管理者”,有时会让 CPU 执行一些 “特权指令”,如:内存清零指令。这些指令影响重大,只允许 “管理者”—— 即操作系统内核来使用

    普通程序员写的程序就是 “应用程序”,应用程序只能使用 “非特权指令”,如:加法指令、减法指令等

    在 CPU 设计和生产的时候就划分了特权指令和非特权指令,因此 CPU 执行一条指令前就能判断出其类型

    CPU 中有一个寄存器叫 程序状态字寄存器(PSW),其中有个二进制位,1 表示 “内核态”,0 表示 “用户态”

    # 内核态 —— 运行内核程序

    CPU 有两种状态,“内核态” 和 “用户态”。处于内核态时,说明此时正在运行的是内核程序,此时可以执行特权指令处于用户态时,说明此时正在运行的是应用程序,此时只能执行非特权指令

    内核态 = 核心态 = 管态;用户态 = 目态

    # 内核态和用户态的切换 (变态)

    变态的过程是有成本的,要消耗不少时间,频繁地变态会降低系统性能

    # 内核态 -> 用户态

    执行一条特权指令一一修改 PSW 的标志位为 “用户态”,这个动作意味着操作系统将主动让出 CPU 使用权

    # 用户态 -> 内核态

    由 “中断” 引发,硬件自动完成变态过程,触发中断信号意味着操作系统将强行夺回 CPU 的使用权。除了非法使用特权指令之外,还有很多事件会触发中断信号。一个共性是,旦凡需要操作系统介入的地方,都会触发中断信号

    # 系统调用

    对象与系统调用、库函数的关系
    普通应用程序可直接进行系统调用,也可使用库函数。 有的库函数涉及系统调用,有的不涉及
    编程语言向上提供库函数。有时会将系统调用封装成库函数,以隐藏系统调用的一些细节,使程序员编程更加方便。
    操作系统向上提供系统调用,使得上层程序能请求内核的服务
    裸机

    不涉及系统调用的库函数.eg

    • “取绝对值” 的函数

    涉及系统调用的库函数.eg

    • “创建一个新文件” 的函数

    # 按功能分类

    • 设备管理 —— 完成设备请求 / 释放 / 启动 等功能
    • 文件管理 —— 完成文件读 / 写 / 创建 / 删除 等功能
    • 进程控制 —— 完成进程创建 / 撒销 / 阻塞 / 唤醒 等功能
    • 进程通信 —— 完成进程之间的 消息传递 / 信号传递 等功能
    • 内存管理 —— 完成内存分配 / 回收 等功能

    应用程序通过系统调用请求操作系统的服务。而系统中的各种共享资源都由操作系统内核统一掌管,因此凡是与共享资源有关的操作(如存储分配、1/0 操作、文件管理等),都必须通过系统调用的方式向操作系统内核提出服务请求,由操作系统内核代为完成。这样可以保证系统的稳定性和安全性,防止用户进行非法操作。

    # 系统调用过程

    1. 传递系统调用参数
      • 寄存器传递参数
      • 内存中存储在块或表中的参数,以及作为参数在寄存器中传递的块地址
        • linux 和 solaris 采用这种方式
      • 由程序放置或推送到堆栈上并由操作系统弹出堆栈到参数

    块和堆栈的方式传递参数大小不受限制

    1. 执行陷入指令(用户态)
    2. 执行相应的内请求核程序处理系统调用(内核态)
    3. 返回应用程序

    陷入指令 = trap 指令 = 访管指令

    陷入指令是在用户态执行的,执行陷入指令之后立即引发一个内中断,使 CPU 进入内核态

    发出系统调用请求是在用户态,而对系统调用的相应处理内核态下进行

    # 计算计组成原理相关基础概念

    # 中断和异常

    参考这个

    没有中断就没有并发,也就没有操作系统

    中断是让操作系统内核夺回 CPU 使用权的唯一途径,使 CPU 由用户态变为内核态

    通用概念产生 原因AArch64x86-64
    中断硬件 异步异步异常 (重置 / 中断)中断 (可屏蔽 / 不可屏蔽)
    异常软件 同步同步异常 (终止 / 异常指令)异常 (Fault/Trap/Abort)

    中断

    (广义上的中断包括)Interrupt Handlers used in all 3 scenarios :

    • HW -> CPU -> Kernel: Classic HW Interrupt (狭义上的中断)
    • User -> Kernel: SW Interrupt
    • CPU -> Kernel: Exception

    # 其他类型 (非狭义) 中断

    • SW Interrupt
      • Interrupts caused by the execution of a software instruction: INT<interrupt_number>
      • Used by the system call interrupt()
      • Initiated by the running (user level) process
      • Cause current processing to be interrupted and transfers control to the corresponding interrupt handler in the kernel
    • Exceptions
      • Initiated by processor hardware itself
      • Example: divide by zero
      • Cause a transfer of control to the kernel to handle the exception (like a software interrupt)

    # Linux 系统的中断

    • 在中断处理中做尽量少的事
    • 推迟非关键行为

    结构:

    • Top half :马上做,找到 handler

      • 最小的、公共行为

        • 保存寄存器、屏蔽其他中断
        • 恢复寄存器,返回原来场景

        最重要:调用合适的由硬件驱动提供的中断处理 handler

        因为中断被屏蔽,所以不要做太多事情 (时间、空间)

        使用将请求放入队列,或者设置标志位将其他处理推迟到 bottom half

        现代处理器中,多个 I/O 设备共享一个 IRQ 和中断向量 v 多个 ISR (interrupt service routines) 可以结合在一个向量上
        调用每个设备对应该 IRQ 的 ISR

    • Bottom half : 推迟处理 (softirq, tasklets, 工作队列,内核线程)

      • 提供一些推迟完成任务的机制

      • softirqs

      • tasklets (建立在 softirqs 之上)

      • 工作队列

      • 内核线程

      这些工作可以被中断

    linux中断

    中断处理没有进程上下文!

    中断 (和异常相比) 和具体的某条指令无关

    也和中断时正在跑的进程、用户程序无关

    中断处理 handler 不能睡眠!

    中断处理的一些约束条件:

    • 不能睡眠
      • 或者调用可能会睡眠的任务
    • 不能调用 schedule () 调度
    • 不能释放信号或调用可能睡眠的操作
    • 不能和用户地址空间交换数据

    # 中断向量表

    每一个处理器有一个中断向量表,由操作系统内核建立,指向处理不同时间的中断处理程序入口

    结构:

    中断向量表

    # 需要中断的场景举例

    Hardware devices may need asynchronous and immediate service. For example:

    • Timer interrupt: Timers and time-dependent activities need to be updated with the passage of time at precise intervals
    • Network interrupt: The network card interrupts the CPU when data arrives from the network
    • I/O device interrupt: I/O devices (such as mouse and keyboard) issue hardware interrupts when they have input (e.g., a new character or mouse click)

    # 多核系统上的中断

    Howareinterruptshandledonmulticoremachines?

    • On x86 systems each CPU gets its own local Advanced Programmable Interrupt Controller (APIC). They are wired in a way that allows routing device interrupts to any selected local APIC.
    • The OS can program the APICs to determine which interrupts get routed to which CPUs.
    • The default (unless OS states otherwise) is to route all interrupts to processor 0

    # 陷入指令

    执行 “陷入指令”,意味着应用程序主动地将 CPU 控制权还给操作系统内核。“系统调用” 就是通过陷入指令完成的

    # I/O (输入输出系统)

    # BUS (总线)

    # 并行系统

    # 多核处理器

    # 多处理器系统

    又称并行系统、紧耦合系统

    区分 多处理器系统多核系统 ,实际计算机往往既是多核的也是多处理器的

    多核和多处理器系统都用于加速计算过程。多核包含在单个 CPU 中的多个核或处理单元。多处理器由多个 CPU 组成。多核处理器不需要像多处理器那样复杂的配置。相反,多处理器更可靠并且能够运行许多程序。

    # 三个优点一个缺点???

    老师出的作业题😅难道本来是想问多道批处理系统?而且那个也是单处理器系统啊)

    • 增加吞吐量,并行处理?
    • 规模经济,降低成本,比多核处理器便宜?
    • 增加可靠性?
    • ???

    # 非对称多处理器系统 (AMP)

    非对称多处理是指一种计算机结构,其中有多个处理器,但它们并不都是一样的。这意味着,一个 CPU 可能正在处理操作系统代码,而另一个 CPU 正在执行输入和输出工作。它使用两个或更多的处理器由一个主处理器处理。所有的 CPU 都是相互连接的,但不是自我调度的。它用于根据任务的优先级和重要性将特定任务安排到一个 CPU。

    非对称处理器的应用场景:被用于嵌入式系统,在这些系统中,特定任务需要并发执行,但系统不需要像对称多处理系统那样的通用计算能力

    # 对称多处理器系统 (SMP)

    SMP(Sysmmetric Multi-Processor System,对称多处理器),它由多个具有对称关系的处理器组成。所谓对称,即处理器之间没有主从之分。SMP 架构使得一台计算机不再由单个 CPU 组成,而是由两个或更多的处理器连接到一个共享的主存储器上,并能够共同执行任务。这些处理器通常是相同的,能够运行分配给它们的任何任务。这可以提高性能,因为任务可以在处理器之间分配,使其更快完成。

    对称多处理将多个 CPU 应用于一个任务,以并行和更快的方式完成。因此,在对称多处理系统中,两个或更多的 CPU 被连接到一个共享的主存储器。同时,所有这些 CPU 都可以完全访问输入和输出设备。在对称多处理系统中,操作系统认为所有的处理器都是平等的。

    对称处理器 (SMP) 架构同时也是 UMA 架构

    SMP 是一个紧密耦合的多处理器系统,它有一组独立运行的相同的 CPU。每个 CPU 处理不同的程序,执行不同的数据集。对称多处理系统共享公共资源,如输入设备、输出设备、内存等。

    # 问题

    所有处理器相同,没有考虑不同任务要求处理器的差异性

    # 多处理器环境的 3 种内存共享模型

    # UMA

    UMA(Uniform Memory Access,一致性存储器访问),实际上就是 SMP 。其中,一致性指的就是在任意时刻,多个处理器只能为内存的每个数据保存或共享一个唯一的数值。它的结构特征就是多处理器共享一个集中式存储器,每个处理器访问存储器所需的时间一致,工作负载能够均匀的分配到所有可用处理器上,极大地提高了整个系统的数据处理能力。但是,如果多个处理器同时请求访问共享资源时,就会引发资源竞争,需要软硬件实现加锁机制来解决这个问题。

    因此,这样的架构设计无法拥有良好的处理器数量扩展性,因为共享内存的资源竞态总是存在的,处理器利用率最好的情况只能停留在 2 到 4 颗。

    # NUMA

    NUMA(Non-Uniform Memory Access,非一致性存储器访问)架构优化了 SMP 架构扩展性差以及存储器带宽窄的问题。采用了分布式存储器,将处理器和存储器划分为不同的节点(NUMA Node),每个节点都包含了若干的处理器与内存资源。多节点的设计有效提高了存储器的带宽和处理器的扩展性。

    另一方面,NUMA 节点的处理器可以访问到整体存储器。按照节点内外,内存被分为:

    • 节点内部的本地内存
    • 节点外部的远程内存

    当处理器访问本地内存时,走的是内部总线,当处理器访问远程内存时,走的是主板上的 QPI 互联模块。访问前者的速度要远快于后者,NUMA(非一致性存储器访问)因此而得名。

    # 问题

    各个处理器访问自己的内存速度很快,但如果出现需要跨处理器访存,即访问另一处理器的内存时会非常缓慢

    # COMA

    Cache-only Memory Access (COMA) 是 ccNUMA (Cache Coherent NUMA, 缓存一致性 NUMA) 的一种演变。COMA 可以看成是专用的 NUMA,将 NUMA 中的分布式内存用高速缓存来取代。全局地址空间由高速缓存组成。访问远端的高速缓存借助分布式高速缓存目录进行。

    # 集群系统

    类似多处理系统,集群系统时多个机器,每台机器有自己的操作系统,并行工作。

    • Usually sharing storage via a storage-area network (SAN)

    • Provides a high-availability service which survives failures

      • Asymmetric clustering has one machine in hot-standby mode
      • Symmetric clustering has multiple nodes running applications, monitoring each other
    • Some clusters are for high-performance computing (HPC)

      • Applications must be written to use parallelization
    • Some have distributed lock manager (DLM) to avoid conflicting operations

    # 操作系统的结构

    不同结构操作系统举例:

    • 大内核 / 宏内核 / 单内核: Windows、Android、iOS、UNIX、Linux、MacOS
    • 微内核:Windows NT、Mach、Fuchsia、鸿蒙
    • 模块化:Solaris
    操作系统体系结构区别优点缺点
    大内核 / 宏内核 / 单内核将操作系统的主要功能模块都作为系统内核,运行在核心态高性能内核代码庞大,结构混乱,难以维护
    微内核只把最基本的功能保留在内核内核功能少,结构清晰,方便维护需要频繁地在核心态和用户态之间切换,性能低
    特性、思想优点缺点
    分层结构内核分多层,每层可单向调用更低一层提供的接1. 便于调试和验证,自底向上逐层调试验证1. 仅可调用相邻低层,难以合理定义各层的边界
    2. 效率低,不可跨层调用,系统调用执行时间长2. 效率低,不可跨层调用,系统调用执行时间长
    模块化将内核划分为多个模块,各模块之间相互协作。内核 = 主模块 + 可加载内核模块1. 模块间逻辑清晰易于维护,确定模块间接口后即可多模块同时开发1. 模块间的接口定义未必合理、实用
    主模块:只负责核心功能,如进程调度、内存管2. 支持动态加载新的内核模块(如:安装设备驱动程序、安装新的文件系统模块到内核),增强 OS 适应性2. 模块间相互依赖,更难调试和验证
    可加载内核模块:可以动态加载新模块到内核,而无需重新编译整个内核3. 任何模块都可以直接调用其他模块,无需采用消息传递进行通信,效率高
    宏内核 (大内核)所有的系统功能都放在内核里(大内核结构的 OS 通常也采用了 " 模块化” 的设计思想)性能高,内核内部各种功能都可以直接相互调用1. 内核庞大功能复杂,难以维护
    2. 大内核中某个功能模块出错,就可能导致整个 系统崩溃
    微内核只把中断、原语、进程通信等最核心的功能放入内核。进程管理、文件管理、设备管理等功能以用户进程的形式运行在用户态1. 内核小功能少、易于维护,内核可靠性高1. 性能低,需要频繁的切换 用户态 / 核心态。用户态下的各功能模块不可以直接相互调用,只能通过内核的 "消息传递" 来间接通信
    2. 内核外的某个功能模块出错不会导致整个系统 崩溃2. 用户态下的各功能模块不可以直接相互调用,只能通过内核的 "消息传递" 来间接通信
    外核 (exokernel)内核负责进程调度、进程通信等功能,外核负责为用户进程分配未经抽象的硬件资源,且由外核负责保证资源使用安全1. 外核可直接给用户进程分配 "不虚拟、不抽象" 的硬件资源,使用户进程可以更灵活的使用硬件1. 降低了系统的一致性
    2. 减少了虚拟硬件资源的 “映射层”,提升效率2. 使系统变得更复杂

    # 非内核功能(如 GUI)

    操作系统的非内核功能运行在用户态

    # (大) 内核

    内核是操作系统最基本、最核心的部分。

    实现操作系统内核功能的那些程序就是内核程序。

    操作系统内核需要运行在内核态

    # 对系统资源进行管理的功能

    大内核结构中运行在内核态,所以大内核结构操作系统变态不那么频繁

    微内核结构中运行在用户态,所以微内核结构操作系统变态更加频繁

    # 进程管理

    # 存储器管理

    # 设备管理等功能

    # (微内核)

    # 时钟管理

    实现计时功能

    # 中断处理

    负责实现中断机制

    # 原语(设备驱动、CPU 切换等)

    是一种特殊的程序,具有如下特点:

    • 处于操作系统最底层,是最接近硬件的部分
    • 这种程序的运行具有原子性 —— 其运行只能一气呵成,不可中断
    • 运行时间较短、调用频繁

    # 操作系统的引导

    操作系统引导过程

    操作系统引导过程

    1. CPU 从一个特定主存地址开始,取指令,执行 ROM 中的引导程序(先进行硬件自检,再开机)
    2. 将磁盘的第一块 —— 主引导记录 读入内存,执行磁盘引导程序,扫描分区表
    3. 从活动分区(又称主分区,即安装了操作系统的分区)读入分区引导记录,执行其中的程序
    4. 从根目录下找到完整的操作系统初始化程序(即启动管理器)并执行,完成 “开机” 的一系列动作
      1. 加载内核
        • BIOS
        • UEFI,逐渐取代 BIOS,加载更简洁
        • GRUB,更加复杂,可以选择加载不同的内核
      2. 启动系统守护进程(在内核之外提供的服务)
      3. 启动内核中断驱动(硬件和软件)
        1. 启动设备的硬件中断
        2. 启动软件中断(异常或陷阱)
          • 软件错误(例如,除零)
          • 操作系统服务请求 - 系统调用
          • 故障处理:其他进程问题包括无限循环、进程相互修改,甚至修改操作系统;

    不同体系结构下,不同 OS 启动也有不同:

    x86 需要兼顾 “历史包袱”,需要兼容 32 位

    ARM

    完整的操作系统初始化程序(即启动管理器)可在根目录下找到

    Eg : windows 操作系统完整的开机初始化程序在 根目录/Windows/Boot

    # 双 / 多 模式操作

    # 双模式操作

    即经典的内核态、用户态两个模式

    # 多重模式操作

    两模式扩展到多模式,典型场景是虚拟化

    支持虚拟化技术的 CPU 有一种单独模式,用于表示虚拟机管理器 VMM;

    这种模式的特权多于用户,但少于内核;

    多重模式执行

    # 虚拟机

    虚拟机:使用虚拟化技术,将一台物理机器虚拟化多台虚拟机器(Virtual Machine, VM),每个虚拟机器都可以独立运行一个操作系统

    同义术语:

    • 虚拟机管理程序
    • 虚拟机监控程序
    • Virtual Machine Monitor
    • Hypervisor

    当源 CPU 类型不同于目标类型(例如 PowerPC 到 Intel x86)时使用仿真

    当计算机语言未编译本机代码时进行解释

    # 第一类 VMM

    直接运行在硬件上

    第一类VMM

    KVM(Kernel-based Virtual Machine)是一种基于硬件虚拟化扩展的完全虚拟化技术。使用 KVM,虚拟机可以在真实硬件上运行,并获得对硬件的直接访问。每个 KVM 虚拟机都被视为一个独立的虚拟计算机,具有自己的虚拟硬件,包括虚拟的 CPU、内存、磁盘和网络接口等。

    WSL(Windows Subsystem for Linux)是一项由 Microsoft 开发的技术,旨在在 Windows 操作系统上运行本地的 Linux 环境。WSL 提供了一个兼容层,允许在 Windows 上运行原生的 Linux 二进制文件(如命令行工具、实用程序和应用程序)。在 WSL 中,并没有采用传统的虚拟化技术来创建完全虚拟化的虚拟机。相反,WSL 利用 Windows 操作系统的内核功能,通过在用户模式和内核模式之间进行转换,提供了对 Linux 系统调用的兼容支持。

    # 第二类 VMM

    运行在宿主操作系统上

    第二类VMM.png

    LXC(Linux Containers)是一种操作系统级虚拟化技术,也被称为容器化。使用 LXC,虚拟机实际上是在宿主机的操作系统内运行的,共享宿主机的内核。这些虚拟机被称为容器,它们与宿主机共享相同的内核和一些系统资源。相比于 KVM,LXC 提供了更轻量级的虚拟化解决方案,更适合运行基于 Linux 的应用程序。

    VMware:完全虚拟化

    VirtualBox:“半虚拟化”

    PVE 中的虚拟机就是 KVM 技术实现,而容器 CT 就是 LXC 技术实现

    # 两类 VMM 的对比

    第一类 VMM第二类 VMM
    对物理资源的控制权直接运行在硬件之上,能直接控制和分配物理资 源运行在 Host OS 之上,依赖于 Host OS 为其分配物理资源
    资源分配方式在安装 Guest OS 时,VMM 要在原本的硬盘上自行分配存储空间,类似于 “外核” 的分配方式,分配未经抽象的物理硬件GuestOS 拥有自己的虚拟磁盘,该盘实际上是 Host OS 文件系统中的一个大文件。GuestOS 分配到的内存是虚拟内存
    性能性能更好性能更差,需要 HostOS 作为 “中介 "
    可支持的虚拟机数量更多,不需要和 Host OS 竞争资源,相同的硬件资源可以支持更多的虚拟机更少,Host OS 本身需要使用物理资源,Host OS 上运行的其他进程也需要物理资源
    虚拟机的可迁移性更差更好,只需导出虚拟机镜像文件即可迁移到另一台 HostOS 上,商业化应用更广泛
    运行模式第一类 VMM 运行在最高特权级(Ring O),可以执行最高特权的指令;上层操作系统内核运行在次高特权级(Ring 1);上层操作系统非内核程序程序运行在次高特权级(Ring 1)第二类 VMM 部分运行在用户态、部分运行在内核态。GuestOS 发出的系统调用会被 VMM 截获,并转化为 VMM 对 HostOS 的系统调用
    Edited on Views times
    \ No newline at end of file +操作系统绪论 - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 操作系统的定义

    实际上操作系统没有普遍接受的定义

    操作系统 (Operating System,OS)是指控制管理整个计算机系统的硬件和软件资源,并合理地组织调度计算机的工作资源分配;以提供给用户其他软件方便的接口环境;它是计算机系统中最基本的系统软件

    # 操作系统的功能和目标

    # 作为系统资源的管理者

    # 提供的功能

    • 处理机管理

    • 存储器管理

    • 文件管理

    • 设备管理

    # 目标

    安全、高效

    # 向上层提供方便易用的服务

    # GUI

    (Graphical User Interface) 图形化用户接口

    # CLI

    (Command Line Interface) 命令行接口

    • 联机命令接口 = 交互式命令接口
    • 脱机命令接口 = 批处理命令接口

    # 程序接口

    可以在程序中进行系统调用来使用程序接口。普通用户不能直接使用程序接口,只能通过程序代码间接使用。

    软件和程序员才能使用程序接口

    封装思想:操作系统把一些丑陋的硬件功能封装成简单易用的服务,使用户能更方便地使用计算机,用户无需关心底层硬件的原理,只需要对操作系统发出命令即可

    # 作为最接近硬件的层次

    目标:实现对硬件机器的拓展

    没有任何软件支持的计算机称为裸机。在裸机上安装的操作系统,可以提供资源管理功能和方便用户的服务功能,将裸机改造成功能更强、使用更方便的机器

    通常把覆盖了软件的机器成为扩充机器,又称之虚拟机

    # 操作系统的特征

    # 并发

    并发:指两个或多个事件在同一时间间隔内发生。这些事件宏观上是同时发生的,但微观上是交替发生的

    操作系统的并发性:计算机系统中 “同时” 运行着多个程序,这些程序宏观上看是同时运行着的,而微观上看是交替运行的。

    操作系统就是伴随着 “多道程序技术” 而出现的。因此,操作系统和程序并发是一起诞生的。

    注意(重要考点):单核 CPU 同一时刻只能执行一个程序,各个程序只能并发地执行多核 CPU 同一时刻可以同时执行多个程序,多个程序可以并行地执行

    # 共享

    共享即资源共享,是指系统中的资源可供内存中多个并发执行的进程共同使用。

    # 互斥共享方式

    系统中的某些资源,虽然可以提供给多个进程使用,但一个时间段内只允许一个进程访问该资源

    # 同时共享方式

    系统中的某些资源,允许一个时间段内由多个进程 “同时” 对它们进行访问

    所谓的 “同时” 往往是宏观上的,而在微观上,这些进程可能是交替地对该资源进行访问的(即分时共享)

    并发共享 的关系是相互依存,不可分割

    如果失去并发性,则系统中只有一个程序正在运行,则共享性失去存在的意义;

    如果失去共享性,则不同的程序不能同时访问硬盘资源,就无法实现同时发送文件,也就无法并发

    # 虚拟

    虚拟是指把一个物理上的实体变为若干个逻辑上的对应物。物理实体(前者)是实际存在的,而逻辑上对应物(后者)是用户感受到的。

    # 空分复用技术

    如虚拟存储器技术

    # 时分复用技术

    如虚拟处理器

    并发共享 的关系是相互依存,不可分割

    如果失去了并发性,则一个时间段内系统中只需运行一道程序,那么就失去了实现虚拟性的意义了。因此,没有并发性,就谈不上虚拟性

    # 异步

    异步指,在多道程序环境下,允许多个程序并发执行,但由于资源有限,进程的执行不是一贯到底的,而是走走停停,以不可预知的速度向前推进,这就是进程的异步性。

    # 操作系统的发展和分类

    # 手工操作阶段

    主要缺点:

    用户独占全机、人机速度矛盾导致资源利用率极低

    # 批处理阶段

    引入脱机输入 / 输出技术(用外围机 + 磁带完成),并由监督程序负责控制作业的输入、输出,通过外围机把程序提前存到磁带里

    目标是追求吞吐量,提高资源利用率

    # 脱机

    在主机以外的设备上进行输入 / 输出操作,需要时再送主机处理。

    # 批处理 bash

    后端运行的作业,开始运行之后就不再需要任何交互。由于无法交互控制程序,所以还不算真正意义上的操作系统。

    # 单道批处理系统

    也叫简单批处理系统。已经出现操作系统的一些特征:

    • 用户态、内核态(monitor)
    • 时钟

    主要优点:

    缓解了一定程度的人机速度矛盾,资源利用率有所提升。


    主要缺点:

    内存中仅能有一道程序运行,只有该程序运行结束之后才能调入下一道程序。

    CPU 有大量的时间是在空闲等待 1/0 完成。资源利用率依然很低

    没有人机交互功能。

    # 多道程序设计

    多道程序的运行环境比单道程序等更加复杂。

    特点:

    • 多道 —— 计算机内存中同时存放多道相互独立的程序
    • 宏观上并行
    • 微观上串行
    • 非封闭、非顺序 —— 引入多道程序后,程序的执行就失去了封闭性和顺序性
    • 相互制约 —— 程序执行因为共享资源以及相互协同的原因相互制约
    • 断续性 —— 考虑到竞争的公平性,程序的执行是断续的

    多道程序设计的实现需要解决如下问题:

    • 处理机的分配
    • 多道程序的内存分配
    • I/O 设备的分配
    • 如何组织和存放大量的程序和数据,以方便用户使用并保证一致性安全性
      • 一定需要实现虚拟存储吗?—— 不一定,早期多道批处理操作系统将所有进程数据全部调入主存

    这些问题的处理也会增加系统开销。

    多道程序设计开始引入中断,目的是使 CPU 和 I/O 设备并行运行,提高资源利用率。

    # 多道批处理系统(操作系统开始出现)

    在批处理系统中采用多道程序设计就形成了多道批处理系统。

    每次往内存中读入多道程序,操作系统正式诞生,用于支持多道程序并发运行。


    主要优点:

    多道程序并发执行,共享计算机资源。资源利用率大幅提升,CPU 和其他资源更能保持 “忙碌” 状态,系统吞吐量增大。


    主要缺点:

    用户响应时间长,仍然没有人机交互功能(用户提交自己的作业之后就只能等待计算机处理完成,中间不能控制自己的作业执行。eg:无法调试程序 / 无法在程序运行过程中输入一些参数)

    # 分时操作系统

    计算机以时间片为单位轮流为各个用户 / 作业服务,各个用户可通过终端与计算机进行交互。

    目标是追求时延(latency)


    主要特征:

    • 同时性、也叫多路性
    • 交互性
    • 独立性
    • 及时性

    主要优点:

    用户请求可以被即时响应,解决了人机交互问题。允许多个用户同时使用一台计算机,并且用户对计算机的操作相互独立,感受不到别人的存在。


    主要缺点:

    不能优先处理一些紧急任务。操作系统对各个用户 / 作业都是完全公平的,循环地为每个用户 / 作业服务一个时间片,不区分任务的紧急性。

    # 实时操作系统

    主要优点:

    能够优先响应一些紧急任务,某些紧急任务不需时间片排队。

    在实时操作系统的控制下,计算机系统接收到外部信号后及时进行处理,并且要在严格的时限内处理完事件。实时操作系统的主要特点是及时性可靠性

    # 硬实时系统

    必须在绝对严格的规定时间内完成处理

    eg: 导弹控制系统、自动驾驶系统

    # 软实时系统

    能接受偶尔违反时间规定

    如:12306 火车订票系统

    # 网络操作系统

    是伴随着计算机网络的发展而诞生的,能把网络中各个计算机有机地结合起来,实现数据传送等功能,实现网络中各种资源的共享(如文件共享)和各台计算机之间的通信。(如:Windows NT 就是一种典型的网络操作系统,网站服务器就可以使用)

    # 分布式操作系统

    主要特点是分布性和并行性。系统中的各台计算机地位相同,任何工作都可以分布在这些计算机上,由它们并行、协同完成这个任务。

    # 个人计算机操作系统

    eg: windows、macos

    # 操作系统服务

    • 资源分配
    • 记账
    • 日志服务

    # 内核

    内核是操作系统最重要最核心的部分,也是最接近硬件的部分

    甚至可以説,一个操作系統只要有内核就够了(eg:Docker-> 仅需 Linux 内核)

    操作系统的功能未必都在内核中,如图形化用户界面 GUI

    由很多内核程序组成了 “操作系统内核”,或简称 “内核、(Kernel)”

    # 内核程序 —— 运行在内核态

    微软、苹果有一帮人负责实现操作系统,他们写的是 “内核程序”。操作系统内核作为 “管理者”,有时会让 CPU 执行一些 “特权指令”,如:内存清零指令。这些指令影响重大,只允许 “管理者”—— 即操作系统内核来使用

    普通程序员写的程序就是 “应用程序”,应用程序只能使用 “非特权指令”,如:加法指令、减法指令等

    在 CPU 设计和生产的时候就划分了特权指令和非特权指令,因此 CPU 执行一条指令前就能判断出其类型

    CPU 中有一个寄存器叫 程序状态字寄存器(PSW),其中有个二进制位,1 表示 “内核态”,0 表示 “用户态”

    # 内核态 —— 运行内核程序

    CPU 有两种状态,“内核态” 和 “用户态”。处于内核态时,说明此时正在运行的是内核程序,此时可以执行特权指令处于用户态时,说明此时正在运行的是应用程序,此时只能执行非特权指令

    内核态 = 核心态 = 管态;用户态 = 目态

    # 内核态和用户态的切换 (变态)

    变态的过程是有成本的,要消耗不少时间,频繁地变态会降低系统性能

    # 内核态 -> 用户态

    执行一条特权指令一一修改 PSW 的标志位为 “用户态”,这个动作意味着操作系统将主动让出 CPU 使用权

    # 用户态 -> 内核态

    由 “中断” 引发,硬件自动完成变态过程,触发中断信号意味着操作系统将强行夺回 CPU 的使用权。除了非法使用特权指令之外,还有很多事件会触发中断信号。一个共性是,旦凡需要操作系统介入的地方,都会触发中断信号

    # 系统调用

    对象与系统调用、库函数的关系
    普通应用程序可直接进行系统调用,也可使用库函数。 有的库函数涉及系统调用,有的不涉及
    编程语言向上提供库函数。有时会将系统调用封装成库函数,以隐藏系统调用的一些细节,使程序员编程更加方便。
    操作系统向上提供系统调用,使得上层程序能请求内核的服务
    裸机

    不涉及系统调用的库函数.eg

    • “取绝对值” 的函数

    涉及系统调用的库函数.eg

    • “创建一个新文件” 的函数

    # 按功能分类

    • 设备管理 —— 完成设备请求 / 释放 / 启动 等功能
    • 文件管理 —— 完成文件读 / 写 / 创建 / 删除 等功能
    • 进程控制 —— 完成进程创建 / 撒销 / 阻塞 / 唤醒 等功能
    • 进程通信 —— 完成进程之间的 消息传递 / 信号传递 等功能
    • 内存管理 —— 完成内存分配 / 回收 等功能

    应用程序通过系统调用请求操作系统的服务。而系统中的各种共享资源都由操作系统内核统一掌管,因此凡是与共享资源有关的操作(如存储分配、1/0 操作、文件管理等),都必须通过系统调用的方式向操作系统内核提出服务请求,由操作系统内核代为完成。这样可以保证系统的稳定性和安全性,防止用户进行非法操作。

    # 系统调用过程

    1. 传递系统调用参数
      • 寄存器传递参数
      • 内存中存储在块或表中的参数,以及作为参数在寄存器中传递的块地址
        • linux 和 solaris 采用这种方式
      • 由程序放置或推送到堆栈上并由操作系统弹出堆栈到参数

    块和堆栈的方式传递参数大小不受限制

    1. 执行陷入指令(用户态)
    2. 执行相应的内请求核程序处理系统调用(内核态)
    3. 返回应用程序

    陷入指令 = trap 指令 = 访管指令

    陷入指令是在用户态执行的,执行陷入指令之后立即引发一个内中断,使 CPU 进入内核态

    发出系统调用请求是在用户态,而对系统调用的相应处理内核态下进行

    # 计算计组成原理相关基础概念

    # 中断和异常

    参考这个

    没有中断就没有并发,也就没有操作系统

    中断是让操作系统内核夺回 CPU 使用权的唯一途径,使 CPU 由用户态变为内核态

    通用概念产生 原因AArch64x86-64
    中断硬件 异步异步异常 (重置 / 中断)中断 (可屏蔽 / 不可屏蔽)
    异常软件 同步同步异常 (终止 / 异常指令)异常 (Fault/Trap/Abort)

    中断

    (广义上的中断包括)Interrupt Handlers used in all 3 scenarios :

    • HW -> CPU -> Kernel: Classic HW Interrupt (狭义上的中断)
    • User -> Kernel: SW Interrupt
    • CPU -> Kernel: Exception

    # 其他类型 (非狭义) 中断

    • SW Interrupt
      • Interrupts caused by the execution of a software instruction: INT<interrupt_number>
      • Used by the system call interrupt()
      • Initiated by the running (user level) process
      • Cause current processing to be interrupted and transfers control to the corresponding interrupt handler in the kernel
    • Exceptions
      • Initiated by processor hardware itself
      • Example: divide by zero
      • Cause a transfer of control to the kernel to handle the exception (like a software interrupt)

    # Linux 系统的中断

    • 在中断处理中做尽量少的事
    • 推迟非关键行为

    结构:

    • Top half :马上做,找到 handler

      • 最小的、公共行为

        • 保存寄存器、屏蔽其他中断
        • 恢复寄存器,返回原来场景

        最重要:调用合适的由硬件驱动提供的中断处理 handler

        因为中断被屏蔽,所以不要做太多事情 (时间、空间)

        使用将请求放入队列,或者设置标志位将其他处理推迟到 bottom half

        现代处理器中,多个 I/O 设备共享一个 IRQ 和中断向量 v 多个 ISR (interrupt service routines) 可以结合在一个向量上
        调用每个设备对应该 IRQ 的 ISR

    • Bottom half : 推迟处理 (softirq, tasklets, 工作队列,内核线程)

      • 提供一些推迟完成任务的机制

      • softirqs

      • tasklets (建立在 softirqs 之上)

      • 工作队列

      • 内核线程

      这些工作可以被中断

    linux中断

    中断处理没有进程上下文!

    中断 (和异常相比) 和具体的某条指令无关

    也和中断时正在跑的进程、用户程序无关

    中断处理 handler 不能睡眠!

    中断处理的一些约束条件:

    • 不能睡眠
      • 或者调用可能会睡眠的任务
    • 不能调用 schedule () 调度
    • 不能释放信号或调用可能睡眠的操作
    • 不能和用户地址空间交换数据

    # 中断向量表

    每一个处理器有一个中断向量表,由操作系统内核建立,指向处理不同时间的中断处理程序入口

    结构:

    中断向量表

    # 需要中断的场景举例

    Hardware devices may need asynchronous and immediate service. For example:

    • Timer interrupt: Timers and time-dependent activities need to be updated with the passage of time at precise intervals
    • Network interrupt: The network card interrupts the CPU when data arrives from the network
    • I/O device interrupt: I/O devices (such as mouse and keyboard) issue hardware interrupts when they have input (e.g., a new character or mouse click)

    # 多核系统上的中断

    Howareinterruptshandledonmulticoremachines?

    • On x86 systems each CPU gets its own local Advanced Programmable Interrupt Controller (APIC). They are wired in a way that allows routing device interrupts to any selected local APIC.
    • The OS can program the APICs to determine which interrupts get routed to which CPUs.
    • The default (unless OS states otherwise) is to route all interrupts to processor 0

    # 陷入指令

    执行 “陷入指令”,意味着应用程序主动地将 CPU 控制权还给操作系统内核。“系统调用” 就是通过陷入指令完成的

    # I/O (输入输出系统)

    # BUS (总线)

    # 并行系统

    # 多核处理器

    # 多处理器系统

    又称并行系统、紧耦合系统

    区分 多处理器系统多核系统 ,实际计算机往往既是多核的也是多处理器的

    多核和多处理器系统都用于加速计算过程。多核包含在单个 CPU 中的多个核或处理单元。多处理器由多个 CPU 组成。多核处理器不需要像多处理器那样复杂的配置。相反,多处理器更可靠并且能够运行许多程序。

    # 三个优点一个缺点???

    老师出的作业题😅难道本来是想问多道批处理系统?而且那个也是单处理器系统啊)

    • 增加吞吐量,并行处理?
    • 规模经济,降低成本,比多核处理器便宜?
    • 增加可靠性?
    • ???

    # 非对称多处理器系统 (AMP)

    非对称多处理是指一种计算机结构,其中有多个处理器,但它们并不都是一样的。这意味着,一个 CPU 可能正在处理操作系统代码,而另一个 CPU 正在执行输入和输出工作。它使用两个或更多的处理器由一个主处理器处理。所有的 CPU 都是相互连接的,但不是自我调度的。它用于根据任务的优先级和重要性将特定任务安排到一个 CPU。

    非对称处理器的应用场景:被用于嵌入式系统,在这些系统中,特定任务需要并发执行,但系统不需要像对称多处理系统那样的通用计算能力

    # 对称多处理器系统 (SMP)

    SMP(Sysmmetric Multi-Processor System,对称多处理器),它由多个具有对称关系的处理器组成。所谓对称,即处理器之间没有主从之分。SMP 架构使得一台计算机不再由单个 CPU 组成,而是由两个或更多的处理器连接到一个共享的主存储器上,并能够共同执行任务。这些处理器通常是相同的,能够运行分配给它们的任何任务。这可以提高性能,因为任务可以在处理器之间分配,使其更快完成。

    对称多处理将多个 CPU 应用于一个任务,以并行和更快的方式完成。因此,在对称多处理系统中,两个或更多的 CPU 被连接到一个共享的主存储器。同时,所有这些 CPU 都可以完全访问输入和输出设备。在对称多处理系统中,操作系统认为所有的处理器都是平等的。

    对称处理器 (SMP) 架构同时也是 UMA 架构

    SMP 是一个紧密耦合的多处理器系统,它有一组独立运行的相同的 CPU。每个 CPU 处理不同的程序,执行不同的数据集。对称多处理系统共享公共资源,如输入设备、输出设备、内存等。

    # 问题

    所有处理器相同,没有考虑不同任务要求处理器的差异性

    # 多处理器环境的 3 种内存共享模型

    # UMA

    UMA(Uniform Memory Access,一致性存储器访问),实际上就是 SMP 。其中,一致性指的就是在任意时刻,多个处理器只能为内存的每个数据保存或共享一个唯一的数值。它的结构特征就是多处理器共享一个集中式存储器,每个处理器访问存储器所需的时间一致,工作负载能够均匀的分配到所有可用处理器上,极大地提高了整个系统的数据处理能力。但是,如果多个处理器同时请求访问共享资源时,就会引发资源竞争,需要软硬件实现加锁机制来解决这个问题。

    因此,这样的架构设计无法拥有良好的处理器数量扩展性,因为共享内存的资源竞态总是存在的,处理器利用率最好的情况只能停留在 2 到 4 颗。

    # NUMA

    NUMA(Non-Uniform Memory Access,非一致性存储器访问)架构优化了 SMP 架构扩展性差以及存储器带宽窄的问题。采用了分布式存储器,将处理器和存储器划分为不同的节点(NUMA Node),每个节点都包含了若干的处理器与内存资源。多节点的设计有效提高了存储器的带宽和处理器的扩展性。

    另一方面,NUMA 节点的处理器可以访问到整体存储器。按照节点内外,内存被分为:

    • 节点内部的本地内存
    • 节点外部的远程内存

    当处理器访问本地内存时,走的是内部总线,当处理器访问远程内存时,走的是主板上的 QPI 互联模块。访问前者的速度要远快于后者,NUMA(非一致性存储器访问)因此而得名。

    # 问题

    各个处理器访问自己的内存速度很快,但如果出现需要跨处理器访存,即访问另一处理器的内存时会非常缓慢

    # COMA

    Cache-only Memory Access (COMA) 是 ccNUMA (Cache Coherent NUMA, 缓存一致性 NUMA) 的一种演变。COMA 可以看成是专用的 NUMA,将 NUMA 中的分布式内存用高速缓存来取代。全局地址空间由高速缓存组成。访问远端的高速缓存借助分布式高速缓存目录进行。

    # 集群系统

    类似多处理系统,集群系统时多个机器,每台机器有自己的操作系统,并行工作。

    • Usually sharing storage via a storage-area network (SAN)

    • Provides a high-availability service which survives failures

      • Asymmetric clustering has one machine in hot-standby mode
      • Symmetric clustering has multiple nodes running applications, monitoring each other
    • Some clusters are for high-performance computing (HPC)

      • Applications must be written to use parallelization
    • Some have distributed lock manager (DLM) to avoid conflicting operations

    # 操作系统的结构

    不同结构操作系统举例:

    • 大内核 / 宏内核 / 单内核: Windows、Android、iOS、UNIX、Linux、MacOS
    • 微内核:Windows NT、Mach、Fuchsia、鸿蒙
    • 模块化:Solaris
    操作系统体系结构区别优点缺点
    大内核 / 宏内核 / 单内核将操作系统的主要功能模块都作为系统内核,运行在核心态高性能内核代码庞大,结构混乱,难以维护
    微内核只把最基本的功能保留在内核内核功能少,结构清晰,方便维护需要频繁地在核心态和用户态之间切换,性能低
    特性、思想优点缺点
    分层结构内核分多层,每层可单向调用更低一层提供的接1. 便于调试和验证,自底向上逐层调试验证1. 仅可调用相邻低层,难以合理定义各层的边界
    2. 效率低,不可跨层调用,系统调用执行时间长2. 效率低,不可跨层调用,系统调用执行时间长
    模块化将内核划分为多个模块,各模块之间相互协作。内核 = 主模块 + 可加载内核模块1. 模块间逻辑清晰易于维护,确定模块间接口后即可多模块同时开发1. 模块间的接口定义未必合理、实用
    主模块:只负责核心功能,如进程调度、内存管2. 支持动态加载新的内核模块(如:安装设备驱动程序、安装新的文件系统模块到内核),增强 OS 适应性2. 模块间相互依赖,更难调试和验证
    可加载内核模块:可以动态加载新模块到内核,而无需重新编译整个内核3. 任何模块都可以直接调用其他模块,无需采用消息传递进行通信,效率高
    宏内核 (大内核)所有的系统功能都放在内核里(大内核结构的 OS 通常也采用了 " 模块化” 的设计思想)性能高,内核内部各种功能都可以直接相互调用1. 内核庞大功能复杂,难以维护
    2. 大内核中某个功能模块出错,就可能导致整个 系统崩溃
    微内核只把中断、原语、进程通信等最核心的功能放入内核。进程管理、文件管理、设备管理等功能以用户进程的形式运行在用户态1. 内核小功能少、易于维护,内核可靠性高1. 性能低,需要频繁的切换 用户态 / 核心态。用户态下的各功能模块不可以直接相互调用,只能通过内核的 "消息传递" 来间接通信
    2. 内核外的某个功能模块出错不会导致整个系统 崩溃2. 用户态下的各功能模块不可以直接相互调用,只能通过内核的 "消息传递" 来间接通信
    外核 (exokernel)内核负责进程调度、进程通信等功能,外核负责为用户进程分配未经抽象的硬件资源,且由外核负责保证资源使用安全1. 外核可直接给用户进程分配 "不虚拟、不抽象" 的硬件资源,使用户进程可以更灵活的使用硬件1. 降低了系统的一致性
    2. 减少了虚拟硬件资源的 “映射层”,提升效率2. 使系统变得更复杂

    # 非内核功能(如 GUI)

    操作系统的非内核功能运行在用户态

    # (大) 内核

    内核是操作系统最基本、最核心的部分。

    实现操作系统内核功能的那些程序就是内核程序。

    操作系统内核需要运行在内核态

    # 对系统资源进行管理的功能

    大内核结构中运行在内核态,所以大内核结构操作系统变态不那么频繁

    微内核结构中运行在用户态,所以微内核结构操作系统变态更加频繁

    # 进程管理

    # 存储器管理

    # 设备管理等功能

    # (微内核)

    # 时钟管理

    实现计时功能

    # 中断处理

    负责实现中断机制

    # 原语(设备驱动、CPU 切换等)

    是一种特殊的程序,具有如下特点:

    • 处于操作系统最底层,是最接近硬件的部分
    • 这种程序的运行具有原子性 —— 其运行只能一气呵成,不可中断
    • 运行时间较短、调用频繁

    # 操作系统的引导

    操作系统引导过程

    操作系统引导过程

    1. CPU 从一个特定主存地址开始,取指令,执行 ROM 中的引导程序(先进行硬件自检,再开机)
    2. 将磁盘的第一块 —— 主引导记录 读入内存,执行磁盘引导程序,扫描分区表
    3. 从活动分区(又称主分区,即安装了操作系统的分区)读入分区引导记录,执行其中的程序
    4. 从根目录下找到完整的操作系统初始化程序(即启动管理器)并执行,完成 “开机” 的一系列动作
      1. 加载内核
        • BIOS
        • UEFI,逐渐取代 BIOS,加载更简洁
        • GRUB,更加复杂,可以选择加载不同的内核
      2. 启动系统守护进程(在内核之外提供的服务)
      3. 启动内核中断驱动(硬件和软件)
        1. 启动设备的硬件中断
        2. 启动软件中断(异常或陷阱)
          • 软件错误(例如,除零)
          • 操作系统服务请求 - 系统调用
          • 故障处理:其他进程问题包括无限循环、进程相互修改,甚至修改操作系统;

    不同体系结构下,不同 OS 启动也有不同:

    x86 需要兼顾 “历史包袱”,需要兼容 32 位

    ARM

    完整的操作系统初始化程序(即启动管理器)可在根目录下找到

    Eg : windows 操作系统完整的开机初始化程序在 根目录/Windows/Boot

    # 双 / 多 模式操作

    # 双模式操作

    即经典的内核态、用户态两个模式

    # 多重模式操作

    两模式扩展到多模式,典型场景是虚拟化

    支持虚拟化技术的 CPU 有一种单独模式,用于表示虚拟机管理器 VMM;

    这种模式的特权多于用户,但少于内核;

    多重模式执行

    # 虚拟机

    虚拟机:使用虚拟化技术,将一台物理机器虚拟化多台虚拟机器(Virtual Machine, VM),每个虚拟机器都可以独立运行一个操作系统

    同义术语:

    • 虚拟机管理程序
    • 虚拟机监控程序
    • Virtual Machine Monitor
    • Hypervisor

    当源 CPU 类型不同于目标类型(例如 PowerPC 到 Intel x86)时使用仿真

    当计算机语言未编译本机代码时进行解释

    # 第一类 VMM

    直接运行在硬件上

    第一类VMM

    KVM(Kernel-based Virtual Machine)是一种基于硬件虚拟化扩展的完全虚拟化技术。使用 KVM,虚拟机可以在真实硬件上运行,并获得对硬件的直接访问。每个 KVM 虚拟机都被视为一个独立的虚拟计算机,具有自己的虚拟硬件,包括虚拟的 CPU、内存、磁盘和网络接口等。

    WSL(Windows Subsystem for Linux)是一项由 Microsoft 开发的技术,旨在在 Windows 操作系统上运行本地的 Linux 环境。WSL 提供了一个兼容层,允许在 Windows 上运行原生的 Linux 二进制文件(如命令行工具、实用程序和应用程序)。在 WSL 中,并没有采用传统的虚拟化技术来创建完全虚拟化的虚拟机。相反,WSL 利用 Windows 操作系统的内核功能,通过在用户模式和内核模式之间进行转换,提供了对 Linux 系统调用的兼容支持。

    # 第二类 VMM

    运行在宿主操作系统上

    第二类VMM.png

    LXC(Linux Containers)是一种操作系统级虚拟化技术,也被称为容器化。使用 LXC,虚拟机实际上是在宿主机的操作系统内运行的,共享宿主机的内核。这些虚拟机被称为容器,它们与宿主机共享相同的内核和一些系统资源。相比于 KVM,LXC 提供了更轻量级的虚拟化解决方案,更适合运行基于 Linux 的应用程序。

    VMware:完全虚拟化

    VirtualBox:“半虚拟化”

    PVE 中的虚拟机就是 KVM 技术实现,而容器 CT 就是 LXC 技术实现

    # 两类 VMM 的对比

    第一类 VMM第二类 VMM
    对物理资源的控制权直接运行在硬件之上,能直接控制和分配物理资 源运行在 Host OS 之上,依赖于 Host OS 为其分配物理资源
    资源分配方式在安装 Guest OS 时,VMM 要在原本的硬盘上自行分配存储空间,类似于 “外核” 的分配方式,分配未经抽象的物理硬件GuestOS 拥有自己的虚拟磁盘,该盘实际上是 Host OS 文件系统中的一个大文件。GuestOS 分配到的内存是虚拟内存
    性能性能更好性能更差,需要 HostOS 作为 “中介 "
    可支持的虚拟机数量更多,不需要和 Host OS 竞争资源,相同的硬件资源可以支持更多的虚拟机更少,Host OS 本身需要使用物理资源,Host OS 上运行的其他进程也需要物理资源
    虚拟机的可迁移性更差更好,只需导出虚拟机镜像文件即可迁移到另一台 HostOS 上,商业化应用更广泛
    运行模式第一类 VMM 运行在最高特权级(Ring O),可以执行最高特权的指令;上层操作系统内核运行在次高特权级(Ring 1);上层操作系统非内核程序程序运行在次高特权级(Ring 1)第二类 VMM 部分运行在用户态、部分运行在内核态。GuestOS 发出的系统调用会被 VMM 截获,并转化为 VMM 对 HostOS 的系统调用
    Edited on Views times
    \ No newline at end of file diff --git a/cs/os/deadlock/index.html b/cs/os/deadlock/index.html index 704a36bc..68f88c05 100644 --- a/cs/os/deadlock/index.html +++ b/cs/os/deadlock/index.html @@ -1 +1 @@ -死锁 - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # Intro

    死锁的必要条件

    • 互斥:一次只有一个进程使用一个资源,其他进程不能访问分配给其他进程的资源
    • 占有等待 (持有且等待):当一个进程等待其他进程时,继续占有已分配的资源
    • 非抢占:不能强行抢占进程已占有的资源
    • 循环等待:存在一个闭合的进程链,每个进程至少占有此链中下一个进程所需的一个资源前三个只会导致死锁的可能性,而最后一个循环等待是导致死锁的 “最终极条件”

    处理死锁的方法:

    • 确保系统永远不会进入死锁状态
      • 锁预防,在 OS 设计阶段就设计成不会出现死锁
      • 死锁避免,在 OS 运行的时候尽量避免死锁产生条件
    • 允许系统进入死锁状态,然后检测和恢复
    • 忽略该问题,并假装系统中从未发生死锁
      • 绝大多数操作系统所采用的的方法
      • 合理性:死锁的发生实际上是一个小概率性事件

    # 死锁预防

    使死锁的四个必要条件之一无效,破坏任意一个死锁必要条件:

    • 互斥:通常不能否定互斥条件来预防死锁,必须保留不可共享的资源
    • 持有且等待:必须保证每当进程请求资源时,它不会占有任何其他资源
      • 一种实现:在进程开始执行之前请求并分配其所有资源
      • 另一种实现:仅当进程在未分配任何资源时才允许进程请求资源
    • 非抢占
      • 如果持有某些资源的进程请求另一个无法立即分配给它的资源,那么当前持有的所有资源都将被释放 (隐式释放)
      • 被抢占资源被添加到进程正在等待的资源列表中
      • 只有当进程能够恢复其原有的资源以及它所请求的新资源 时,才会重新启动进程
    • 循环等待 (最常见)
      • 强制所有类型的资源进行完全排序,也就是为每个资源 (即互斥锁) 分配一个唯一的编号,并要求每个进程以同一特定顺序 (例如递增) 请求资源
      • 可以反证这样的做法不存在循环等待
      • 如果能够动态获得锁,那么制定一个加锁顺序并不保证死锁预防,因为并不能保证请求资源的顺序是同一特定顺序

    动态获得锁:获得锁的函数与输入的参数有关

    问题

    • 一个进程需要多少资源无法精确预测
    • 副作用是设备使用率低,系统吞吐率低

    # 死锁避免

    死锁避免使用算法动态检查资源分配状态,以确保不存在循环等待条件

    优势:

    • 无须死锁预防中的抢占和回滚进程
    • 与死锁预防相比限制较少

    限制:

    • 必须事先声明每一个进程请求的最大资源;
    • 所讨论的进程必须是无关的,即他们的执行顺序必须没有任何同步要求的限制;
    • 分配的资源数量必须是固定的;
    • 在占有资源时,进程不能退出;

    这些限制使得银行家算法这样的死锁避免算法实际上是不可行的,只能用于理论检测

    # 安全状态

    只有至少存在一个安全序列,系统才处于安全状态

    系统中存在所有进程的序列 <P1,P2,,Pn><P_1,P_2,\cdots,P_n> 在当前分配状态下为安全序列是指:

    对于每个 PiP_iPiP_i 请求的资源可以通过 当前可用资源 + 所有 Pj(j<i)P_j(j<i) 持有的资源 来满足。

    即:

    • 如果 PiP_i 资源需求不能立即可用,则 PiP_i 可以等待所有 PjP_j 完成,释放资源
    • PjP_j 完成时,PiP_i 可以获得所需的资源、执行、返回分配的资源并终止
    • PiP_i 终止时,Pi+1P_{i+1} 可以获得它所需的资源,依此类推

    ⚠️安全状态一定不是死锁状态,非安全状态只是死锁的必要条件,可能导致死锁,但不是死锁的充分条件。

    safe-unsafe-deadlock

    死锁避免两种方法

    • 在进程某次请求资源的时候,如果为其分配资源后系统进入非安全状态,则拒绝分配
    • 进程启动拒绝:在进程初始化启动的时候,如果对某种资源,它的最大需求 + 当前所以进程的需求 > 系统总资源量,则拒绝分配

    # 资源分配图算法

    ⚠️每种资源都只有一个实例(都是唯一)的时候才能依据资源分配图判断死锁是否产生:

    • 资源分配图中没有环,那么系统就没有进程死锁
    • 如果有环,可能存在死锁

    ⚠️必须在系统中预先声明资源;也就是只有当进程 PiP_i 的所有边都为需求边时,才能允许将需求边增加到图中

    # 银行家算法

    ⚠️每个进程都必须先验地给出最大需求资源的数量

    # 数据结构

    n = 进程数, m = 资源类型数。

    Available : 长度为 m 的向量。如果 Available[j]=k ,则有 k 个资源类型 Rj 的实例可用;

    Max : nxm 矩阵,定义每个进程的最大需求。如果 Max[i, j]=k ,那么进程 Pi 最多可以请求 k 个资源类型 Rj 的实例

    Allocation : n×m 矩阵,定义每个进程现在分配的每种资 源类型的实例数量。如果 Allocation[i,j]=k ,则 Pi 当前被分配给了 Rj 的 k 个实例

    Need : nxm 矩阵,表示每个进程还需要的剩余资源。如果 Need[i,j]=k ,那么 Pi 可能还需要 k 个 Rj 实例来完成它的任务 Need[i,j]=Max[i,j]–Allocation[i,j]

    # 算法流程

    判断安全状态:

    judge_safe()

    1. Let Work and Finish be vectors of length m and n, respectively.

      Initialize:

      Work = Available

      Finish [i] = false for i = 0, 1, ..., n- 1

    2. Find an i such that both:

      • (a) Finish [i] = false

      • (b) Needi <= Work

      If no such i exists, go to step 4

    3. Work = Work + Allocationi

      Finish[i] = true

      go to step 2

    4. If Finish [i] == true for all i, then the system is in a safe state

    判断是否允许某次资源请求:

    Requesti = request vector for process Pi. If Requesti [j] = k then process Pi wants k instances of resource type Rj

    1. If Requesti <= Needi go to step 2.

      Otherwise, raise error condition, since process has exceeded its maximum claim

    2. If Requesti <= Available, go to step 3.

      Otherwise Pi must wait, since resources are not available

    3. Pretend to allocate requested resources to Pi by modifying the state as follows:

      • Available = Available – Requesti;

      • Allocationi = Allocationi + Requesti;

      • Needi = Needi – Requesti;

      If judge_safe() => the resources are allocated to Pi

      else => Pi must wait, and the old resource-allocation state is restored

    # 死锁检测与恢复

    # 死锁检测

    允许系统进入死锁状态,死锁检测可以频繁地在每个资源请求发生时进行,也可以检测得少一些,具体取决于发生死锁的可能性。检测出死锁之后进行死锁恢复。

    优点:

    • 尽早地检测死锁情况
    • 算法相对简单

    缺点

    • 频繁的检测会耗费相当多的处理器时间

    何时以及多久调用一次取决于:

    • 死锁可能发生的频率有多高?
    • 需要回滚多少个进程?

    如果任意调用检测算法,资源图中可能会有许多循环,因此我们无法判断是哪些死锁进程 “导致” 了死锁

    死锁检测产生大量的代价,可以周期性检测死锁,或者 CPU 的利用率低于某个阈值时再执行死锁检测

    # 等待图

    在资源分配图的基础上,只画出进程,不再画出资源。

    和资源分配图一样,只适用于所有类型的资源都是唯一的,只有一个实例的情况。

    # 死锁恢复

    • 取消所有的死锁进程,这是实际操作系统最常用的方法
    • 把每个死锁进程回滚到前面定义的某些检查点,并重新启动所有进程,要求在系统中构建回滚和重启机制
    • 连续取消死锁进程直到不再存在死锁,所取消的进程的顺序应基于某种最小代价原则
    • 连续抢占资源直到不再存在死锁,需要使用一种基于代价的选择方法,且需要在抢占后重新调用检测算法

    后两种策略的选择 (受害者) 原则:

    • 目前为止消耗的处理器时间最少
    • 目前为止产生的输出最少
    • 预计剩下的时间最长
    • 目前为止分配的资源总量最少
    • 优先级最低

    实际上 OS 的死锁恢复

    • 重启
    • 随机 sleep
    Edited on Views times
    \ No newline at end of file +死锁 - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # Intro

    死锁的必要条件

    • 互斥:一次只有一个进程使用一个资源,其他进程不能访问分配给其他进程的资源
    • 占有等待 (持有且等待):当一个进程等待其他进程时,继续占有已分配的资源
    • 非抢占:不能强行抢占进程已占有的资源
    • 循环等待:存在一个闭合的进程链,每个进程至少占有此链中下一个进程所需的一个资源前三个只会导致死锁的可能性,而最后一个循环等待是导致死锁的 “最终极条件”

    处理死锁的方法:

    • 确保系统永远不会进入死锁状态
      • 锁预防,在 OS 设计阶段就设计成不会出现死锁
      • 死锁避免,在 OS 运行的时候尽量避免死锁产生条件
    • 允许系统进入死锁状态,然后检测和恢复
    • 忽略该问题,并假装系统中从未发生死锁
      • 绝大多数操作系统所采用的的方法
      • 合理性:死锁的发生实际上是一个小概率性事件

    # 死锁预防

    使死锁的四个必要条件之一无效,破坏任意一个死锁必要条件:

    • 互斥:通常不能否定互斥条件来预防死锁,必须保留不可共享的资源
    • 持有且等待:必须保证每当进程请求资源时,它不会占有任何其他资源
      • 一种实现:在进程开始执行之前请求并分配其所有资源
      • 另一种实现:仅当进程在未分配任何资源时才允许进程请求资源
    • 非抢占
      • 如果持有某些资源的进程请求另一个无法立即分配给它的资源,那么当前持有的所有资源都将被释放 (隐式释放)
      • 被抢占资源被添加到进程正在等待的资源列表中
      • 只有当进程能够恢复其原有的资源以及它所请求的新资源 时,才会重新启动进程
    • 循环等待 (最常见)
      • 强制所有类型的资源进行完全排序,也就是为每个资源 (即互斥锁) 分配一个唯一的编号,并要求每个进程以同一特定顺序 (例如递增) 请求资源
      • 可以反证这样的做法不存在循环等待
      • 如果能够动态获得锁,那么制定一个加锁顺序并不保证死锁预防,因为并不能保证请求资源的顺序是同一特定顺序

    动态获得锁:获得锁的函数与输入的参数有关

    问题

    • 一个进程需要多少资源无法精确预测
    • 副作用是设备使用率低,系统吞吐率低

    # 死锁避免

    死锁避免使用算法动态检查资源分配状态,以确保不存在循环等待条件

    优势:

    • 无须死锁预防中的抢占和回滚进程
    • 与死锁预防相比限制较少

    限制:

    • 必须事先声明每一个进程请求的最大资源;
    • 所讨论的进程必须是无关的,即他们的执行顺序必须没有任何同步要求的限制;
    • 分配的资源数量必须是固定的;
    • 在占有资源时,进程不能退出;

    这些限制使得银行家算法这样的死锁避免算法实际上是不可行的,只能用于理论检测

    # 安全状态

    只有至少存在一个安全序列,系统才处于安全状态

    系统中存在所有进程的序列 <P1,P2,,Pn><P_1,P_2,\cdots,P_n> 在当前分配状态下为安全序列是指:

    对于每个 PiP_iPiP_i 请求的资源可以通过 当前可用资源 + 所有 Pj(j<i)P_j(j<i) 持有的资源 来满足。

    即:

    • 如果 PiP_i 资源需求不能立即可用,则 PiP_i 可以等待所有 PjP_j 完成,释放资源
    • PjP_j 完成时,PiP_i 可以获得所需的资源、执行、返回分配的资源并终止
    • PiP_i 终止时,Pi+1P_{i+1} 可以获得它所需的资源,依此类推

    ⚠️安全状态一定不是死锁状态,非安全状态只是死锁的必要条件,可能导致死锁,但不是死锁的充分条件。

    safe-unsafe-deadlock

    死锁避免两种方法

    • 在进程某次请求资源的时候,如果为其分配资源后系统进入非安全状态,则拒绝分配
    • 进程启动拒绝:在进程初始化启动的时候,如果对某种资源,它的最大需求 + 当前所以进程的需求 > 系统总资源量,则拒绝分配

    # 资源分配图算法

    ⚠️每种资源都只有一个实例(都是唯一)的时候才能依据资源分配图判断死锁是否产生:

    • 资源分配图中没有环,那么系统就没有进程死锁
    • 如果有环,可能存在死锁

    ⚠️必须在系统中预先声明资源;也就是只有当进程 PiP_i 的所有边都为需求边时,才能允许将需求边增加到图中

    # 银行家算法

    ⚠️每个进程都必须先验地给出最大需求资源的数量

    # 数据结构

    n = 进程数, m = 资源类型数。

    Available : 长度为 m 的向量。如果 Available[j]=k ,则有 k 个资源类型 Rj 的实例可用;

    Max : nxm 矩阵,定义每个进程的最大需求。如果 Max[i, j]=k ,那么进程 Pi 最多可以请求 k 个资源类型 Rj 的实例

    Allocation : n×m 矩阵,定义每个进程现在分配的每种资 源类型的实例数量。如果 Allocation[i,j]=k ,则 Pi 当前被分配给了 Rj 的 k 个实例

    Need : nxm 矩阵,表示每个进程还需要的剩余资源。如果 Need[i,j]=k ,那么 Pi 可能还需要 k 个 Rj 实例来完成它的任务 Need[i,j]=Max[i,j]–Allocation[i,j]

    # 算法流程

    判断安全状态:

    judge_safe()

    1. Let Work and Finish be vectors of length m and n, respectively.

      Initialize:

      Work = Available

      Finish [i] = false for i = 0, 1, ..., n- 1

    2. Find an i such that both:

      • (a) Finish [i] = false

      • (b) Needi <= Work

      If no such i exists, go to step 4

    3. Work = Work + Allocationi

      Finish[i] = true

      go to step 2

    4. If Finish [i] == true for all i, then the system is in a safe state

    判断是否允许某次资源请求:

    Requesti = request vector for process Pi. If Requesti [j] = k then process Pi wants k instances of resource type Rj

    1. If Requesti <= Needi go to step 2.

      Otherwise, raise error condition, since process has exceeded its maximum claim

    2. If Requesti <= Available, go to step 3.

      Otherwise Pi must wait, since resources are not available

    3. Pretend to allocate requested resources to Pi by modifying the state as follows:

      • Available = Available – Requesti;

      • Allocationi = Allocationi + Requesti;

      • Needi = Needi – Requesti;

      If judge_safe() => the resources are allocated to Pi

      else => Pi must wait, and the old resource-allocation state is restored

    # 死锁检测与恢复

    # 死锁检测

    允许系统进入死锁状态,死锁检测可以频繁地在每个资源请求发生时进行,也可以检测得少一些,具体取决于发生死锁的可能性。检测出死锁之后进行死锁恢复。

    优点:

    • 尽早地检测死锁情况
    • 算法相对简单

    缺点

    • 频繁的检测会耗费相当多的处理器时间

    何时以及多久调用一次取决于:

    • 死锁可能发生的频率有多高?
    • 需要回滚多少个进程?

    如果任意调用检测算法,资源图中可能会有许多循环,因此我们无法判断是哪些死锁进程 “导致” 了死锁

    死锁检测产生大量的代价,可以周期性检测死锁,或者 CPU 的利用率低于某个阈值时再执行死锁检测

    # 等待图

    在资源分配图的基础上,只画出进程,不再画出资源。

    和资源分配图一样,只适用于所有类型的资源都是唯一的,只有一个实例的情况。

    # 死锁恢复

    • 取消所有的死锁进程,这是实际操作系统最常用的方法
    • 把每个死锁进程回滚到前面定义的某些检查点,并重新启动所有进程,要求在系统中构建回滚和重启机制
    • 连续取消死锁进程直到不再存在死锁,所取消的进程的顺序应基于某种最小代价原则
    • 连续抢占资源直到不再存在死锁,需要使用一种基于代价的选择方法,且需要在抢占后重新调用检测算法

    后两种策略的选择 (受害者) 原则:

    • 目前为止消耗的处理器时间最少
    • 目前为止产生的输出最少
    • 预计剩下的时间最长
    • 目前为止分配的资源总量最少
    • 优先级最低

    实际上 OS 的死锁恢复

    • 重启
    • 随机 sleep
    Edited on Views times
    \ No newline at end of file diff --git a/cs/os/fs/index.html b/cs/os/fs/index.html index 51837ba0..d97ff04a 100644 --- a/cs/os/fs/index.html +++ b/cs/os/fs/index.html @@ -1 +1 @@ -文件系统 - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    磁盘的 block 和文件系统的 block 不是一回事:

    • 磁盘的 block :往往是固定的 512 B;
    • 文件系统的 block 是基本单位,可以自定义,不一定和磁盘的 block 大小相等。
      • 文件系统分配空间的基本单位往往是页面大小 4 KB ,

    文件系统结构:(自上而下分层)

    • application programs,应用程序
    • logical file system,逻辑文件系统
    • file-organization module,文件组织模块
    • basic file system,基本文件系统
    • I/O control,I/O 控制
    • devices,设备

    修改文件要写回磁盘!计算访问磁盘次数的时候要注意写操作比读操作多 1

    # 数据结构

    面向对象

    # FCB

    别名:文件目录项文件控制块

    包含:

    • 指向关联的 inode 的指针
    • 指向父目录、子目录的指针

    # 文件目录

    FCB 的有序集合

    # inode

    别名:索引节点文件描述信息、index node

    ext2ext3 采用 inode 表

    将 FCB 中的一部分文件描述信息分离出来,这些是检索目录时不需要的不必放在内存的,所以只将指向这部分的一个指针放在内存的目录项里。

    采用 inode 可以减少查找文件时的 I/O 信息量,相当于简化了内存中的文件目录项 (FCB),将其中一部分留在磁盘,只保留指向他们的指针。例如 Linux 的文件目录项就只包含文件名和指向 indoe 的指针。

    ⭐️硬连接和软链接与 inode

    • 硬链接文件指向同一个 inode
    • 软链接文件指向不同的 inode,相当于不同的文件,只不过存的是路径

    在文件的物理结构中,则 inode 中就存储了索引文件的主索引表

    直接以数组方式存储的 indoe 表,相当于单级索引;也可以有多级索引

    • 磁盘索引节点,每个文件唯一都有一个
      • 文件主标识符
      • 文件类型
      • 文件存取权限
      • 文件物理地址
      • 文件长度
      • 文件链接计数
        • ⭐️ count ,实现硬链接文件共享,表示链接到本索引节点的用户目录项的数量。只有 count=0 的时候才能删除文件。
      • 文件存取时间
    • 内存索引节点,文件被打开时,磁盘索引节点复制放在内存的副本
      • 索引节点号,用于标识内存索引节点
      • 状态,是否被上锁
      • 访问计数,为 0 时文件关闭
      • 逻辑设备号
      • 链接指针,指向空闲列表或散列队列

    # 打开文件表

    本质上是实现只需要一次访问磁盘定位文件在磁盘的位置,将其存储到内存,后续在内存可以找到文件在磁盘的位置。

    两级表:

    • 整个系统一个表,包含与进程无关的信息,每一个文件条目包含:
      • 文件描述符 (File Descriptor, FD)
        • 这是 Linux 中的名字,Windows 中也叫文件句柄
      • 文件打开次数
        • 变为 0 之后就可以 “关闭文件”,从表中删除该项
      • 文件磁盘位置
      • 访问日期
      • 文件大小
    • 每个进程有一个表,包含与进程有关的信息,每一个文件条目包含:
      • 读写指针
      • 访问权限
      • 指向系统打开文件表的对应文件条目指针

    ⚠️系统只会在第一次打开一个文件的时候使用其文件名在磁盘上定位,然后在内存中创建文件表中的条目,此后就不再需要文件名。

    ⚠️ open 系统调用由用户调用,一个文件可能对某一用户是第一次打开,但是对系统而言不是第一次,只会在用户进程的打开文件表增加条目。

    # 文件保护

    • 口令保护
    • 加密保护
    • 访问控制

    # 文件访问控制表

    可能也有叫做存取控制矩阵之类的 (?)

    为每个文件和目录设置一个文件访问控制表。用来管理不同用户、用户组对该文件或目录的访问权限

    # 文件结构

    # 文件逻辑结构

    文件访问、存取,或者文件中查找数据的方式,是由文件的逻辑结构决定的。

    文件的逻辑结构分为:

    • 无结构文件,也叫做流式文件,对该类型文件中记录的访问只能通过穷举搜索。
    • 有结构文件,也叫做记录式文件,这里讨论的都是这种,这样的文件是由一个以上的记录构成的。
      • 定长记录
      • 变长记录

    # 顺序

    每次读写大批量数据时,顺序文件的效率是最高的。

    # 直接

    直接文件也叫做散列文件、哈希文件。

    • 优点
      • 存取速度快

    # 索引

    建立一张索引表,索引表本身是定长记录的顺序文件,其中的每一个索引表项包含:指向记录的指针、记录的长度

    • 优点
      • 提升了查找速度
    • 缺点
      • 索引表增加了存储空间

    # 索引顺序

    在索引文件的基础上,索引表项指向的单个记录变为一组记录组成的顺序表,就得到了一级索引的索引顺序文件。

    优缺点和索引文件仍然一样...

    # 记录成组分解

    记录成组分解技术是指若干逻辑记录存入一个块,一个逻辑记录不能跨越 2 个块。

    搭配隐式链接,每个个块存储固定长度的若干逻辑记录,剩余的字节用于存储指向下一个块的指针。

    # 文件物理结构

    # 连续分配

    配合使用空闲表法管理空闲磁盘空间。

    进程访问磁盘时所需的寻道数和寻道时间最小

    文件-连续分配

    ⚠️磁带只能使用连续分配,只能顺序存取!

    # 扩展连续分配

    例如: ext4ext3

    FCB 中比连续分配多增加一项为指向下一个扩展块的指针

    # 链接分配 (隐式链接)

    每一个块包含该文件指向下一个块的指针,这些指针对用户来说是透明的,空指针表示文件末尾。

    • 优点:

      • 不会产生外部碎片,提高了磁盘利用率。
    • 缺点:

      • 只支持顺序访问,随机访问效率低

      • 稳定性可靠性低,链指针 “断掉” 就会导文件数据丢失

      • 指向下一个盘块的指针消耗了磁盘空间

    文件-隐式链接

    # 显式链接 (FAT)

    用于链接文件各个物理块的指针显式地存放在内存地一张唯一的链接表中,称为文件分配表 (Fils Alloction Table, FAT),每个表项存放指向下一个块的指针。

    ⚠️FAT 本身就可以实现空闲磁盘块管理,若磁盘块空闲,则其表项可标记为 -1

    FAT 在系统启动时被读入内存,检索记录都在内存中进行。

    • 优点:

      • 支持顺序访问随机访问(磁盘,不是 FAT)

      • 检索在内存中完成,速度快效率高,减少了磁盘访问次数

    • 缺点:

      • FAT 表需要占用一定的内存空间

    FAT

    以 “簇”(cluster) 为单位分配,cluster 01 保留,从 2 开始计数

    LFN 目录项

    # 索引分配

    建立一张索引表

    • 优点:

      • 支持直接访问,索引表的第 i 个条目指向的就是第 i 个块
      • 不会产生外部碎片,提高了磁盘利用率。
    • 缺点:

      • 索引块增加了额外磁盘存储开销

    文件-索引分配

    可以分为:

    • 单级索引

      • 局限:
        • 只适用于小文件
    • 多级索引

      • 通过主索引查找二级索引......
      • 优点:
        • 极大加快了对大型文件的查找速度
      • 问题:
        • 对小文件来说没必要使用深层级的索引,会造成磁盘存储浪费和访问开销增大
    • 混合索引

      • 文件-混合索引

    UNIX 采用最深为三级的混合索引

    # 目录

    # 目录逻辑结构

    # 单级目录

    实现了按名存取,每个文件名对应唯一一个文件。

    缺点:

    • 文件不允许重名
    • 不方便文件共享

    # 两级目录

    • 主文件目录 Maste File Directory (MFD)
      • 用户文件目录 User File Directory (UFD)
      • 用户文件目录 User File Directory (UFD)
      • 用户文件目录 User File Directory (UFD)
      • ......

    解决了多用户之间文件重名的问题。

    # 树形目录

    理论上可以实现无限深度的目录层次。

    出现了相对路径绝对路径的概念。

    # 无环图目录

    比树形目录多了共享子目录,实现了文件、目录共享。

    一个文件可以有多个路径,称为别名。例如:

    • Linux 中的软链接、硬链接
    • Windows 的快捷方式

    # 通用图目录

    允许了环的出现,实现了自引用的特殊需求。

    # 目录物理实现

    目录第物理实现决定了目录检索的方式,包括 2 中检索方式:

    • 顺序检索
    • 散列法 / 哈希表

    # 线性列表

    查找比较费时。

    # 哈希表

    • 优点:
      • 查找非常迅速
      • 插入、删除比较简单
    • 缺点:
      • 会有冲突,链式处理溢出。

    # 空闲空间管理

    # 空闲表

    属于连续分配方式,建立一张空闲表,和连续分配方式的目录类似,每一个表项记录一个空闲盘区,包括第一个空闲盘块号和空闲盘块数。

    空闲盘区分配与回收与内存动态分配、回收类似。

    # 空闲链表

    free-space list

    • 空闲盘块链
      • 以盘块为单位拉成一条链
      • 优点:回收分配过程简单
      • 缺点:简单但是需要重复多次效率低,且盘块链会很长
    • 空闲盘区链
      • 以盘区为单位拉成一条链
      • 优点:分配回收效率高
      • 缺点:回收分配过程复杂

    # 位 (示) 图

    bit vector or bit map,似乎也叫空闲向量表

    • 优点:
      • 容易找到一个或一组连续的空闲磁盘块
      • 位示图小,可以放在内存中,节省访问磁盘的开销
    • 缺点:
      • 位示图大小随磁盘容量增大而增大,因而常用于小型计算机

    # Space Maps

    位示图的改进,将磁盘设备分为 metaslab 的单位,每一个 metslab 内对空闲盘块计数

    # 成组链接

    grouping

    ⭐️ UNIX 采用该方法

    第一组空闲盘块总数和空闲盘块号称为空闲盘块号栈

    # 超级块

    super block

    包含:

    • 空闲向量表 / 位 (示) 图 / 空闲盘块号栈
    • 卷中目录区、文件区的划分信息

    一般放在卷头位置,超级块要预先读入内存才能对该卷文件操作,并且经常保持内存与磁盘超级块的一致性。

    缓存

    • page cache,为用户程序服务(用户程序以页的方式访问)
    • disk block cache,为操作系统管理文件服务

    A unified buffer cache uses the same page cache to cache both memory-mapped pages and ordinary file system I/O to avoid double caching

    But which caches get priority, and what replacement algorithms to use?

    Linux VFS 对象:

    • 超级块
    • 索引节点
    • 目录项
    • 文件
    Edited on Views times
    \ No newline at end of file +文件系统 - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    磁盘的 block 和文件系统的 block 不是一回事:

    • 磁盘的 block :往往是固定的 512 B;
    • 文件系统的 block 是基本单位,可以自定义,不一定和磁盘的 block 大小相等。
      • 文件系统分配空间的基本单位往往是页面大小 4 KB ,

    文件系统结构:(自上而下分层)

    • application programs,应用程序
    • logical file system,逻辑文件系统
    • file-organization module,文件组织模块
    • basic file system,基本文件系统
    • I/O control,I/O 控制
    • devices,设备

    修改文件要写回磁盘!计算访问磁盘次数的时候要注意写操作比读操作多 1

    # 数据结构

    面向对象

    # FCB

    别名:文件目录项文件控制块

    包含:

    • 指向关联的 inode 的指针
    • 指向父目录、子目录的指针

    # 文件目录

    FCB 的有序集合

    # inode

    别名:索引节点文件描述信息、index node

    ext2ext3 采用 inode 表

    将 FCB 中的一部分文件描述信息分离出来,这些是检索目录时不需要的不必放在内存的,所以只将指向这部分的一个指针放在内存的目录项里。

    采用 inode 可以减少查找文件时的 I/O 信息量,相当于简化了内存中的文件目录项 (FCB),将其中一部分留在磁盘,只保留指向他们的指针。例如 Linux 的文件目录项就只包含文件名和指向 indoe 的指针。

    ⭐️硬连接和软链接与 inode

    • 硬链接文件指向同一个 inode
    • 软链接文件指向不同的 inode,相当于不同的文件,只不过存的是路径

    在文件的物理结构中,则 inode 中就存储了索引文件的主索引表

    直接以数组方式存储的 indoe 表,相当于单级索引;也可以有多级索引

    • 磁盘索引节点,每个文件唯一都有一个
      • 文件主标识符
      • 文件类型
      • 文件存取权限
      • 文件物理地址
      • 文件长度
      • 文件链接计数
        • ⭐️ count ,实现硬链接文件共享,表示链接到本索引节点的用户目录项的数量。只有 count=0 的时候才能删除文件。
      • 文件存取时间
    • 内存索引节点,文件被打开时,磁盘索引节点复制放在内存的副本
      • 索引节点号,用于标识内存索引节点
      • 状态,是否被上锁
      • 访问计数,为 0 时文件关闭
      • 逻辑设备号
      • 链接指针,指向空闲列表或散列队列

    # 打开文件表

    本质上是实现只需要一次访问磁盘定位文件在磁盘的位置,将其存储到内存,后续在内存可以找到文件在磁盘的位置。

    两级表:

    • 整个系统一个表,包含与进程无关的信息,每一个文件条目包含:
      • 文件描述符 (File Descriptor, FD)
        • 这是 Linux 中的名字,Windows 中也叫文件句柄
      • 文件打开次数
        • 变为 0 之后就可以 “关闭文件”,从表中删除该项
      • 文件磁盘位置
      • 访问日期
      • 文件大小
    • 每个进程有一个表,包含与进程有关的信息,每一个文件条目包含:
      • 读写指针
      • 访问权限
      • 指向系统打开文件表的对应文件条目指针

    ⚠️系统只会在第一次打开一个文件的时候使用其文件名在磁盘上定位,然后在内存中创建文件表中的条目,此后就不再需要文件名。

    ⚠️ open 系统调用由用户调用,一个文件可能对某一用户是第一次打开,但是对系统而言不是第一次,只会在用户进程的打开文件表增加条目。

    # 文件保护

    • 口令保护
    • 加密保护
    • 访问控制

    # 文件访问控制表

    可能也有叫做存取控制矩阵之类的 (?)

    为每个文件和目录设置一个文件访问控制表。用来管理不同用户、用户组对该文件或目录的访问权限

    # 文件结构

    # 文件逻辑结构

    文件访问、存取,或者文件中查找数据的方式,是由文件的逻辑结构决定的。

    文件的逻辑结构分为:

    • 无结构文件,也叫做流式文件,对该类型文件中记录的访问只能通过穷举搜索。
    • 有结构文件,也叫做记录式文件,这里讨论的都是这种,这样的文件是由一个以上的记录构成的。
      • 定长记录
      • 变长记录

    # 顺序

    每次读写大批量数据时,顺序文件的效率是最高的。

    # 直接

    直接文件也叫做散列文件、哈希文件。

    • 优点
      • 存取速度快

    # 索引

    建立一张索引表,索引表本身是定长记录的顺序文件,其中的每一个索引表项包含:指向记录的指针、记录的长度

    • 优点
      • 提升了查找速度
    • 缺点
      • 索引表增加了存储空间

    # 索引顺序

    在索引文件的基础上,索引表项指向的单个记录变为一组记录组成的顺序表,就得到了一级索引的索引顺序文件。

    优缺点和索引文件仍然一样...

    # 记录成组分解

    记录成组分解技术是指若干逻辑记录存入一个块,一个逻辑记录不能跨越 2 个块。

    搭配隐式链接,每个个块存储固定长度的若干逻辑记录,剩余的字节用于存储指向下一个块的指针。

    # 文件物理结构

    # 连续分配

    配合使用空闲表法管理空闲磁盘空间。

    进程访问磁盘时所需的寻道数和寻道时间最小

    文件-连续分配

    ⚠️磁带只能使用连续分配,只能顺序存取!

    # 扩展连续分配

    例如: ext4ext3

    FCB 中比连续分配多增加一项为指向下一个扩展块的指针

    # 链接分配 (隐式链接)

    每一个块包含该文件指向下一个块的指针,这些指针对用户来说是透明的,空指针表示文件末尾。

    • 优点:

      • 不会产生外部碎片,提高了磁盘利用率。
    • 缺点:

      • 只支持顺序访问,随机访问效率低

      • 稳定性可靠性低,链指针 “断掉” 就会导文件数据丢失

      • 指向下一个盘块的指针消耗了磁盘空间

    文件-隐式链接

    # 显式链接 (FAT)

    用于链接文件各个物理块的指针显式地存放在内存地一张唯一的链接表中,称为文件分配表 (Fils Alloction Table, FAT),每个表项存放指向下一个块的指针。

    ⚠️FAT 本身就可以实现空闲磁盘块管理,若磁盘块空闲,则其表项可标记为 -1

    FAT 在系统启动时被读入内存,检索记录都在内存中进行。

    • 优点:

      • 支持顺序访问随机访问(磁盘,不是 FAT)

      • 检索在内存中完成,速度快效率高,减少了磁盘访问次数

    • 缺点:

      • FAT 表需要占用一定的内存空间

    FAT

    以 “簇”(cluster) 为单位分配,cluster 01 保留,从 2 开始计数

    LFN 目录项

    # 索引分配

    建立一张索引表

    • 优点:

      • 支持直接访问,索引表的第 i 个条目指向的就是第 i 个块
      • 不会产生外部碎片,提高了磁盘利用率。
    • 缺点:

      • 索引块增加了额外磁盘存储开销

    文件-索引分配

    可以分为:

    • 单级索引

      • 局限:
        • 只适用于小文件
    • 多级索引

      • 通过主索引查找二级索引......
      • 优点:
        • 极大加快了对大型文件的查找速度
      • 问题:
        • 对小文件来说没必要使用深层级的索引,会造成磁盘存储浪费和访问开销增大
    • 混合索引

      • 文件-混合索引

    UNIX 采用最深为三级的混合索引

    # 目录

    # 目录逻辑结构

    # 单级目录

    实现了按名存取,每个文件名对应唯一一个文件。

    缺点:

    • 文件不允许重名
    • 不方便文件共享

    # 两级目录

    • 主文件目录 Maste File Directory (MFD)
      • 用户文件目录 User File Directory (UFD)
      • 用户文件目录 User File Directory (UFD)
      • 用户文件目录 User File Directory (UFD)
      • ......

    解决了多用户之间文件重名的问题。

    # 树形目录

    理论上可以实现无限深度的目录层次。

    出现了相对路径绝对路径的概念。

    # 无环图目录

    比树形目录多了共享子目录,实现了文件、目录共享。

    一个文件可以有多个路径,称为别名。例如:

    • Linux 中的软链接、硬链接
    • Windows 的快捷方式

    # 通用图目录

    允许了环的出现,实现了自引用的特殊需求。

    # 目录物理实现

    目录第物理实现决定了目录检索的方式,包括 2 中检索方式:

    • 顺序检索
    • 散列法 / 哈希表

    # 线性列表

    查找比较费时。

    # 哈希表

    • 优点:
      • 查找非常迅速
      • 插入、删除比较简单
    • 缺点:
      • 会有冲突,链式处理溢出。

    # 空闲空间管理

    # 空闲表

    属于连续分配方式,建立一张空闲表,和连续分配方式的目录类似,每一个表项记录一个空闲盘区,包括第一个空闲盘块号和空闲盘块数。

    空闲盘区分配与回收与内存动态分配、回收类似。

    # 空闲链表

    free-space list

    • 空闲盘块链
      • 以盘块为单位拉成一条链
      • 优点:回收分配过程简单
      • 缺点:简单但是需要重复多次效率低,且盘块链会很长
    • 空闲盘区链
      • 以盘区为单位拉成一条链
      • 优点:分配回收效率高
      • 缺点:回收分配过程复杂

    # 位 (示) 图

    bit vector or bit map,似乎也叫空闲向量表

    • 优点:
      • 容易找到一个或一组连续的空闲磁盘块
      • 位示图小,可以放在内存中,节省访问磁盘的开销
    • 缺点:
      • 位示图大小随磁盘容量增大而增大,因而常用于小型计算机

    # Space Maps

    位示图的改进,将磁盘设备分为 metaslab 的单位,每一个 metslab 内对空闲盘块计数

    # 成组链接

    grouping

    ⭐️ UNIX 采用该方法

    第一组空闲盘块总数和空闲盘块号称为空闲盘块号栈

    # 超级块

    super block

    包含:

    • 空闲向量表 / 位 (示) 图 / 空闲盘块号栈
    • 卷中目录区、文件区的划分信息

    一般放在卷头位置,超级块要预先读入内存才能对该卷文件操作,并且经常保持内存与磁盘超级块的一致性。

    缓存

    • page cache,为用户程序服务(用户程序以页的方式访问)
    • disk block cache,为操作系统管理文件服务

    A unified buffer cache uses the same page cache to cache both memory-mapped pages and ordinary file system I/O to avoid double caching

    But which caches get priority, and what replacement algorithms to use?

    Linux VFS 对象:

    • 超级块
    • 索引节点
    • 目录项
    • 文件
    Edited on Views times
    \ No newline at end of file diff --git a/cs/os/io/index.html b/cs/os/io/index.html index da262cc5..28fb47b9 100644 --- a/cs/os/io/index.html +++ b/cs/os/io/index.html @@ -1 +1 @@ -I/O - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    I/O 的访问、管理分为两种类型:(或者说按照编址类型划分)

    • I/O 指令(独立编址)
      • 通过 I/O 端口号访问设备管理器
      • 涉及模态转换、状态更迭时才采用
    • 内存映射 I/O (统一编址)
      • 设备的寄存器或存储被映射到内存物理地址空间中;
      • 通过内存 load/store 指令完成 I/O 操作;
      • MMU 设置映射,硬件跳线或程序在启动时设置地址;
      • 实际上绝大多数采用的方式

    I/O 控制:

    • 轮询,或者程序直接控制
      • 通过忙等待(轮询)检查 I/O 控制器状态
      • 完全交给 CPU
    • 中断
      • 等待中断来得到 I/O 控制器状态信息
      • 中断到来前 CPU 可以干其他事(CPU、I/O 并行)
    • DMA
      • 磁盘和 I/O 设备之间直接传输数据
      • CPU 极少介入,仅在数据传输完成时介入,其他时候 CPU、I/O 并行工作
      • CPU 上专门的部件作为 DMA 控制器
    • * 通道控制
      • 一种特殊的、独立于 CPU 的处理机,相当于独立性更强的 DMA
      • 指令类型单一,与 CPU 共享内存
      • 实现 CPU、通道、I/O 三者并行工作

    # 磁盘调度算法

    给定磁盘访问 LBA 序列: 55, 58, 39, 18, 90, 160, 150, 38, 184

    # FIFO

    无论在哪个场景都是最公平的调度算法

    FIFO

    # 优先级 (PRI)

    调度的控制超出了磁盘管理软件的控制范围

    目标不是优化磁盘利用率,而是实现其他目标

    短批作业和交互式作业的优先级更
    提供良好的交互式响应时间

    较长的工作可能需要等待过长的时间
    对数据库系统是一种糟糕策略

    # 最短服务时间优先 (SSTF)

    选择使磁头臂从当前位置开始移动最少的磁盘 I/O 请求

    总是选择导致最小寻道时间的请求

    SSTF

    # 扫描算法 (SCAN)

    也叫电梯算法、NOOP 算法、SCAN 算法。

    非公平的,不同请求获得的优先级是不一样的:越靠近外侧的扇区请求响应的优先级更高

    SCAN

    # 循环扫描算法 (C-SCAN)

    C-SCAN 也就是 Circular Scan 算法。

    C-SCAN

    # 其他 SCAN 变种

    以下介绍的 2 中变种 SCAN 调度算法的目的在于避免磁臂黏着现象,具体可参考:https://blog.csdn.net/m0_52733659/article/details/133840705

    # N 步 SCAN

    设置多个队列,每个队列容量为 N ,轮流对每个队列采用 SCAN 扫描算法,该队列不接受请求,所有请求按照 FIFO 放入下一个队列。每一次扫描,都只处理 N 个请求。

    # FSCAN

    设置 2 个队列,轮流对 2 个队列采用 SCAN 扫描算法,扫描其中一个队列的时候,该队列不接受请求,所有请求按照 FIFO 放入另一个队列

    # 总结

    SSTF 是常见的,具有天然的吸引力

    SCAN 和 C-SCAN 对于磁盘负载较大的系统性能更好

    # NVM 调度

    没有磁头或旋转延迟,但仍有优化空间

    在 RHEL 7 中,使用 NOOP (无操作、电梯调度),但合并相邻的 LBA 请求

    • ⭐️ NVM 最佳随机 I/O
    • ⭐️ 硬盘最佳顺序 I/O

    # 存储设备管理

    # 低级格式化

    低级格式化,或者说物理格式化

    将磁盘划分为磁盘控制器可以读写的扇区

    每个扇区可以保存头信息、数据和纠错码 (ECC)

    通常 512 字节的数据,但可以选择

    # 高级格式化

    高级格式化,或者说逻辑格式化、“创建文件系统”

    文件系统的单位往往和低级格式化的扇区不一样!

    将磁盘划分为一组或多组柱面,每组柱面都视为逻辑磁盘

    为了提高效率,大多数文件系统将块分组到簇中

    • 磁盘 I/O 在块中完成
    • 文件 I/O 在簇中完成

    # 原始磁盘访问

    绕过操作系统的文件系统,也就是绕过操作系统的高级格式化,往往适用于自己进行数据管理的应用程序(例如数据库)

    # RAID

    CategoryLevelDescriptionDisks requiredData availabilityLarge I/O data transfer capacitySmall I/O request rate
    Striping0NonredundantNNLower than single diskVery highVery high for both read and write
    Mirroring1Mirrored2N2NHigher than RAID 2, 3, 4, or 5; lower than RAID 6Higher than single disk for read; similar to single disk for writeUp to twice that of a single disk for read; similar to single disk for write
    Parallel access2Redundant via Hamming codeN+mN+mMuch higher than single disk; comparable to RAID 3, 4, or 5Highest of all listed alternativesApproximately twice that of a single disk
    3Bit-interleaved parityN+1N+1Much higher than single disk; comparable to RAID 2, 4, or 5Highest of all listed alternativesApproximately twice that of a single disk
    Independent access4Block-interleaved parityN+1N+1Much higher than single disk; comparable to RAID 2,3, or 5Similar to RAID O for read; significantly lower than single disk for writeSimilar to RAID 0 for read; significantly lower than single disk for write
    5Block-interleaved distributed parityN+1N+1Much higher than single disk; comparable to RAID 2,3, or 4Similar to RAID 0 for read; lower than single disk for writeSimilar to RAID 0 for read; generally lower than single disk for write
    6Block-interleaved dual distributed parityN+2N+2Highest of all listed alternativesSimilar to RAID 0 for read; lower than RAID 5 for writeSimilar to RAID 0 for read; significantly lower than RAID 5 for write
    Edited on Views times
    \ No newline at end of file +I/O - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    I/O 的访问、管理分为两种类型:(或者说按照编址类型划分)

    • I/O 指令(独立编址)
      • 通过 I/O 端口号访问设备管理器
      • 涉及模态转换、状态更迭时才采用
    • 内存映射 I/O (统一编址)
      • 设备的寄存器或存储被映射到内存物理地址空间中;
      • 通过内存 load/store 指令完成 I/O 操作;
      • MMU 设置映射,硬件跳线或程序在启动时设置地址;
      • 实际上绝大多数采用的方式

    I/O 控制:

    • 轮询,或者程序直接控制
      • 通过忙等待(轮询)检查 I/O 控制器状态
      • 完全交给 CPU
    • 中断
      • 等待中断来得到 I/O 控制器状态信息
      • 中断到来前 CPU 可以干其他事(CPU、I/O 并行)
    • DMA
      • 磁盘和 I/O 设备之间直接传输数据
      • CPU 极少介入,仅在数据传输完成时介入,其他时候 CPU、I/O 并行工作
      • CPU 上专门的部件作为 DMA 控制器
    • * 通道控制
      • 一种特殊的、独立于 CPU 的处理机,相当于独立性更强的 DMA
      • 指令类型单一,与 CPU 共享内存
      • 实现 CPU、通道、I/O 三者并行工作

    # 磁盘调度算法

    给定磁盘访问 LBA 序列: 55, 58, 39, 18, 90, 160, 150, 38, 184

    # FIFO

    无论在哪个场景都是最公平的调度算法

    FIFO

    # 优先级 (PRI)

    调度的控制超出了磁盘管理软件的控制范围

    目标不是优化磁盘利用率,而是实现其他目标

    短批作业和交互式作业的优先级更
    提供良好的交互式响应时间

    较长的工作可能需要等待过长的时间
    对数据库系统是一种糟糕策略

    # 最短服务时间优先 (SSTF)

    选择使磁头臂从当前位置开始移动最少的磁盘 I/O 请求

    总是选择导致最小寻道时间的请求

    SSTF

    # 扫描算法 (SCAN)

    也叫电梯算法、NOOP 算法、SCAN 算法。

    非公平的,不同请求获得的优先级是不一样的:越靠近外侧的扇区请求响应的优先级更高

    SCAN

    # 循环扫描算法 (C-SCAN)

    C-SCAN 也就是 Circular Scan 算法。

    C-SCAN

    # 其他 SCAN 变种

    以下介绍的 2 中变种 SCAN 调度算法的目的在于避免磁臂黏着现象,具体可参考:https://blog.csdn.net/m0_52733659/article/details/133840705

    # N 步 SCAN

    设置多个队列,每个队列容量为 N ,轮流对每个队列采用 SCAN 扫描算法,该队列不接受请求,所有请求按照 FIFO 放入下一个队列。每一次扫描,都只处理 N 个请求。

    # FSCAN

    设置 2 个队列,轮流对 2 个队列采用 SCAN 扫描算法,扫描其中一个队列的时候,该队列不接受请求,所有请求按照 FIFO 放入另一个队列

    # 总结

    SSTF 是常见的,具有天然的吸引力

    SCAN 和 C-SCAN 对于磁盘负载较大的系统性能更好

    # NVM 调度

    没有磁头或旋转延迟,但仍有优化空间

    在 RHEL 7 中,使用 NOOP (无操作、电梯调度),但合并相邻的 LBA 请求

    • ⭐️ NVM 最佳随机 I/O
    • ⭐️ 硬盘最佳顺序 I/O

    # 存储设备管理

    # 低级格式化

    低级格式化,或者说物理格式化

    将磁盘划分为磁盘控制器可以读写的扇区

    每个扇区可以保存头信息、数据和纠错码 (ECC)

    通常 512 字节的数据,但可以选择

    # 高级格式化

    高级格式化,或者说逻辑格式化、“创建文件系统”

    文件系统的单位往往和低级格式化的扇区不一样!

    将磁盘划分为一组或多组柱面,每组柱面都视为逻辑磁盘

    为了提高效率,大多数文件系统将块分组到簇中

    • 磁盘 I/O 在块中完成
    • 文件 I/O 在簇中完成

    # 原始磁盘访问

    绕过操作系统的文件系统,也就是绕过操作系统的高级格式化,往往适用于自己进行数据管理的应用程序(例如数据库)

    # RAID

    CategoryLevelDescriptionDisks requiredData availabilityLarge I/O data transfer capacitySmall I/O request rate
    Striping0NonredundantNNLower than single diskVery highVery high for both read and write
    Mirroring1Mirrored2N2NHigher than RAID 2, 3, 4, or 5; lower than RAID 6Higher than single disk for read; similar to single disk for writeUp to twice that of a single disk for read; similar to single disk for write
    Parallel access2Redundant via Hamming codeN+mN+mMuch higher than single disk; comparable to RAID 3, 4, or 5Highest of all listed alternativesApproximately twice that of a single disk
    3Bit-interleaved parityN+1N+1Much higher than single disk; comparable to RAID 2, 4, or 5Highest of all listed alternativesApproximately twice that of a single disk
    Independent access4Block-interleaved parityN+1N+1Much higher than single disk; comparable to RAID 2,3, or 5Similar to RAID O for read; significantly lower than single disk for writeSimilar to RAID 0 for read; significantly lower than single disk for write
    5Block-interleaved distributed parityN+1N+1Much higher than single disk; comparable to RAID 2,3, or 4Similar to RAID 0 for read; lower than single disk for writeSimilar to RAID 0 for read; generally lower than single disk for write
    6Block-interleaved dual distributed parityN+2N+2Highest of all listed alternativesSimilar to RAID 0 for read; lower than RAID 5 for writeSimilar to RAID 0 for read; significantly lower than RAID 5 for write
    Edited on Views times
    \ No newline at end of file diff --git a/cs/os/linux-customized-syscall/index.html b/cs/os/linux-customized-syscall/index.html index 98b3b4be..228bb5f2 100644 --- a/cs/os/linux-customized-syscall/index.html +++ b/cs/os/linux-customized-syscall/index.html @@ -1 +1 @@ -给 Linux 内核增加自定义系统调用 - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    这是老师布置的一个小作业,让我们在 linux 内核源码里增加自定义的系统调用程序,用这篇文章记录一下完整的操作过程;

    这样扩展 Linux 内核的传统方式需要每次都重新编译内核,更现代的方式是使用 eBPF

    使用清华镜像源下载 linux 内核 5.10.19 版本,解压然后进入解压目录:

    shell
    wget https://mirrors.tuna.tsinghua.edu.cn/kernel/v5.x/linux-5.10.19.tar.gz
    ......
    tar -zxvf linux-5.10.19.tar.gz
    ......
    cd linux-5.10.19

    kernel/sys.c 中用 SYSCALL 宏增加系统调用源代码:

    c
    SYSCALL_DEFINE1(myhello, int, arg)
    {
    	printk("\x1B[1;46mHello, world! from LSY %d\x1B[m", arg);
    	return 0;
    }

    include/linux/syscalls.h 中增加自定义系统调用的声明:

    c
    asmlinkage void sys_myhello(int arg);

    arch/x86/entry/syscalls/syscall_32.tblarch/x86/entry/syscalls/syscall_64.tbl 中增加调用号:

    raw
    441	64	myhello		sys_myhello

    然后配置:

    shell
    make i386_defconfig
    make menuconfig

    make menuconfig 打开的图像界面中依次选择 Kernel hackingCompile-time checks and compiler options ,最后在 [ ] Compile the kernel with debug info 输入 Y 勾选,保存退出;

    然后就可以编译内核:

    shell
    make -j$(nproc)

    写一个简单的 initramfs 程序 helloworld.c 调用自定义的系统调用函数,传入我的学号作为参数,源代码如下:

    c
    #include<stdio.h>
    void main(){
        syscall(441, 22342043);
        fflush(stdout);
        while (1);
    }

    将该程序编译:

    shell
    gcc -o helloworld -m32 -static helloworld.c

    用 cpio 打包 initramfs :

    shell
    echo helloworld | cpio -o --format=newc > hwinitramfs

    调试模式启动内核,并加载 initramfs ,让内核程序暂停运行等待 gdb 连接 :

    shell
    qemu-system-i386 -kernel linux-5.10.19/arch/x86/boot/bzImage -initrd hwinitramfs -s -S -append "console=ttyS0 rdinit=helloworld" -nographic

    然后启动 gdb :

    shell
    gdb

    加载符号表,并连接 qemu 调试服务提供的默认端口 1234 ,然后就可以输入 c 继续运行内核:

    gdb
    file linux-5.10.19/vmlinux
    target remote:1234
    c

    输出结果:

    结果

    \ No newline at end of file +给 Linux 内核增加自定义系统调用 - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    这是老师布置的一个小作业,让我们在 linux 内核源码里增加自定义的系统调用程序,用这篇文章记录一下完整的操作过程;

    这样扩展 Linux 内核的传统方式需要每次都重新编译内核,更现代的方式是使用 eBPF

    使用清华镜像源下载 linux 内核 5.10.19 版本,解压然后进入解压目录:

    shell
    wget https://mirrors.tuna.tsinghua.edu.cn/kernel/v5.x/linux-5.10.19.tar.gz
    ......
    tar -zxvf linux-5.10.19.tar.gz
    ......
    cd linux-5.10.19

    kernel/sys.c 中用 SYSCALL 宏增加系统调用源代码:

    c
    SYSCALL_DEFINE1(myhello, int, arg)
    {
    	printk("\x1B[1;46mHello, world! from LSY %d\x1B[m", arg);
    	return 0;
    }

    include/linux/syscalls.h 中增加自定义系统调用的声明:

    c
    asmlinkage void sys_myhello(int arg);

    arch/x86/entry/syscalls/syscall_32.tblarch/x86/entry/syscalls/syscall_64.tbl 中增加调用号:

    raw
    441	64	myhello		sys_myhello

    然后配置:

    shell
    make i386_defconfig
    make menuconfig

    make menuconfig 打开的图像界面中依次选择 Kernel hackingCompile-time checks and compiler options ,最后在 [ ] Compile the kernel with debug info 输入 Y 勾选,保存退出;

    然后就可以编译内核:

    shell
    make -j$(nproc)

    写一个简单的 initramfs 程序 helloworld.c 调用自定义的系统调用函数,传入我的学号作为参数,源代码如下:

    c
    #include<stdio.h>
    void main(){
        syscall(441, 22342043);
        fflush(stdout);
        while (1);
    }

    将该程序编译:

    shell
    gcc -o helloworld -m32 -static helloworld.c

    用 cpio 打包 initramfs :

    shell
    echo helloworld | cpio -o --format=newc > hwinitramfs

    调试模式启动内核,并加载 initramfs ,让内核程序暂停运行等待 gdb 连接 :

    shell
    qemu-system-i386 -kernel linux-5.10.19/arch/x86/boot/bzImage -initrd hwinitramfs -s -S -append "console=ttyS0 rdinit=helloworld" -nographic

    然后启动 gdb :

    shell
    gdb

    加载符号表,并连接 qemu 调试服务提供的默认端口 1234 ,然后就可以输入 c 继续运行内核:

    gdb
    file linux-5.10.19/vmlinux
    target remote:1234
    c

    输出结果:

    结果

    \ No newline at end of file diff --git a/cs/os/memory-management/index.html b/cs/os/memory-management/index.html index f071a437..5cb7621c 100644 --- a/cs/os/memory-management/index.html +++ b/cs/os/memory-management/index.html @@ -1 +1 @@ -内存管理 - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    冯・诺伊曼计算机体系决定了操作系统需要内存管理

    # 虚拟内存物理实现

    • 连续分配
      • 单一连续分配
      • 分区
        • 固定分区分配
        • 动态分区分配
    • 非连续分配
      • 段式
      • 页式
      • 段页式

    ⭐️内部碎片和外部碎片

    • 内部碎片
      • 是已经被分配给某个进程的,该进程没有利用的
      • 固定分区分配产生这种碎片
    • 外部碎片
      • 是无法被分配给任何进程的,不属于任何进程的
      • 动态分区分配产生这种碎片
      • 可以通过紧凑技术克服,需要动态重定位寄存器硬件支持

    # 分区 (连续分配)

    # 单一连续分配

    将内存分为两部分:

    • 系统区
    • 用户区
      • 唯一一个用户程序独占整个用户区

    内存中只有一个用户程序,只能用于单用户、单任务的操作系统。

    有内部碎片,无外部碎片。

    # 固定分区

    使用空闲分区链 / 表

    有内部碎片,无外部碎片。

    # 动态分区

    使用空闲分区链 / 表

    无内部碎片,有外部碎片。

    # 首次适应

    空闲分区按地址递增顺序排列。

    每次分配时从头开始顺序查找到第一个满足大小的空闲分区。

    性能最好。

    # 邻近适应

    空闲分区按地址递增顺序排列。和首次适应一样。

    每次分配时从上次查找结束位置开始查找到第一个满足大小的空闲分区。

    # 最佳适应

    空闲分区按容量递增顺序排列。

    产生最多外部碎片

    # 最坏适应

    空闲分区按容量递减顺序排列。

    前面三种都基于顺序搜索,后面三种则都基于索引搜索

    # 快速适应

    基于索引分配,对空闲分区按大小相同的分为一类,每一类单独设置一个空闲分区链。建立一张索引表管理所有空闲分区链,索引表项包含对应大小的空闲分区链的头指针

    根据进程需要的大小,在索引表中找到能容纳它的最小的空闲分区链表,直接从链表取出第一块进行分配。

    优劣:

    • 优点
      • 查找效率高
      • 不产生内部碎片(当然啊这不是动态分区吗)
    • 缺点
      • 回收分区复杂,导致算法复杂、系统开销大

    # 伙伴系统

    我感觉应该算固定分区分配才对?

    规定所有分区大小为 22kk 次幂。

    每次分配大小为 nn (2i1<n2i2^{i-1}<n\leq 2^i) 的分区时,在

    导致过多内部碎片⚠️

    Slab 分配器 —— 一般只有内核内存使用

    # 哈希算法

    根据空闲分区链表的分布规律建立哈希函数,构建一张以哈希表:

    • 关键字 key 为空闲分区大小
    • 表项 value 为对应空闲分区链的头指针

    # 分段

    程序员或者编译器会把程序和数据指定到不同的分段,需要知道段的最大长度限制

    将程序产生的二维地址即 <段号,偏移><段号,偏移> 映射到一维物理地址, 这个地址是通过段表实现;

    段表中每个条目由段基址和段界限构成

    CPU 去哪里找段表?—— GDTR 寄存器,GDTR 寄存器的结构如下:

    • 低 2 字节(16 位)存储了 limit ,全局描述符表的大小(以字节为单位),减去 1 的结果。
    • 高 4 字节(32 位)存储了 base ,全局描述符表的基地址。

    # 分页

    页放到页帧 / 框里,区分:⚠️

    • 页(Page)
    • 页帧(Page frame)或者页框,是空的,没有数据

    页框 \geq 页,现代操作系统大多二者是相等的,方便管理

    # 页表

    系统中有一个唯一的页表寄存器(PTR),包含页表首地址(PTBR),只能由特权指令访问。

    页表长度寄存器(PTLR),可以直接用于判断是否越界(好像不一定有)

    x86_64 , i386 中的 PTR 寄存器叫做 CR3 寄存器

    页表从属于进程,每创建一个新进程就需要创建一个独立的新页表给它。

    # TLB

    转换表缓冲区 / 快表 /... 详细参考计组

    # 共享页

    共享代码

    • 进程 (即文本编辑器、编译器、窗口系统) 之间共享的只读 (可重入) 代码的一个副本;
    • 类似于共享同一进程空间的多个线程;
    • 如果允许共享读写页面,则还可用于进程间通信;

    # 多级 / 层页表

    经典:在现代计算机上考虑 32 位逻辑地址空间,页面大小为 4KB (2122^{12}),页面表将有 4M 个条目 (232/212=2202^{32}/2^{12}=2^{20}) ,如果每个条目为 4B ,每个进程需要为页表分配 16MB 的物理地址空间,这片连续的物理空间太大了!

    相同条件下考虑 64 位,即使采用 2 级分页,外层页表也需要 16384GB 大小的连续物理空间!所以 x86_64 采用 4 级页表

    现代 OS 大多采用 3 到 4 级页表

    64 位下的 2 级页表 和 3 级页表:

    64-2-3

    # 散列页表 / 哈希表

    # 聚簇页表

    64 位地址的变体,哈希值对应的不是一个页,而是一簇页(一个页族)

    特别适用于稀疏地址空间 (内存引用不连续且分散)

    # 倒排页表

    不再为每个进程设置一个页表并跟踪所有可能的逻辑页,而是跟踪所有物理页,此时整个架构自 CPU 开始到 OS 都有较大变化

    内存的每一物理页有一个条目,条目包含:

    • 存储在该实际内存位置的页的虚拟地址
    • 有关拥有该页的进程的信息组成

    CPU 产生的逻辑地址:

    倒排页表

    优点:

    • 减少存储每个页表所需的内存

    缺点:

    • 发生页引用时增加搜索表所需的时间
    • 共享内存不太容易实现:虚拟地址到共享物理地址的一种映射

    # 现代 OS 举例

    # Oracle SPARC Solaris

    考虑现代 64 位操作系统目标是效率,低开销

    基于哈希,但更复杂

    两个哈希表

    • 一个内核
    • 一个用于所有用户进程

    每个都将内存地址从虚拟内存映射到物理内存

    每个条目表示映射虚拟内存的连续区域

    比为每个页面提供单独的哈希表条目更高效

    每个条目都有基址和跨距 (表示条目所代表的页数)

    # IA-32/64

    英特尔 IA-32 和 IA-64 均支持分段、分页管理

    记录段号的基地址的描述符表,按私有、公有分为:

    • LDT 局部描述符表
    • GDT 全局描述符表

    早期 IA-32 利用 PAE 实现页地址扩展

    # ARM

    Apple

    段页混合式管理,较大连续内存走分段,较小连续内存走分页:

    arm

    # 虚拟内存软件实现

    ⭐️虚拟内存的实现建立在离散分配的内存管理方式基础上

    三个特征:

    • 多次性
    • 对换性
    • 虚拟性

    虚拟内存可以通过以下方式实现:

    • 请求分页
    • 请求分段(按需分段)
    • 请求段页式存储管理

    # 请求分页

    按需分页

    • 固定分配局部置换
    • 可变分配全局置换
    • 可变分配局部置换

    请求分页系统的外存

    • 文件区
      • 存放文件,采用离散分配方式
    • 对换区(交换区)
      • 存放对换页面,采用连续分配方式,I/O 速度更快

    可以在加载时将整个进程放入内存或者只在需要时才将页面放入内存

    所需的 I/O 更少,没有不必要的 I/O

    • 所需内存更少
    • 更快的响应
    • 更支持多用户

    类似于带交换的分页系统 (需要硬件支持)

    • 页面是必需的 —— 引用它
    • 无效引用 —— 终止
    • 不在内存中 —— 调入内存

    惰性交换程序:除非需要页,否则永远不要将页交换到内存中

    处理页面的交换程序是 “调页程序”

    # 处理缺页

    1. 检查进程的内部表,通常与 PCB 一起保存,以确定内存引用是否有效;
    2. 如果:(页面错误 (缺页) )
      • 无效,那么终止进程
      • 引用有效但是未调入页面, 那么发生缺页中断;
    3. 找到空闲帧;(或 victim 帧)
    4. 通过磁盘调度操作将页面交换到刚分配的帧中;
    5. 当磁盘操作读取完成后,重置内部表和页表,以指示当前页面在内存中;
    6. 重新启动导致缺页的指令,该进程能够访问所需的页;(指令重启

    # 问题

    # 启动进程时内存中没有页面

    操作系统将指令指针设置为进程的第一条指令,非内存驻留 -> 页面错误

    • 进程其他页面的访问也是一样;
    • 纯请求分页;

    # 多页错误

    • 考虑从内存中将 2 个数加和并将结果存储回内存的指令的读取和解码,可能产生多个缺页错误;
    • 产生性能痛点,幸运的是因为局部引用的存在,使得请求分页 具有较为合理的性能;

    # 硬件支持

    具有有效 / 无效位的页表
    辅助内存 (带交换空间的交换设备)

    指令重启,在缺页错误后能够重启任何指令;

    # 指令重启

    • 微代码计算试图访问源块和目的块的两端,如果出现缺页错误,那么在这一步出现,完成缺页逻辑后,执行移动操作;
    • 使用临时寄存器来保存覆盖位置的值,如果有缺页错误,则在陷阱发生之前,所有旧值都将写回到内存中。该动作是将内存恢复到指令启动之前的状态,这样能够重复执行该指令;

    # 写时复制

    写时复制 (CoW) 允许父进程和子进程最初共享内存中的相同页面

    如果任一进程修改了共享页面,则该页面被复制

    CoW 允许更高效的进程创建 (fork) ,因为只复制修改过的页面

    # 系统抖动

    频繁发生缺页,系统忙于进行页面的换入和换出,而没有进行有效的计算。

    系统抖动

    导致 CPU 利用率低,并发度低。

    系统还会自动检测 CPU 利用率,系统抖动时检测到 CPU 利用率降低,会创建更多精进程来提高 CPU 利用率,但这会反而加剧系统抖动。

    局部性原理告诉我们,访问磁盘总是在一个局部到另一个局部变化。而系统抖动从这个角度来说,就是局部性访问的大小超过了内存大小,所以导致了系统抖动。

    本地或优先级替换算法来缓解系统抖动。

    # 工作集

    WSSiWSS_i (Working Set of Process PiP_i ):进程在某一个固定的时间窗口 Δ\Delta(可以按时间算、指令数算)内,访问的所有页面

    • if Δ\Delta too small will not encompass entire locality
    • if Δ\Delta too large will encompass several localities
    • if Δ=\Delta=\infin will encompass entire program

    最理想的情况就是 WSS 能够被全部放在 TLB 中

    ⭐️ tt 时刻的工作集为该时刻及其以前 Δ\Delta 时间窗口范围内的访问的页面。

    解决方案 1

    建立 “可接受的” PFF (page-fault frequency) 并使用本地替换策略

    内核的内存管理就比较简单,因为它在内存里一定是连续的,可以完全利用线性地址偏移解决。

    预调页

    访问到某一内存页,很有可能下次就要访问它的邻居内存页,所以利用空间局部性

    在程序启动的时候可以减少大量的缺页

    当然也是有可能造成 I/O 和内存的浪费的

    # 帧分配

    在进程之间如何分配固定数量的可用内存?

    • 固定分配

    • 优先级分配

    • 全局置换 —— 进程从所有帧集中选择替换帧,一个进程可以从另一个进程获取帧

      • 高优先级进程可以置换低优先级进程
      • 缺点:进程不能控制他自己的缺页错误率
    • 局部置换 —— 每个进程仅从其自己的一组分配帧中进行选择

      • 每个进程的性能更加一致
      • 但可能是未充分利用的内存

    # 页面回收

    # 性能指标

    Three major activities

    • Service the interrupt – careful coding means just several hundred instructions needed
    • Read the page – lots of time
    • Restart the process – again just a small amount of time

    Effective Access Time (EAT)

    EAT=(1p)×MA+p×(PFO+SPOSPI)EAT=(1–p)\times MA+p\times(PFO+SPO-SPI)

    • Page Fault Rate 0p10\leq p\leq1
      • if p=0p=0, no page faults
      • if p=1p=1, every reference is a fault
    • MAMA : memory access
    • PFOPFO : page fault overhead
    • SPOSPO : swap page out ( if dirty )
    • SPISPI : swap page in

    # 性能优化

    交换空间 I/O 比文件系统 I/O 快,即使在同一设备上也是如此

    交换分配在更大的块中,比文件系统所需的管理更少;

    在进程加载时将整个进程映像复制到交换空间

    然后在交换空间内执行调页;
    在旧的 BSD Unix 中使用;

    从磁盘上的程序二进制文件请求分页,但在释放帧时将页面直 接放弃 (被覆盖),而不是分页调出;

    在 Solaris 和当前 BSD 中使用 § 仍然需要写入以交换空间

    与文件不关联的页面 (如栈和堆)- 匿名内存

    在内存中修改但尚未写回文件系统的页面;v 移动系统

    通常不支持交换
    相反,从文件系统请求页面并回收只读页面 (如代码)

    # 页面置换算法

    # 空闲帧列表

    当出现页面错误时,操作系统必须将所需页面从辅助存储器带入主存。

    大多数操作系统都维护一个空闲帧列表 —— 一个用于满足此类请求的空闲帧池。

    操作系统通常使用一种称为按需零填充 (zero-fill-on-demand) 的技术来分配空闲帧,即在分配前将帧的内容清零。

    当系统启动时,所有可用内存都会放在空闲帧列表中。

    # 页面置换

    一定发生在缺页故障处理中,空闲帧列表基础上的补充,防止内存过度分配

    • 不一定要在空闲帧列表为空的时候才进行页面置换,也可能在空闲帧数量低于某一阈值的时候就开始
    • 往往不是一个页面一个页面置换,因为这样效率比较低,而是批处理多个页面进行置换

    使用修改 (dirty) 位来减少⻚面(内存到磁盘)传输的开销 —— 只有修改过的 victim ⻚面才会写入磁盘;

    ⻚面替换完成了逻辑内存和物理内存之间的分离 —— 可以 在较小的物理内存上提供较大的虚拟内存;

    # FIFO

    页面错误与帧数关系图

    Belady 异常,出现帧数增加后缺页异常数量也增加的现象,只会在 FIFO 调度中出现

    页面错误与帧数关系图belady

    # OPT 算法

    最优页面置换算法

    替换在最长时间内不使用的页面,实际上不能预测,所以是不可行的,但可以用于理论上测量其他算法性能,作为比较的基线

    # LRU 算法

    最少最近使用 (LRU) 算法

    使用过去的知识而不是未来的知识,替换在最长时间内未使用的页面,将最后一次使用的时间与每页关联

    一般来说,算法很好,经常使用

    • 计数器实现,为每一个页面(页表项)设置计数位,每次页面被引用,将当前时钟的时间戳复制到计数位
    • 使用堆栈记录最近的页面引用,栈顶为最近一次引用的页面,使用双向链表排序

    # 额外引用位算法

    通过定期记录引用位,可以获得额外的排序信息;

    为每个页面的页表项设置一个 8 位的字节;

    将每个页面的引用位移到 8 位字节的高位,右移一位,并丢 弃最低位,保存 8 个时间周期的页面使用情况;

    具有 11000100 的历史寄存器值的页面比具有值为 01110111 的 页面更为 “最近使用”;

    替换最小编号;

    # 第二次机会 / 时钟算法

    通常为 FIFO,加上硬件提供的引用位。为每个页面设置一个访问位,再将内存中的页面都通过链接指针链接成一个循环队列

    一般默认初始情况没有调入任何页面,依次调入页面后,各个页面的访问位为 1 ,替换指针指向第一个页帧。

    替换指针总是指向上一次被调入页面的下一个页帧!若为最后一个页帧则回到第一个页帧!

    替换规则:

    • 引用位 = 0 ,替换它,然后替换指针指向下一个页帧。
    • 引用位 = 1 ,则:
      • 设置引用位 0 ,将页面保留在内存中
      • 替换指针指向下一个页帧,继续找按该规则 victim

    第二轮扫描中一定会有访问位为 0 的页面,因此简单的 CLOCK 算法选择一个淘汰页面最多会经过两轮扫描

    # 增强型第二次机会算法

    在第二次机会 / 时钟置换算法的基础上,除了考虑一个页面最近有没有被访问过之外,操作系统还应考虑页面有没有被修改过。在其他条件都相同时,应优先淘汰没有修改过的页面。

    为每个页面页表项增加修改位(dirty):

    • 修改位 = 0 ,表示页面没有被修改过;
    • 修改位 = 1 ,表示页面被修改过。

    方便讨论,用 (访问位,修改位) 的形式表示各页面状态:

    • (0,0) 最近未使用未修改 —— 要替换的最佳页面
    • (0,1) 最近未使用但已修改 —— 不太好,必须在置换之前将页面写出;
    • (1,0) 最近使用但没有修改 —— 可能很快会再次使用
    • (1,1) 最近使用和修改的 —— 可能很快会再次使用,需要在更换前将页面写出;

    将所有可能被置换的页面排成一个循环队列,和原本的时钟置换相比,每一轮的操作都有不同。

    • 第一轮:从当前位置开始扫描到第一个 (0,0) 的帧用于替换。本轮扫描不修改任何标志位
    • 第二轮:若第一轮扫描失败,则重新扫描,查找第一个 (0,1) 的帧用于替换。本轮将所有扫描过的帧访问位设为 0
    • 第三轮:若第二轮扫描失败,则重新扫描,查找第一个 (0,0) 的帧用于替换。本轮扫描不修改任何标志位
    • 第四轮:若第三轮扫描失败,则重新扫描,查找第一个 (0,1) 的帧用于替换。

    这里的第 2、3 轮相当于原来的时钟置换算法的 (可能的), 2 轮。

    由于第二轮已将所有帧的访问位设为 0 ,因此经过第三轮、第四轮扫描一定会有一个帧被选中,因此改进型时钟置换算法选择一个淘汰页面最多会进行四轮扫描

    # 页面缓冲算法

    始终保持一个空闲帧池

    • 当出现缺页错误时,会像以前一样选择一个牺牲帧;

    • 将页面读入空闲帧,选择要退出的牺牲帧并添加到空闲 池;

    • 方便时,驱逐牺牲帧,无需等待写出牺牲帧;v 可能的话,保留修改过的页面列表

      当备份存储为空闲时,在那里写入页面并设置为非脏;增加了无需写出的页面的概率;

    可能的话,保持空闲帧池,并且记住哪些页面在哪些帧内;

    如果在重用之前再次引用,则无需再次从磁盘加载内容

    如果选择了错误的受害者帧,通常有助于减少惩罚

    双缓冲

    • 操作系统将页面副本作为 I/O 缓冲区保存在内存中
    • 应用程序将页保留在内存中以供自己工作;

    可能带来好处,也可能带来坏处

    \ No newline at end of file +内存管理 - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    冯・诺伊曼计算机体系决定了操作系统需要内存管理

    # 虚拟内存物理实现

    • 连续分配
      • 单一连续分配
      • 分区
        • 固定分区分配
        • 动态分区分配
    • 非连续分配
      • 段式
      • 页式
      • 段页式

    ⭐️内部碎片和外部碎片

    • 内部碎片
      • 是已经被分配给某个进程的,该进程没有利用的
      • 固定分区分配产生这种碎片
    • 外部碎片
      • 是无法被分配给任何进程的,不属于任何进程的
      • 动态分区分配产生这种碎片
      • 可以通过紧凑技术克服,需要动态重定位寄存器硬件支持

    # 分区 (连续分配)

    # 单一连续分配

    将内存分为两部分:

    • 系统区
    • 用户区
      • 唯一一个用户程序独占整个用户区

    内存中只有一个用户程序,只能用于单用户、单任务的操作系统。

    有内部碎片,无外部碎片。

    # 固定分区

    使用空闲分区链 / 表

    有内部碎片,无外部碎片。

    # 动态分区

    使用空闲分区链 / 表

    无内部碎片,有外部碎片。

    # 首次适应

    空闲分区按地址递增顺序排列。

    每次分配时从头开始顺序查找到第一个满足大小的空闲分区。

    性能最好。

    # 邻近适应

    空闲分区按地址递增顺序排列。和首次适应一样。

    每次分配时从上次查找结束位置开始查找到第一个满足大小的空闲分区。

    # 最佳适应

    空闲分区按容量递增顺序排列。

    产生最多外部碎片

    # 最坏适应

    空闲分区按容量递减顺序排列。

    前面三种都基于顺序搜索,后面三种则都基于索引搜索

    # 快速适应

    基于索引分配,对空闲分区按大小相同的分为一类,每一类单独设置一个空闲分区链。建立一张索引表管理所有空闲分区链,索引表项包含对应大小的空闲分区链的头指针

    根据进程需要的大小,在索引表中找到能容纳它的最小的空闲分区链表,直接从链表取出第一块进行分配。

    优劣:

    • 优点
      • 查找效率高
      • 不产生内部碎片(当然啊这不是动态分区吗)
    • 缺点
      • 回收分区复杂,导致算法复杂、系统开销大

    # 伙伴系统

    我感觉应该算固定分区分配才对?

    规定所有分区大小为 22kk 次幂。

    每次分配大小为 nn (2i1<n2i2^{i-1}<n\leq 2^i) 的分区时,在

    导致过多内部碎片⚠️

    Slab 分配器 —— 一般只有内核内存使用

    # 哈希算法

    根据空闲分区链表的分布规律建立哈希函数,构建一张以哈希表:

    • 关键字 key 为空闲分区大小
    • 表项 value 为对应空闲分区链的头指针

    # 分段

    程序员或者编译器会把程序和数据指定到不同的分段,需要知道段的最大长度限制

    将程序产生的二维地址即 <段号,偏移><段号,偏移> 映射到一维物理地址, 这个地址是通过段表实现;

    段表中每个条目由段基址和段界限构成

    CPU 去哪里找段表?—— GDTR 寄存器,GDTR 寄存器的结构如下:

    • 低 2 字节(16 位)存储了 limit ,全局描述符表的大小(以字节为单位),减去 1 的结果。
    • 高 4 字节(32 位)存储了 base ,全局描述符表的基地址。

    # 分页

    页放到页帧 / 框里,区分:⚠️

    • 页(Page)
    • 页帧(Page frame)或者页框,是空的,没有数据

    页框 \geq 页,现代操作系统大多二者是相等的,方便管理

    # 页表

    系统中有一个唯一的页表寄存器(PTR),包含页表首地址(PTBR),只能由特权指令访问。

    页表长度寄存器(PTLR),可以直接用于判断是否越界(好像不一定有)

    x86_64 , i386 中的 PTR 寄存器叫做 CR3 寄存器

    页表从属于进程,每创建一个新进程就需要创建一个独立的新页表给它。

    # TLB

    转换表缓冲区 / 快表 /... 详细参考计组

    # 共享页

    共享代码

    • 进程 (即文本编辑器、编译器、窗口系统) 之间共享的只读 (可重入) 代码的一个副本;
    • 类似于共享同一进程空间的多个线程;
    • 如果允许共享读写页面,则还可用于进程间通信;

    # 多级 / 层页表

    经典:在现代计算机上考虑 32 位逻辑地址空间,页面大小为 4KB (2122^{12}),页面表将有 4M 个条目 (232/212=2202^{32}/2^{12}=2^{20}) ,如果每个条目为 4B ,每个进程需要为页表分配 16MB 的物理地址空间,这片连续的物理空间太大了!

    相同条件下考虑 64 位,即使采用 2 级分页,外层页表也需要 16384GB 大小的连续物理空间!所以 x86_64 采用 4 级页表

    现代 OS 大多采用 3 到 4 级页表

    64 位下的 2 级页表 和 3 级页表:

    64-2-3

    # 散列页表 / 哈希表

    # 聚簇页表

    64 位地址的变体,哈希值对应的不是一个页,而是一簇页(一个页族)

    特别适用于稀疏地址空间 (内存引用不连续且分散)

    # 倒排页表

    不再为每个进程设置一个页表并跟踪所有可能的逻辑页,而是跟踪所有物理页,此时整个架构自 CPU 开始到 OS 都有较大变化

    内存的每一物理页有一个条目,条目包含:

    • 存储在该实际内存位置的页的虚拟地址
    • 有关拥有该页的进程的信息组成

    CPU 产生的逻辑地址:

    倒排页表

    优点:

    • 减少存储每个页表所需的内存

    缺点:

    • 发生页引用时增加搜索表所需的时间
    • 共享内存不太容易实现:虚拟地址到共享物理地址的一种映射

    # 现代 OS 举例

    # Oracle SPARC Solaris

    考虑现代 64 位操作系统目标是效率,低开销

    基于哈希,但更复杂

    两个哈希表

    • 一个内核
    • 一个用于所有用户进程

    每个都将内存地址从虚拟内存映射到物理内存

    每个条目表示映射虚拟内存的连续区域

    比为每个页面提供单独的哈希表条目更高效

    每个条目都有基址和跨距 (表示条目所代表的页数)

    # IA-32/64

    英特尔 IA-32 和 IA-64 均支持分段、分页管理

    记录段号的基地址的描述符表,按私有、公有分为:

    • LDT 局部描述符表
    • GDT 全局描述符表

    早期 IA-32 利用 PAE 实现页地址扩展

    # ARM

    Apple

    段页混合式管理,较大连续内存走分段,较小连续内存走分页:

    arm

    # 虚拟内存软件实现

    ⭐️虚拟内存的实现建立在离散分配的内存管理方式基础上

    三个特征:

    • 多次性
    • 对换性
    • 虚拟性

    虚拟内存可以通过以下方式实现:

    • 请求分页
    • 请求分段(按需分段)
    • 请求段页式存储管理

    # 请求分页

    按需分页

    • 固定分配局部置换
    • 可变分配全局置换
    • 可变分配局部置换

    请求分页系统的外存

    • 文件区
      • 存放文件,采用离散分配方式
    • 对换区(交换区)
      • 存放对换页面,采用连续分配方式,I/O 速度更快

    可以在加载时将整个进程放入内存或者只在需要时才将页面放入内存

    所需的 I/O 更少,没有不必要的 I/O

    • 所需内存更少
    • 更快的响应
    • 更支持多用户

    类似于带交换的分页系统 (需要硬件支持)

    • 页面是必需的 —— 引用它
    • 无效引用 —— 终止
    • 不在内存中 —— 调入内存

    惰性交换程序:除非需要页,否则永远不要将页交换到内存中

    处理页面的交换程序是 “调页程序”

    # 处理缺页

    1. 检查进程的内部表,通常与 PCB 一起保存,以确定内存引用是否有效;
    2. 如果:(页面错误 (缺页) )
      • 无效,那么终止进程
      • 引用有效但是未调入页面, 那么发生缺页中断;
    3. 找到空闲帧;(或 victim 帧)
    4. 通过磁盘调度操作将页面交换到刚分配的帧中;
    5. 当磁盘操作读取完成后,重置内部表和页表,以指示当前页面在内存中;
    6. 重新启动导致缺页的指令,该进程能够访问所需的页;(指令重启

    # 问题

    # 启动进程时内存中没有页面

    操作系统将指令指针设置为进程的第一条指令,非内存驻留 -> 页面错误

    • 进程其他页面的访问也是一样;
    • 纯请求分页;

    # 多页错误

    • 考虑从内存中将 2 个数加和并将结果存储回内存的指令的读取和解码,可能产生多个缺页错误;
    • 产生性能痛点,幸运的是因为局部引用的存在,使得请求分页 具有较为合理的性能;

    # 硬件支持

    具有有效 / 无效位的页表
    辅助内存 (带交换空间的交换设备)

    指令重启,在缺页错误后能够重启任何指令;

    # 指令重启

    • 微代码计算试图访问源块和目的块的两端,如果出现缺页错误,那么在这一步出现,完成缺页逻辑后,执行移动操作;
    • 使用临时寄存器来保存覆盖位置的值,如果有缺页错误,则在陷阱发生之前,所有旧值都将写回到内存中。该动作是将内存恢复到指令启动之前的状态,这样能够重复执行该指令;

    # 写时复制

    写时复制 (CoW) 允许父进程和子进程最初共享内存中的相同页面

    如果任一进程修改了共享页面,则该页面被复制

    CoW 允许更高效的进程创建 (fork) ,因为只复制修改过的页面

    # 系统抖动

    频繁发生缺页,系统忙于进行页面的换入和换出,而没有进行有效的计算。

    系统抖动

    导致 CPU 利用率低,并发度低。

    系统还会自动检测 CPU 利用率,系统抖动时检测到 CPU 利用率降低,会创建更多精进程来提高 CPU 利用率,但这会反而加剧系统抖动。

    局部性原理告诉我们,访问磁盘总是在一个局部到另一个局部变化。而系统抖动从这个角度来说,就是局部性访问的大小超过了内存大小,所以导致了系统抖动。

    本地或优先级替换算法来缓解系统抖动。

    # 工作集

    WSSiWSS_i (Working Set of Process PiP_i ):进程在某一个固定的时间窗口 Δ\Delta(可以按时间算、指令数算)内,访问的所有页面

    • if Δ\Delta too small will not encompass entire locality
    • if Δ\Delta too large will encompass several localities
    • if Δ=\Delta=\infin will encompass entire program

    最理想的情况就是 WSS 能够被全部放在 TLB 中

    ⭐️ tt 时刻的工作集为该时刻及其以前 Δ\Delta 时间窗口范围内的访问的页面。

    解决方案 1

    建立 “可接受的” PFF (page-fault frequency) 并使用本地替换策略

    内核的内存管理就比较简单,因为它在内存里一定是连续的,可以完全利用线性地址偏移解决。

    预调页

    访问到某一内存页,很有可能下次就要访问它的邻居内存页,所以利用空间局部性

    在程序启动的时候可以减少大量的缺页

    当然也是有可能造成 I/O 和内存的浪费的

    # 帧分配

    在进程之间如何分配固定数量的可用内存?

    • 固定分配

    • 优先级分配

    • 全局置换 —— 进程从所有帧集中选择替换帧,一个进程可以从另一个进程获取帧

      • 高优先级进程可以置换低优先级进程
      • 缺点:进程不能控制他自己的缺页错误率
    • 局部置换 —— 每个进程仅从其自己的一组分配帧中进行选择

      • 每个进程的性能更加一致
      • 但可能是未充分利用的内存

    # 页面回收

    # 性能指标

    Three major activities

    • Service the interrupt – careful coding means just several hundred instructions needed
    • Read the page – lots of time
    • Restart the process – again just a small amount of time

    Effective Access Time (EAT)

    EAT=(1p)×MA+p×(PFO+SPOSPI)EAT=(1–p)\times MA+p\times(PFO+SPO-SPI)

    • Page Fault Rate 0p10\leq p\leq1
      • if p=0p=0, no page faults
      • if p=1p=1, every reference is a fault
    • MAMA : memory access
    • PFOPFO : page fault overhead
    • SPOSPO : swap page out ( if dirty )
    • SPISPI : swap page in

    # 性能优化

    交换空间 I/O 比文件系统 I/O 快,即使在同一设备上也是如此

    交换分配在更大的块中,比文件系统所需的管理更少;

    在进程加载时将整个进程映像复制到交换空间

    然后在交换空间内执行调页;
    在旧的 BSD Unix 中使用;

    从磁盘上的程序二进制文件请求分页,但在释放帧时将页面直 接放弃 (被覆盖),而不是分页调出;

    在 Solaris 和当前 BSD 中使用 § 仍然需要写入以交换空间

    与文件不关联的页面 (如栈和堆)- 匿名内存

    在内存中修改但尚未写回文件系统的页面;v 移动系统

    通常不支持交换
    相反,从文件系统请求页面并回收只读页面 (如代码)

    # 页面置换算法

    # 空闲帧列表

    当出现页面错误时,操作系统必须将所需页面从辅助存储器带入主存。

    大多数操作系统都维护一个空闲帧列表 —— 一个用于满足此类请求的空闲帧池。

    操作系统通常使用一种称为按需零填充 (zero-fill-on-demand) 的技术来分配空闲帧,即在分配前将帧的内容清零。

    当系统启动时,所有可用内存都会放在空闲帧列表中。

    # 页面置换

    一定发生在缺页故障处理中,空闲帧列表基础上的补充,防止内存过度分配

    • 不一定要在空闲帧列表为空的时候才进行页面置换,也可能在空闲帧数量低于某一阈值的时候就开始
    • 往往不是一个页面一个页面置换,因为这样效率比较低,而是批处理多个页面进行置换

    使用修改 (dirty) 位来减少⻚面(内存到磁盘)传输的开销 —— 只有修改过的 victim ⻚面才会写入磁盘;

    ⻚面替换完成了逻辑内存和物理内存之间的分离 —— 可以 在较小的物理内存上提供较大的虚拟内存;

    # FIFO

    页面错误与帧数关系图

    Belady 异常,出现帧数增加后缺页异常数量也增加的现象,只会在 FIFO 调度中出现

    页面错误与帧数关系图belady

    # OPT 算法

    最优页面置换算法

    替换在最长时间内不使用的页面,实际上不能预测,所以是不可行的,但可以用于理论上测量其他算法性能,作为比较的基线

    # LRU 算法

    最少最近使用 (LRU) 算法

    使用过去的知识而不是未来的知识,替换在最长时间内未使用的页面,将最后一次使用的时间与每页关联

    一般来说,算法很好,经常使用

    • 计数器实现,为每一个页面(页表项)设置计数位,每次页面被引用,将当前时钟的时间戳复制到计数位
    • 使用堆栈记录最近的页面引用,栈顶为最近一次引用的页面,使用双向链表排序

    # 额外引用位算法

    通过定期记录引用位,可以获得额外的排序信息;

    为每个页面的页表项设置一个 8 位的字节;

    将每个页面的引用位移到 8 位字节的高位,右移一位,并丢 弃最低位,保存 8 个时间周期的页面使用情况;

    具有 11000100 的历史寄存器值的页面比具有值为 01110111 的 页面更为 “最近使用”;

    替换最小编号;

    # 第二次机会 / 时钟算法

    通常为 FIFO,加上硬件提供的引用位。为每个页面设置一个访问位,再将内存中的页面都通过链接指针链接成一个循环队列

    一般默认初始情况没有调入任何页面,依次调入页面后,各个页面的访问位为 1 ,替换指针指向第一个页帧。

    替换指针总是指向上一次被调入页面的下一个页帧!若为最后一个页帧则回到第一个页帧!

    替换规则:

    • 引用位 = 0 ,替换它,然后替换指针指向下一个页帧。
    • 引用位 = 1 ,则:
      • 设置引用位 0 ,将页面保留在内存中
      • 替换指针指向下一个页帧,继续找按该规则 victim

    第二轮扫描中一定会有访问位为 0 的页面,因此简单的 CLOCK 算法选择一个淘汰页面最多会经过两轮扫描

    # 增强型第二次机会算法

    在第二次机会 / 时钟置换算法的基础上,除了考虑一个页面最近有没有被访问过之外,操作系统还应考虑页面有没有被修改过。在其他条件都相同时,应优先淘汰没有修改过的页面。

    为每个页面页表项增加修改位(dirty):

    • 修改位 = 0 ,表示页面没有被修改过;
    • 修改位 = 1 ,表示页面被修改过。

    方便讨论,用 (访问位,修改位) 的形式表示各页面状态:

    • (0,0) 最近未使用未修改 —— 要替换的最佳页面
    • (0,1) 最近未使用但已修改 —— 不太好,必须在置换之前将页面写出;
    • (1,0) 最近使用但没有修改 —— 可能很快会再次使用
    • (1,1) 最近使用和修改的 —— 可能很快会再次使用,需要在更换前将页面写出;

    将所有可能被置换的页面排成一个循环队列,和原本的时钟置换相比,每一轮的操作都有不同。

    • 第一轮:从当前位置开始扫描到第一个 (0,0) 的帧用于替换。本轮扫描不修改任何标志位
    • 第二轮:若第一轮扫描失败,则重新扫描,查找第一个 (0,1) 的帧用于替换。本轮将所有扫描过的帧访问位设为 0
    • 第三轮:若第二轮扫描失败,则重新扫描,查找第一个 (0,0) 的帧用于替换。本轮扫描不修改任何标志位
    • 第四轮:若第三轮扫描失败,则重新扫描,查找第一个 (0,1) 的帧用于替换。

    这里的第 2、3 轮相当于原来的时钟置换算法的 (可能的), 2 轮。

    由于第二轮已将所有帧的访问位设为 0 ,因此经过第三轮、第四轮扫描一定会有一个帧被选中,因此改进型时钟置换算法选择一个淘汰页面最多会进行四轮扫描

    # 页面缓冲算法

    始终保持一个空闲帧池

    • 当出现缺页错误时,会像以前一样选择一个牺牲帧;

    • 将页面读入空闲帧,选择要退出的牺牲帧并添加到空闲 池;

    • 方便时,驱逐牺牲帧,无需等待写出牺牲帧;v 可能的话,保留修改过的页面列表

      当备份存储为空闲时,在那里写入页面并设置为非脏;增加了无需写出的页面的概率;

    可能的话,保持空闲帧池,并且记住哪些页面在哪些帧内;

    如果在重用之前再次引用,则无需再次从磁盘加载内容

    如果选择了错误的受害者帧,通常有助于减少惩罚

    双缓冲

    • 操作系统将页面副本作为 I/O 缓冲区保存在内存中
    • 应用程序将页保留在内存中以供自己工作;

    可能带来好处,也可能带来坏处

    \ No newline at end of file diff --git a/cs/os/process/index.html b/cs/os/process/index.html index aa814c57..51ee42bc 100644 --- a/cs/os/process/index.html +++ b/cs/os/process/index.html @@ -1 +1 @@ -进程 - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 进程的基本概念

    # 进程的定义

    典型的定义:

    • 进程是进程实体的运行过程,是系统进行资源分配和调度的一个独立单位 ==
    • 一个具有独立功能的程序在一个数据集合上运行的过程

    # 区分进程程序进程实体

    • 程序(Program):是静态的,就是个存放在磁盘里的可执行文件,就是一系列的指令集合

      • 静态连接
      • 动态链接
    • 进程(Process):是动态的,是程序的一次执行过程

    • 进程实体(进程映像):是静态的,反应了进程在某一时刻的状态

    # 进程实体的组成

    一个 == 进程实体(进程映像)== 由三个部分组成:

    • PCB
      • 进程描述信息
      • 进程控制和管理信息
      • 资源分配清单
      • 处理机状态信息(CPU 上下文
    • 程序段
      • 程序的代码(指令序列)
    • 数据段
      • 运行过程中产生的各种数据(如:程序中定义的变量)

    PCB 是给操作系统用的;程序段、数据段是给进程自己用的,与进程自身的运行逻辑有关

    # PCB

    PCB 位于内存的内核空间里,创建之后常驻内存。

    PCB 是进程存在的唯一标志,当进程被创建时,操作系统为其创建 PCB,当进程结束时,会会回收其 PCB。

    PCB 几乎完备的表示了进程的状态,也就是说只要有一个进程的 PCB,就可以恢复对应的进程

    # PCB 的结构

    • 进程描述信息

      • 进程标识符 PID
      • 用户标识符 UID
    • 进程控制和管理信息

      • CPU、磁盘、网络流量使用情况统计⋯
        • 进入内存时间
        • CPU 占用时间
        • 信号量使用
      • 进程当前状态:就绪态 / 阻塞态 / 运行态
      • 代码运行入口地址
      • 程序的外存地址
      • 程序优先级
    • 资源分配清单

      • 正在使用(依赖)哪些文件
        • 文件描述符
      • 正在使用哪些内存区域
        • 代码段指针
        • 数据段指针
        • 堆栈段指针
      • 正在使用哪些 I/O 设备
    • 处理机相关信息,也叫 CPU 的上下文(the context of the processor)

      • user-visible registers
      • control and status registers (CSR)
      • stack pointers
      • PSW
        • eg: EFLAGS in x86

    # Linux 的 PCB

    Linux 中的进程控制块 PCB 源码位于 linux-x.xx.xx/include/linux/sched.h

    Linux 不区分进程和线程,将它们都视为 task

    由 C 结构 task_struct 表示:

    c
    {
      pid t_pid;/* 进程标识符 */
      unsigned int time_slice /* 调度信息 */
      struct task_struct* 父进程;/* 此进程的父级 */
      struct list_head children;/* 此进程的子进程 */
      struct *files;/* 打开文件列表 */
      struct mm_struct*mm;/* 此进程的地址空间 */
      ...
    }

    task_struct

    # 进程的特征

    • 动态性 —— 进程是程序的一次执行过程,是动态地产生、变化和消亡的(动态性是进程最基本的特征)
    • 并发性 —— 内存中有多个进程实体,各进程可并发执行
    • 独立性 —— 进程是能独立运行、独立获得资源、独立接受调度的基本单位
    • 异步性 —— 各进程按各自独立的、不可预知的速度向前推进,操作系统要提供 “进程同步机制 " 来解决异步问题
    • 结构性 —— 每个进程都会配置一个 PCB。结构上看,进程由程序段、数据段、PCB 组成

    # 进程的状态与转换

    The steps in a full process switch are:

    1. save the context of the processor
    2. update the PCB of the process currently in the Running state
    3. move the PCB of this process to the appropriate queue
    4. select another process for execution
    5. update the PCB of the process selected
    6. update memory management data structures
    7. restore the context of the processor to that which existed at the time the selected process was last switched out

    # 五状态模型

    五状态模型

    # 七状态模型

    增加了挂起状态的进程


    挂起进程的特征:

    • 进程不能立即执行(上处理机)

    • 进程可能有也可能么有等待一个事件(为什么那么像废话)

    • 为阻止该进程执行,可通过代理使其处于挂起状态;代理可以是:

      • 进程本身

      • 父进程

      • 操作系统

    • 只有代理程序显示地命令操作系统进行状态轮换,才能使该进程从挂起的状态转移


    进程挂起的原因:

    • Swaping
    • Other OS reason
    • Interactive user request
    • Timing
    • Parent pocess request

    双挂起状态的进程模型:

    七状态模型

    # 进程组织

    进程组织的数据结构

    # 链接式组织进程

    链式组织进程

    # 索引式组织进程

    索引式组织进程

    # 进程控制

    也就是对进程的操作,这些基本操作几乎全部都是系统调用实现的

    # 系统调用

    Some basic and important system calls

    PCB 的访问:

    getpid()

    • fork()
    • exec*()
    • wait()
    • exit()

    # Linux 举例

    父进程创建子进程,子进程又创建其他进程,形成进程树;父子之间不同角度可能有不同关系:

    • 资源共享
      • 父和子共享所有资源 (Linux 中父进程 vfork )
      • 子进程共享父进程资源的子集 (Linux 中父进程 vfork )
      • 父进程和子进程不共享任何资源 (Linux 中父进程 fork )
    • 执行
      • 父进程和子进程同时并发执行
      • 父进程等待直到子进程终止 (Linux 中父进程 wait(0) )
    • 地址空间
      • 子进程是父进程的复制品 (Linux 中父进程 fork )
      • 子进程加载另一个新的程序
      • 子进程完全覆盖父进程 (Linux 中父进程 fork + exec )
        • exec family 有 6 个成员
        • execl 装载新的文件

    # fork

    Linux 中,由父进程调用系统调用 fork() 创建子进程

    c
    #include<stdio.h>
    #include<unistd.h>
    int main (){
      for(int i=0;i<4;++i){
        fork();
      }
    	printf("This is a test printed by %d\n", getpid());
    return 0;
    }

    编译运行上述程序将打印出 16 次!因为这里子进程是父进程的复制品:

    shell
    node1@vm1:~/Desktop$ gcc -o test test.c
    node1@vm1:~/Desktop$ ./test
    This is a test printed by 139377
    This is a test printed by 139382
    This is a test printed by 139380
    This is a test printed by 139379
    This is a test printed by 139378
    This is a test printed by 139383
    node1@vm1:~/Desktop$ This is a test printed by 139381 # 父进程结束
    This is a test printed by 139386
    This is a test printed by 139387
    This is a test printed by 139385
    This is a test printed by 139388
    This is a test printed by 139389
    This is a test printed by 139384
    This is a test printed by 139390
    This is a test printed by 139391
    This is a test printed by 139392

    fork 会克隆的内容:

    Cloned itemsDescriptions
    Program code [File & Memory]They are sharing the same piece of code.
    MemoryIncluding local variables, global variables, and dynamically allocated memory.
    Opened files [Kernel's internal]If the parent has opened a file "A", then the child will also have file "A" opened automatically.
    Program counter [CPU register]That's why they both execute from the same line of code after fork() returns.

    fork 不克隆的内容:

    Distinct itemsParentChild
    Return value of fork() PID of the child process.0
    PIDUnchanged.Different, not necessarily be "Parent PID + 1"
    Parent processUnchanged.Doesn't have the same parent as that of the parent process.
    Running timeCumulated.Just created, so should be 0.

    vfork 函数是 fork 函数基础上的修正, fork 函数创建一个完全分离的进程,而 vfork 用于创建一个共享的进程,没有克隆操作,这意味着父进程子进程可以互相看到对方,共享同样的物理地址

    # exec*

    exec* 系统调用,会 “鸠占鹊巢”,替换掉原进程的代码,但是 PID、父子关系、进程运行时间都不变,也就是:

    • 内核里的内容不变,包括 PCB 等
    • 用户态里的代码被替换
    c
    #include<stdio.h>
    #include<unistd.h>
    int main(void) {
    	printf("before execl ...\n"); // 原进程代码可以执行
    	execl("/bin/ls", "/bin/ls", NULL); //exec 的一个成员函数
    	printf("after execl ...\n"); // 原进程代码无法执行
    	return 0;
    }

    编译执行结果:

    shell
    node1@vm2:~/Desktop$ gcc -o test test.c
    node1@vm2:~/Desktop$ ./test
    before execl ...
    sdb  stack_rwx.c  test  test01  test.c  ysos

    # waitwaitpid

    wait()waitpid()
    Wait for any one of the children.Depending on the parameters, waitpid() will wait for a particular child only.
    Detect child termination only.Depending on the parameters, waitpid() can detect child's status changing: -from running to suspended, and -from suspended to running.

    # fork + exec*

    执行顺序是不确定的

    c
    #include<stdio.h>
    #include<unistd.h>
    int system_test(const char *cmd_str) {
        if(cmd_str == -1)
            return -1;
        if(fork() == 0) {
            execl(cmd_str, cmd_str, NULL);
            fprintf(stderr,"%s: command not found\n", cmd_str);
            exit(-1);
        }
        return 0;
    }
    int main(void) {
        printf("before...\n\n");
        system_test("/bin/ls");
        printf("\nafter...\n");
        return 0;
    }

    编译执行结果:

    shell
    node1@vm2:~/Desktop$ gcc -o test test.c
    node1@vm2:~/Desktop$ ./test
    before...
    after...
    sdb  stack_rwx.c  test  test01  test.c  ysos # 执行 ls

    类似于一个 shell 或 system ,不过 system 创建子进程,时串行执行,执行顺序是确定的。

    # fork + exec* + wait

    等价于 system

    c
    #include<stdio.h>
    #include<unistd.h>
    int system_test(const char *cmd_str) {
        if (cmd_str == -1)
            return -1;
        if(fork() == 0) {
            execl("/bin/sh", "/bin/sh","-c", cmd_str, NULL);
            fprintf(stderr,"%s: command not found\n", cmd_str);
            exit(-1);
        }
        wait(NULL);
        return 0;
    }
    int main(void) {
        printf ("before...\n\n"); system_test("/bin/1s");
        printf("\nafter...\n");
        return 0;
    }

    编译执行结果:

    shell
    node1@vm2:~/Desktop$ gcc -o test test.c
    node1@vm2:~/Desktop$ ./test
    before...
    sdb  stack_rwx.c  test  test01  test.c  ysos
    after...

    # exit

    父进程 wait 和子进程 exit 搭配食用。子进程执行最后一条语句,然后调用 exit 系统调用请求操作系统删除它自己,然后子进程将发送 SIGCHLD 给父进程,并变僵尸进程;父进程接收到 SIGCHLD 后被唤醒(原本被 wait 休眠),处理 SIGCHLD 之后,子进程才被彻底销毁。

    c
    #include<stdio.h>
    #include<unistd.h>
    #include<stdlib.h>
    int main(void){
        int count = 1;
        pid_t childpid, terminatedid;
        childpid = fork(); /* child duplicates parent’s address space */
        if (childpid < 0) {
            perror("fork()");
            return EXIT_FAILURE;
        }
        else if (childpid == 0) { /* This is child pro */
            count++;
            printf("child pro pid = %d, count = %d (addr = %p)\n", getpid(), count, &count);
            printf("child sleeping ...\n");
            sleep(5); /* parent wait() during this period */
            printf("\nchild waking up!\n");
        }
        else { /* This is parent pro */
            terminatedid = wait(0);
            printf("parent pro pid = %d, terminated pid = %d, count = %d (addr = %p)\n",
            getpid(), terminatedid, count, &count);
        }
        printf("\nTesting point by %d\n", getpid()); /* executed by child and parent */
        return EXIT_SUCCESS;
    }

    编译运行结果:

    shell
    node1@vm2:~/Desktop$ ./test
    child pro pid = 197382, count = 2 (addr = 0x7ffc33c444cc)
    child sleeping ...
    # 这里等待了 5 秒钟
    child waking up!
    Testing point by 197382
    parent pro pid = 197381, terminated pid = 197382, count = 1 (addr = 0x7ffc33c444cc)
    Testing point by 197381

    僵尸进程:空有 PCB “外壳”,但是没有用户态的代码,无法执行;

    注意区分孤儿进程

    # abort

    父进程可以使用 abort 系统调用终止子进程的执行。这样做的一些原因:

    • 子进程已超出分配的资源
    • 不再需要分配给子进程的任务
    • 父进程正在退出,如果父进程终止,操作系统不允许子进程继续

    # 进程控制原语

    进程控制由原语实现

    如果不能像原语 “一气呵成”,就有可能导致操作系统中的某些关键数据结构信息不统一的情况,这会影响操作系统进行别的管理工作

    原语的执行具有原子性,即执行过程只能一气呵成,期间不允许被中断。

    可以用 “关中断指令”“开中断指令” 这两个特权指令实现原子性

    无论哪个进程控制原语,要做的无非三类事情:

    • 更新 PCB 中的信息
      • 修改进程状态(state)
      • 保存 / 恢复运行环境
    • 将 PCB 插入合适的队列
    • 分配 / 回收资源

    # 创建

    • 申请空白 PCB
    • 为新进程分配所需资源
    • 初始化 PCB
    • 将 PCB 插入就绪队列(进程由阻塞态转变为就绪态

    引起进程创建的事件可能有:

    • 用户登录 —— 分时系统中,用户登录成功,系统会建立为其建立一个新的进程
    • 作业调度 —— 多道批处理系统中,有新的作业放入内存时,会为其建立一个新的进程
    • 提供服务 —— 用户向操作系统提出某些请求时,会新建一个进程处理该请求
    • 应用请求 —— 由用户进程主动请求创建一个子进程

    所有进程都是以父进程创建子进程的方式创建的:

    • 有些是显式的
      • 用户进程创建的
    • 有些是隐式的
      • 操作系统 “0 号” 根进程创建的(所有进程的祖先)

    # 根进程

    The first process -- The kernel, while it is booting up, creates the firstprocess -- init .

    The init process:

    • has PID = 1
    • and is running the program code “/sbin/init”.
    • Its first task is to create more processes…

    # 孤儿进程

    父进程创建子进程一层一层创建,但若其中某一个父进程 terminate 了,也就是被终止了,那么被终止进程所创建的进程就变成了孤儿进程。孤儿进程会消耗资源,需要被重新管理起来。

    • Linux 中,使用 re-parent operation 处理孤儿进程, The “init” process will become the step-mother of all orphans.
    • Windows maintains a forest-like hierarchy.

    # 撤销

    这一过程进程: 就绪态 / 阻塞态 / 运行态 -> 终止态 -> 无

    • 从 PCB 集合中找到终止进程的 PCB
    • 若进程正在运行,立即剥夺 CPU,将 CPU 分配给其他进程
    • 终止其所有子进程
    • 将该进程拥有的所有资源归还给父进程或操作系统
    • 删除 PCB

    引起进程终止的事件可能有:

    • 正常结束 —— 进程自己请求终止(exit 系统调用)
    • 异常结束 —— 被操作系统强行杀掉
      • 整数除以 O
      • 非法使用特权指令
    • 外界干预 —— 用户选择杀掉进程
      • Ctrl+Alt+delete
      • kill pid

    # 阻塞

    • 找到要阻塞的进程对应的 PCB
    • 保护进程运行现场,将 PCB 状态信息设置为 “阻塞态 ",暂时停止进程运行
    • 将 PCB 插入相应事件的等待队列

    引起进程阻塞的事件可能有:

    • 需要等待系统分配某种资源
    • 需要等待相互合作的其他进程完成工作

    # 唤醒

    • 在事件等待队列中找到 PCB
    • 将 PCB 从等待队列移除,设置进程为就绪态
    • 将 PCB 插入就绪队列,等待被调度

    引起进程唤醒的事件可能有:

    • 等待的事件发生

    # 切换

    • 将运行环境信息存入 PCB
    • PCB 移入相应队列
    • 选择另一个进程执行,并更新其 PCB
    • 根据 PCB 恢复新进程所需的运行环境

    系统切换:

    系统切换

    引起进程切换的事件可能有:

    • 当前进程时间片到
    • 有更高优先级的进程到达
    • 当前进程主动阻塞
    • 当前进程终止

    # 进程通信

    进程间通信(Inter-Process Communication, IPC)是指两个进程之间产生数据交互。

    ⚠️为什么需要进程通信:

    • 进程是分配系统资源的单位(包括内存地址空间),因此各进程拥有的内存地址空间相互独立
    • 为了保证安全,一个进程不能直接访问另一个进程的地址空间
    • 多核处理器系统中,消息传递优于共享内存,因为没有 cache 的一致性问题

    Linux 的 IPC 限制信息可在 /etc/sysctl.conf 中查看

    ipcs 可以查看当前操作系统里有多少消息队列、共享内存块、信号量

    • 低级通信
      • 共享文件,信号量 PV 操作
    • 高级通信 —— 以较高的效率传输大量数据
      • 共享内存
      • 消息传递
      • 管道通信

    # 共享存储

    为避免出错,各个进程对共享空间的访问应该是互斥

    # 基于存储区共享

    操作系统在内存中划出一块共享存储区,数据的形式、存放位置都由通信进程控制,而不是操作系统。这种共享方式速度很快,是一种高级通信方式

    通过 “增加页表项 / 段表项” 即可将同一片共享内存区映射到各个进程的地址空间中

    # Linux 共享内存

    c
    //shm 即 share memory 的缩写
    int shm_open (...);
    // 通过 shm_open 系统调用,申请一片共享内存区
    void * mmap (...);
    // 通过 mmap 系统调用,将共享内存区映射到进程自己的地址空间
    # Key ID
    c
    #include <sys/shm.h>
    key_t ftok(const char *pathname, int id);
    /* key_t is of type int. ftok() convert a pathname and a project identifie
    to an IPC key */
    key_t key = ftok(/home/myshm", 0x27);
    if((key == -1) {
    	perror(ftok()");
    } else
    	printf("key = 0x%x\n", key);
    # Create
    c
    int shmget(key_t key, int size, int shmflg);
    /* shmget() allocates a shared memory segment */
    /* upper bound of size: 1.9G */
    int shmid = shmget(IPC_PRIVATE, 4096, IPC_CREATE|IPC_EXCL|0660);
    if(shmid == -1) {
      perror("shmget()");
    }

    # 基于数据结构的共享

    比如共享空间里只能放一个长度为 10 的数组。这种共享方式速度慢限制多,是一种低级通信方式

    # 消息传递

    进程间的数据交换以格式化的消息(Message)单位。进程通过操作系统提供的两个原语进行数据交换:

    • 发送原语, send(X, msg) ,X 是发送目标
    • 接收消息, receive(X, &msg) ,X 是从哪里接收

    格式化的消息(Message)的结构:

    • 消息头
      • 发送进程 ID
      • 接受进程 ID
      • 消息长度等格式化的信息
    • 消息体

    # 直接通信方式

    P 传递消息给 Q 的过程:

    • P 发送原语 send(Q, msg)
      • msg 数据被封装为格式化消息
      • 格式化消息被从 P 的地址空间转移到操作系统内核的地址空间中进程 Q 的消息队列
    • Q 接收原语 receive(P, &msg)
      • 格式化消息被从操作系统内核的地址空间中进程 Q 的消息队列转移到 Q 的地址空间
      • 解析格式化的消息

    特性:

    • 链接是自动建立的
    • 链路仅与一对通信进程相关联
    • 每对之间只存在一个链接
    • 链路可能是单向的,但通常是双向的

    这种通信方式展示了寻址的对称性,即发送和接收进程必须指定对方,以便通信。

    也可以采用寻址的非对称性,即只要发送者指定接收者,而接收者不需要指定发送者

    # 间接通信方式

    以 “信箱”(或某些特定的端口)作为中间实体进行消息传递,可以多个进程往同一个信箱 send 消息,也可以多个进程从同一个信箱中 receive 消息;

    每个邮箱都有一个唯一的 id ,进程只有在共享邮箱时才能通信;

    允许链接最多与两个进程关联

    一次只允许一个进程执行接收操作

    允许系统任意选择接收器。通知发送方接收者是谁。

    间接通信

    P 传递消息给 Q 的过程:

    • (创建邮箱)
    • P 发送原语 send(A, msg) ,发送给信箱 A
      • msg 数据被封装为格式化消息
      • 格式化消息被从 P 的地址空间转移到操作系统内核的地址空间中信箱 A 中
    • Q 接收原语 receive(A, &msg) ,从信箱 A 接收消息
      • 格式化消息被从操作系统内核的地址空间中信箱 A 转移到 Q 的地址空间
      • 解析格式化的消息
    • (删除邮箱)

    特性:

    • 仅当进程共享公共邮箱时才建立链接
    • 链接可能与许多进程相关联
    • 每对进程可以共享多个通信链路
    • 链路可以是单向的或双向的

    邮箱所有者:

    • 邮箱可以被进程拥有;
    • 邮箱被操作系统拥有;
    • 邮箱可以转移

    # 同步 / 异步

    消息传递可以是阻塞的,也可以是非阻塞的

    • 阻塞被认为是同步的
      • 阻塞发送 — 在接收进程或者邮箱收到消息之前,发送进程阻塞;
      • Blocking receive(阻塞接收)—— 在消息可用之前,接收进程将被阻塞
    • 非阻塞被认为是异步的
      • 非阻塞发送 —— 发送进程发送消息并继续执行;
      • 非阻塞接收 —— 接收进程接收:
        • 有效消息
        • 或空消息
    • 可能有不同的组合
      • 如果发送和接收都是阻塞的,双方之间有一个交会。

    与其他课程的阻塞、同步异步关系可能不太一样

    # 缓存

    附加到链接的消息队列,不管通信是直接还是间接的;以三种方式之一实现队列:

    • 1. 零容量–链路上没有消息排队。发送方必须等待接收方
    • 2. 有限容量–有限长度的 n 条消息,如果链路已满,则发送者必须等待
    • 3. 无限容量–无限长,发送方从不等待

    # Linux 接口实现

    msgdata.h

    c
    #define TEXT_SIZE 512
    /* considering
    ------ Messages Limits --------
    max queues system wide = 32000
    max size of message (bytes) = 8192
    default max size of queue (bytes) = 16384
    ------------------------------------------
    The size of message is set to be 512, the total number of messages is 16384/512 = 32
    If we take the max size 8192, the number would be 16384/8192 = 2. It is not reasonable
    */
    /* message structure */
    struct msg_struct {
    long int msg_type;
    char mtext[TEXT_SIZE]; /* binary data */
    };
    #define PERM S_IRUSR|S_IWUSR|IPC_CREAT
    #define ERR_EXIT(m) \
      do { \
        perror(m); \
        exit(EXIT_FAILURE); \
      } while(0)

    msgsnd.c

    c
    #include <stdio.h>
    #include <stdlib.h>
    #include <string.h>
    #include <unistd.h>
    #include <sys/msg.h>
    #include <sys/stat.h>
    #include <fcntl.h>
    #include "msgdata.h"
    int main(int argc, char *argv[])
    {
      struct msg_struct data;
      long int msg_type;
      char buffer[TEXT_SIZE], pathname[80];
      int msqid, ret, count = 0;
      key_t key;
      FILE *fp;
      struct stat fileattr;
      if(argc < 2) {
        printf("Usage: ./a.out pathname\n");
        return EXIT_FAILURE;
      }
      strcpy(pathname, argv[1]);
      if(stat(pathname, &fileattr) == -1) {
        ret = creat(pathname, O_RDWR);
        if (ret == -1) {
        	ERR_EXIT("creat()");
        }
        printf("shared file object created\n");
      }
      key = ftok(pathname, 0x27); /* project_id can be any nonzero integer */
      if(key < 0) {
      	ERR_EXIT("ftok()");
      }
      printf("\nIPC key = 0x%x\n", key);
      msqid = msgget((key_t)key, 0666 | IPC_CREAT);
      if(msqid == -1) {
      	ERR_EXIT("msgget()");
      }
      fp = fopen("./msgsnd.txt", "rb");
      if(!fp) {
      	ERR_EXIT("source data file: ./msgsnd.txt fopen()");
      }
      struct msqid_ds msqattr;
      ret = msgctl(msqid, IPC_STAT, &msqattr);
      printf("number of messages remainded = %ld, empty slots = %ld\n",
      msqattr.msg_qnum, 16384/TEXT_SIZE-msqattr.msg_qnum);
      printf("Blocking Sending ... \n");
      while (!feof(fp)) {
        ret = fscanf(fp, "%ld %s", &msg_type, buffer);
        if (ret == EOF) break;
        printf("%ld %s\n", msg_type, buffer);
        data.msg_type = msg_type;
        strcpy(data.mtext, buffer);
        ret = msgsnd(msqid, (void *)&data, TEXT_SIZE, 0);
        /* 0: blocking send, waiting when msg queue is full */
        if(ret == -1) {
        	ERR_EXIT("msgsnd()");
        }
        count++;
      }
      printf("number of sent messages = %d\n", count);
      fclose(fp);
      system("ipcs -q");
      exit(EXIT_SUCCESS);
    }

    msgrcv.c

    c
    #include <stdio.h>
    #include <stdlib.h>
    #include <string.h>
    #include <unistd.h>
    #include <sys/msg.h>
    #include <sys/stat.h>
    #include "msgdata.h"
    int main(int argc, char *argv[])
    {
      key_t key;
      struct stat fileattr;
      char pathname[80];
      int msqid, ret, count = 0;
      struct msg_struct data;
      long int msgtype = 0; /* 0 - type of any messages */
      if(argc < 2) {
        printf("Usage: ./msgrcv pathname msg_type\n");
        return EXIT_FAILURE;
      }
      strcpy(pathname, argv[1]);
      if(stat(pathname, &fileattr) == -1) {
      	ERR_EXIT("shared file object stat error");
      }
      if((key = ftok(pathname, 0x27)) < 0) {
      	ERR_EXIT("ftok()");
      }
      printf("\nIPC key = 0x%x\n", key);
      msqid = msgget((key_t)key, 0666); /* do not create a new msg queue */
      if(msqid == -1) {
      	ERR_EXIT("msgget()");
      }
      if(argc < 3)
      	msgtype = 0;
      else {
      	msgtype = atol(argv[2]);
      	if (msgtype < 0)
     			msgtype = 0;
      } /* determin msgtype (class number) */
      printf("Selected message type = %ld\n", msgtype);
      while (1) {
        ret = msgrcv(msqid, (void *)&data, TEXT_SIZE, msgtype, IPC_NOWAIT);
        /* Non_blocking receive */
        if(ret == -1) { /* end of this msgtype */
          printf("number of received messages = %d\n", count);
          break;
        }
        printf("%ld %s\n", data.msg_type, data.mtext);
        count++;
      }
      struct msqid_ds msqattr;
      ret = msgctl(msqid, IPC_STAT, &msqattr);
      printf("number of messages remainding = %ld\n", msqattr.msg_qnum);
      if(msqattr.msg_qnum == 0) {
      	printf("do you want to delete this msg queue?(y/n)");
      if (getchar() == 'y') {
        if(msgctl(msqid, IPC_RMID, 0) == -1)
        	perror("msgctl(IPC_RMID)");
        }
      }
      system("ipcs -q");
      exit(EXIT_SUCCESS);
    }

    # POSIX (pthread) 实现

    c
    #include <mqueue.h>
    # Open, Close and Unlink
    mqd_t mq_open(const char *name, int oflag, mode_t mode, struct mq_attr *attr ); /* return the mqdes, or -1 if failed */
    mqd_t mqID;
    mqID = mq_open("/anonymQueue", O_RDWR | O_CREAT, 0666, NULL);
    mqd_t mq_close(mqd_t mqdes);
    mqd_t mq_unlink(const char *name); /* return -1 if failed */
    # Send and Receive
    mqd_t mq_send(mqd_t mqdes, const char *msg_ptr, size_t msg_len, unsigned msg_prio); /* return 0, or -1 if failed */
    mq_send(mqID, msg, sizeof(msg), i)
    mqd_t mq_receive(mqd_t mqdes, char *msg_ptr, size_t msg_len, unsigned *msg_prio); /* return the number of char received, or -1 if failed */
    mq_attr mqAttr;
    mq_getattr(mqID, &mqAttr);
    mq_receive(mqID, buf, mqAttr.mq_msgsize, NULL)

    Note that the Linux ipcs utility is not fully compatible to the POSIX ipcs utility. The message queues, shared memory and semaphores in POSIX can not be sensed by System V bash command such as ipcs -q

    # Mach

    Mach 作为微内核结构,其通信是基于消息传递的设计,甚至系统调用也是消息传递

    • 每个任务在创建时获得两个端口 - 内核和通知
    • 使用 mach_msg() 函数发送和接收消息
    • 通信所需的端口,通过创建 mach 端口分配
    • 收发灵活;例如,邮箱已满时有四个选项:
      • 无限期地等待
      • 最多等待 n 毫秒
      • 立即返回
      • 临时缓存消息

    # Windows

    通过高级本地过程调用(ALPC)设施以消息传递为中心,仅在同一系统上的两个进程之间工作,类似 RPC;

    • 使用端口(如邮箱)建立和维护通信通道;
    • 使用两种类型的端口:
      • 连接端口
      • 通信端口

    通讯工作如下:

    • 客户端打开子系统连接端口对象的句柄;
    • 客户端发送一个连接请求;
    • 服务器创建一对专用通信端口,并将其中一个端口的句柄返回给客户端;
    • 客户端和服务器使用相应的端口句柄发送消息或回调,并监听回复;

    windows消息传递

    Windows 对不同大小的消息有不同的策略;When an ALPC channel is created, one of three message-passing techniques is chosen:

    1. For small messages (up to 256 bytes), the port’s message queue is used as intermediate storage, and the messages are copied from one process to the other.

    2. Larger messages must be passed through a section object, which is a region of shared memory associated with the channel.

    3. When the amount of data is too large to fit into a section object, an API is available that allows server processes to read and write directly into the address space of a client.

    # 管道通信

    “管道” 是一个特殊的共享文件,又名 pipe 文件。其实就是在内存中开辟一个大小固定的内存缓冲区

    • 普通管道 (匿名管道)—— 无法从创建它的进程外部访问。通常,父进程创建管道并使用它与它创建的子进程通信。是单向的、半双工的。
      普通管道
    • 命名管道 —— 可以在没有父子关系的情况下访问。是双向的全双工的。

    管道通信的规则、限制:

    • 管道只能采用半双工通信,某一时间段内只能实现单向的传输。
    • 如果要实现双向同时通信,则需要设置两个管道。
    • 各进程要互斥地访问管道(由操作系统实现)
    • 当管道写满时,写进程阻塞,直到读进程将管道中的数据取走,即可唤醒写进程。
    • 当管道读空时,读进程阻塞,直到写进程往管道中写入数据,即可唤醒读进程。
    • 管道中的数据一旦被读出,就彻底消失。因此,当多个进程读同一个管道时,可能会错乱。对此,通常有两种解决方案:
      • 一个管道允许多个写进程一个读进程(2014 年 408 真题高教社官方答案);
      • 允许有多个写进程多个读进程,但系统会让各个读进程轮流从管道中读数据(Linux 的方案)。

    管道通信

    写进程往管道写数据,即便管道没被写满,只要管道没空,读进程就可以从管道读数据

    读进程从管道读数据,即便管道没被读空,只要管道没满,写进程就可以往管道写数据

    # Linux 实现

    c
    #include <sys/types.h>
    #include <stdio.h>
    #include <string.h>
    #include <unistd.h>
    #define BUFFER SIZE 25
    #define READ END 0
    #define WRITE END 1
    int main(void)
    {
      char write msg[BUFFER SIZE] = "Greetings";
      char read msg[BUFFER SIZE];
      int fd[2];
      pid t pid;
      /* Program continues in Figure 3.22 */
      /* create the pipe */
      if (pipe(fd) == -1) {
        fprintf(stderr,"Pipe failed");
        return 1;
      }
      /* fork a child process */
      pid = fork();
      if (pid < 0) { /* error occurred */
        fprintf(stderr, "Fork Failed");
        return 1;
      }
      if (pid > 0) { /* parent process */
        /* close the unused end of the pipe */
        close(fd[READ END]);
        /* write to the pipe */
        write(fd[WRITE END], write msg, strlen(write msg)+1);
        /* close the write end of the pipe */
        close(fd[WRITE END]);
      }
      else { /* child process */
        /* close the unused end of the pipe */
        close(fd[WRITE END]);
        /* read from the pipe */
        read(fd[READ END], read msg, BUFFER SIZE);
        printf("read %s",read msg);
        /* close the read end of the pipe */
        close(fd[READ END]);
      }
      return 0;
    }

    利用 libc 创建、操作管道:

    c
    // 普通管道
    int pipefd[2];
    int pipe(int pipefd);
    int fcntl(int pipefd[0|1], int cmd);
    int fcntl(int pipefd[0|1], int cmd, long arg);
    ssize_t write(int pipefd[1], void* buf, size_t count);
    ssize_t read(int pipefd[0], void* buf, size_t count);
    close(pipefd[0]);
    close(pipefd[1]);
    // 命名管道
    #define FIFO pathname /* pathname: "/tmp/my_fifo" */
    unlink(FIFO); /*delete a name and possibly the file it refers to */
    mkfifo(FIFO, 0666);
    int fdw = open(FIFO, O_RDWR);
    mkfifo(FIFO, 0444);
    int fdr = open(FIFO, O_RDONLY);
    ssize_t write(int fdw, void* buf, size_t count);
    /* ssizt_t = signed int, sizt_t = unsigned int */
    ssize_t read(int fdr, void* buf, size_t count);
    close(fdw);
    close(fdr);

    管道缓冲

    c
    #include <stdio.h>
    #include <stdlib.h>
    #include <unistd.h>
    #include <fcntl.h>
    #define ERR_EXIT(m) \
      do { \
        perror(m); \
        exit(EXIT_FAILURE); \
      } while (0)
    int main(int argc, char *argv[])
    {
      int pipefd[2];
      int bufsize;
      char *buffer;
      int flags, ret, lastwritten, count, totalwritten;
      if(pipe(pipefd) == -1) /* create an ordinary pipe */
      	ERR_EXIT("pipe()");
      flags = fcntl(pipefd[1], F_GETFL);
      fcntl(pipefd[1], F_SETFL, flags | O_NONBLOCK); /* set write_end NONBLOCK */
      bufsize = atoi(argv[1]);
      printf("testing buffer size = %d\n", bufsize);
      buffer = (char *)malloc(bufsize*sizeof(char));
      if(buffer == NULL || bufsize == 0)
      	ERR_EXIT("malloc()");
      count = 0;
      while (1) {
        ret = write(pipefd[1], buffer, bufsize);
        /* bufsize is better to be 2^k */
        if(ret == -1) {
          perror("write()");
          break;
        }
        lastwritten = ret;
        count++;
      }
      totalwritten = (count-1)*bufsize +
      lastwritten;
      printf("single pipe buffer count = %d, last written = %d bytes\n", count, lastwritten);
      printf("total written = %d bytes = %d KiB\n", totalwritten, totalwritten/1024); /* pipe buffer */
      return 0;
    }

    # Pipes in UNIX CLI

    A pipe can be constructed on the UNIX command line using the | character. The complete command is ls | less

    The commands ls and less are running as individual processes. The output of ls is delivered as the input to less.

    这里的 shell 和 ls 之间、ls 和 less 之间都是普通管道通信

    # Windows 实现

    c
    #include <stdio.h>
    #include <stdlib.h>
    #include <windows.h>
    #define BUFFER SIZE 25
    int main(VOID)
    {
      HANDLE ReadHandle, WriteHandle;
      STARTUPINFO si;
      PROCESS INFORMATION pi;
      char message[BUFFER SIZE] = "Greetings";
      DWORD written;
      /* Program continues in Figure 3.24 */
      /* set up security attributes allowing pipes to be inherited */
      SECURITY ATTRIBUTES sa = {sizeof(SECURITY ATTRIBUTES),NULL,TRUE};
      /* allocate memory */
      ZeroMemory(&pi, sizeof(pi));
      /* create the pipe */
      if (!CreatePipe(&ReadHandle, &WriteHandle, &sa, 0)) {
        fprintf(stderr, "Create Pipe Failed");
        return 1;
      }
      /* establish the START INFO structure for the child process */
      GetStartupInfo(&si);
      si.hStdOutput = GetStdHandle(STD OUTPUT HANDLE);
      /* redirect standard input to the read end of the pipe */
      si.hStdInput = ReadHandle;
      si.dwFlags = STARTF USESTDHANDLES;
      /* don’t allow the child to inherit the write end of pipe */
      SetHandleInformation(WriteHandle, HANDLE FLAG INHERIT, 0);
      /* create the child process */
      CreateProcess(NULL, "child.exe", NULL, NULL,
                    TRUE, /* inherit handles */
                    0, NULL, NULL, &si, &pi);
      /* close the unused end of the pipe */
      CloseHandle(ReadHandle);
      /* the parent writes to the pipe */
      if (!WriteFile(WriteHandle, message,BUFFER SIZE,&written,NULL))
      	fprintf(stderr, "Error writing to pipe.");
      /* close the write end of the pipe */
      CloseHandle(WriteHandle);
      /* wait for the child to exit */
      WaitForSingleObject(pi.hProcess, INFINITE);
      CloseHandle(pi.hProcess);
      CloseHandle(pi.hThread);
      return 0;
    }

    # 客户机 - 服务器系统中的通信

    • 套接字(Socket)
    • 远程过程调用(RPC)

    # 套间字 (Socket)

    套接字被定义为通信的端点,是很多协议的底层实现方式,通信由一对套接字组成

    IP 地址和端口的串联–包含在消息包开头的数字,用于区分主机上的网络服务;例如:套接字 161.25.19.8:1625 引用主机 161.25.19.8 上的端口 1625 ;

    • 1024 以下的所有端口都是相对固定的,用于标准服务;
    • 特殊 IP 地址 127.0.0.1 (环回),用于表示正在运行进程的系统;

    三种类型的套接字

    • 面向连接(TCP)
    • 无连接(UDP)
    • Multicast Socket 类:数据可以发送到多个收件人

    # 远程过程调用 (RPC)

    远程过程调用(RPC)是抽象网络系统上进程之间的过程调用,也使用端口进行服务区分

    Stubs(桩函数)—— 服务器上实际过程的客户端代理,提供封装好的调用接口:

    • 客户端存根定位服务器并封送参数
    • 服务器端存根接收此消息,解压缩封送的参数,并在服务器上执行该过程

    通过外部数据表示(XDL)格式处理数据表示,以考虑不同的体系结构,例如大端和小端;

    远程通信比本地通信有更多的故障场景,消息可以只传递一次,而不是最多一次

    操作系统通常提供会合(或配对)服务来连接客户端和服务器

    在 Windows 上,存根代码根据用 Microsoft 接口定义语言(MIDL)编写的规范编译

    远程过程调用

    # DCE RPC

    参考:Distributed Computing Environment ,OSF(开放软件基金会)下的一个软件系统框架(慢慢消亡)

    DCE 的构成:

    • 远程过程调用 DCE/RPC;
    • 命名 / 目录服务;
    • 时间服务;
    • 安全服务;
    • 分布式文件系统 DCE/DFS;

    目前 Linux 标配 RPC

    # 进程调度

    Process scheduler 从可用进程中选择下一个在 CPU 核心上执行的进程

    目标 —— 最大限度地利用 CPU,快速将进程切换到 CPU 核心

    维护进程的调度队列

    • 就绪队列 (ready queue)—— 驻留在主内存中、就绪并等待执行的所有进程的集合
    • 等待队列 (wait queue)—— 等待事件(即 I/O)的一组进程
      • I/O wait queue
      • child termination wait queue
      • interrupt wait queue
    • 进程在各种队列之间迁移
    Edited on Views times
    \ No newline at end of file +进程 - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 进程的基本概念

    # 进程的定义

    典型的定义:

    • 进程是进程实体的运行过程,是系统进行资源分配和调度的一个独立单位 ==
    • 一个具有独立功能的程序在一个数据集合上运行的过程

    # 区分进程程序进程实体

    • 程序(Program):是静态的,就是个存放在磁盘里的可执行文件,就是一系列的指令集合

      • 静态连接
      • 动态链接
    • 进程(Process):是动态的,是程序的一次执行过程

    • 进程实体(进程映像):是静态的,反应了进程在某一时刻的状态

    # 进程实体的组成

    一个 == 进程实体(进程映像)== 由三个部分组成:

    • PCB
      • 进程描述信息
      • 进程控制和管理信息
      • 资源分配清单
      • 处理机状态信息(CPU 上下文
    • 程序段
      • 程序的代码(指令序列)
    • 数据段
      • 运行过程中产生的各种数据(如:程序中定义的变量)

    PCB 是给操作系统用的;程序段、数据段是给进程自己用的,与进程自身的运行逻辑有关

    # PCB

    PCB 位于内存的内核空间里,创建之后常驻内存。

    PCB 是进程存在的唯一标志,当进程被创建时,操作系统为其创建 PCB,当进程结束时,会会回收其 PCB。

    PCB 几乎完备的表示了进程的状态,也就是说只要有一个进程的 PCB,就可以恢复对应的进程

    # PCB 的结构

    • 进程描述信息

      • 进程标识符 PID
      • 用户标识符 UID
    • 进程控制和管理信息

      • CPU、磁盘、网络流量使用情况统计⋯
        • 进入内存时间
        • CPU 占用时间
        • 信号量使用
      • 进程当前状态:就绪态 / 阻塞态 / 运行态
      • 代码运行入口地址
      • 程序的外存地址
      • 程序优先级
    • 资源分配清单

      • 正在使用(依赖)哪些文件
        • 文件描述符
      • 正在使用哪些内存区域
        • 代码段指针
        • 数据段指针
        • 堆栈段指针
      • 正在使用哪些 I/O 设备
    • 处理机相关信息,也叫 CPU 的上下文(the context of the processor)

      • user-visible registers
      • control and status registers (CSR)
      • stack pointers
      • PSW
        • eg: EFLAGS in x86

    # Linux 的 PCB

    Linux 中的进程控制块 PCB 源码位于 linux-x.xx.xx/include/linux/sched.h

    Linux 不区分进程和线程,将它们都视为 task

    由 C 结构 task_struct 表示:

    c
    {
      pid t_pid;/* 进程标识符 */
      unsigned int time_slice /* 调度信息 */
      struct task_struct* 父进程;/* 此进程的父级 */
      struct list_head children;/* 此进程的子进程 */
      struct *files;/* 打开文件列表 */
      struct mm_struct*mm;/* 此进程的地址空间 */
      ...
    }

    task_struct

    # 进程的特征

    • 动态性 —— 进程是程序的一次执行过程,是动态地产生、变化和消亡的(动态性是进程最基本的特征)
    • 并发性 —— 内存中有多个进程实体,各进程可并发执行
    • 独立性 —— 进程是能独立运行、独立获得资源、独立接受调度的基本单位
    • 异步性 —— 各进程按各自独立的、不可预知的速度向前推进,操作系统要提供 “进程同步机制 " 来解决异步问题
    • 结构性 —— 每个进程都会配置一个 PCB。结构上看,进程由程序段、数据段、PCB 组成

    # 进程的状态与转换

    The steps in a full process switch are:

    1. save the context of the processor
    2. update the PCB of the process currently in the Running state
    3. move the PCB of this process to the appropriate queue
    4. select another process for execution
    5. update the PCB of the process selected
    6. update memory management data structures
    7. restore the context of the processor to that which existed at the time the selected process was last switched out

    # 五状态模型

    五状态模型

    # 七状态模型

    增加了挂起状态的进程


    挂起进程的特征:

    • 进程不能立即执行(上处理机)

    • 进程可能有也可能么有等待一个事件(为什么那么像废话)

    • 为阻止该进程执行,可通过代理使其处于挂起状态;代理可以是:

      • 进程本身

      • 父进程

      • 操作系统

    • 只有代理程序显示地命令操作系统进行状态轮换,才能使该进程从挂起的状态转移


    进程挂起的原因:

    • Swaping
    • Other OS reason
    • Interactive user request
    • Timing
    • Parent pocess request

    双挂起状态的进程模型:

    七状态模型

    # 进程组织

    进程组织的数据结构

    # 链接式组织进程

    链式组织进程

    # 索引式组织进程

    索引式组织进程

    # 进程控制

    也就是对进程的操作,这些基本操作几乎全部都是系统调用实现的

    # 系统调用

    Some basic and important system calls

    PCB 的访问:

    getpid()

    • fork()
    • exec*()
    • wait()
    • exit()

    # Linux 举例

    父进程创建子进程,子进程又创建其他进程,形成进程树;父子之间不同角度可能有不同关系:

    • 资源共享
      • 父和子共享所有资源 (Linux 中父进程 vfork )
      • 子进程共享父进程资源的子集 (Linux 中父进程 vfork )
      • 父进程和子进程不共享任何资源 (Linux 中父进程 fork )
    • 执行
      • 父进程和子进程同时并发执行
      • 父进程等待直到子进程终止 (Linux 中父进程 wait(0) )
    • 地址空间
      • 子进程是父进程的复制品 (Linux 中父进程 fork )
      • 子进程加载另一个新的程序
      • 子进程完全覆盖父进程 (Linux 中父进程 fork + exec )
        • exec family 有 6 个成员
        • execl 装载新的文件

    # fork

    Linux 中,由父进程调用系统调用 fork() 创建子进程

    c
    #include<stdio.h>
    #include<unistd.h>
    int main (){
      for(int i=0;i<4;++i){
        fork();
      }
    	printf("This is a test printed by %d\n", getpid());
    return 0;
    }

    编译运行上述程序将打印出 16 次!因为这里子进程是父进程的复制品:

    shell
    node1@vm1:~/Desktop$ gcc -o test test.c
    node1@vm1:~/Desktop$ ./test
    This is a test printed by 139377
    This is a test printed by 139382
    This is a test printed by 139380
    This is a test printed by 139379
    This is a test printed by 139378
    This is a test printed by 139383
    node1@vm1:~/Desktop$ This is a test printed by 139381 # 父进程结束
    This is a test printed by 139386
    This is a test printed by 139387
    This is a test printed by 139385
    This is a test printed by 139388
    This is a test printed by 139389
    This is a test printed by 139384
    This is a test printed by 139390
    This is a test printed by 139391
    This is a test printed by 139392

    fork 会克隆的内容:

    Cloned itemsDescriptions
    Program code [File & Memory]They are sharing the same piece of code.
    MemoryIncluding local variables, global variables, and dynamically allocated memory.
    Opened files [Kernel's internal]If the parent has opened a file "A", then the child will also have file "A" opened automatically.
    Program counter [CPU register]That's why they both execute from the same line of code after fork() returns.

    fork 不克隆的内容:

    Distinct itemsParentChild
    Return value of fork() PID of the child process.0
    PIDUnchanged.Different, not necessarily be "Parent PID + 1"
    Parent processUnchanged.Doesn't have the same parent as that of the parent process.
    Running timeCumulated.Just created, so should be 0.

    vfork 函数是 fork 函数基础上的修正, fork 函数创建一个完全分离的进程,而 vfork 用于创建一个共享的进程,没有克隆操作,这意味着父进程子进程可以互相看到对方,共享同样的物理地址

    # exec*

    exec* 系统调用,会 “鸠占鹊巢”,替换掉原进程的代码,但是 PID、父子关系、进程运行时间都不变,也就是:

    • 内核里的内容不变,包括 PCB 等
    • 用户态里的代码被替换
    c
    #include<stdio.h>
    #include<unistd.h>
    int main(void) {
    	printf("before execl ...\n"); // 原进程代码可以执行
    	execl("/bin/ls", "/bin/ls", NULL); //exec 的一个成员函数
    	printf("after execl ...\n"); // 原进程代码无法执行
    	return 0;
    }

    编译执行结果:

    shell
    node1@vm2:~/Desktop$ gcc -o test test.c
    node1@vm2:~/Desktop$ ./test
    before execl ...
    sdb  stack_rwx.c  test  test01  test.c  ysos

    # waitwaitpid

    wait()waitpid()
    Wait for any one of the children.Depending on the parameters, waitpid() will wait for a particular child only.
    Detect child termination only.Depending on the parameters, waitpid() can detect child's status changing: -from running to suspended, and -from suspended to running.

    # fork + exec*

    执行顺序是不确定的

    c
    #include<stdio.h>
    #include<unistd.h>
    int system_test(const char *cmd_str) {
        if(cmd_str == -1)
            return -1;
        if(fork() == 0) {
            execl(cmd_str, cmd_str, NULL);
            fprintf(stderr,"%s: command not found\n", cmd_str);
            exit(-1);
        }
        return 0;
    }
    int main(void) {
        printf("before...\n\n");
        system_test("/bin/ls");
        printf("\nafter...\n");
        return 0;
    }

    编译执行结果:

    shell
    node1@vm2:~/Desktop$ gcc -o test test.c
    node1@vm2:~/Desktop$ ./test
    before...
    after...
    sdb  stack_rwx.c  test  test01  test.c  ysos # 执行 ls

    类似于一个 shell 或 system ,不过 system 创建子进程,时串行执行,执行顺序是确定的。

    # fork + exec* + wait

    等价于 system

    c
    #include<stdio.h>
    #include<unistd.h>
    int system_test(const char *cmd_str) {
        if (cmd_str == -1)
            return -1;
        if(fork() == 0) {
            execl("/bin/sh", "/bin/sh","-c", cmd_str, NULL);
            fprintf(stderr,"%s: command not found\n", cmd_str);
            exit(-1);
        }
        wait(NULL);
        return 0;
    }
    int main(void) {
        printf ("before...\n\n"); system_test("/bin/1s");
        printf("\nafter...\n");
        return 0;
    }

    编译执行结果:

    shell
    node1@vm2:~/Desktop$ gcc -o test test.c
    node1@vm2:~/Desktop$ ./test
    before...
    sdb  stack_rwx.c  test  test01  test.c  ysos
    after...

    # exit

    父进程 wait 和子进程 exit 搭配食用。子进程执行最后一条语句,然后调用 exit 系统调用请求操作系统删除它自己,然后子进程将发送 SIGCHLD 给父进程,并变僵尸进程;父进程接收到 SIGCHLD 后被唤醒(原本被 wait 休眠),处理 SIGCHLD 之后,子进程才被彻底销毁。

    c
    #include<stdio.h>
    #include<unistd.h>
    #include<stdlib.h>
    int main(void){
        int count = 1;
        pid_t childpid, terminatedid;
        childpid = fork(); /* child duplicates parent’s address space */
        if (childpid < 0) {
            perror("fork()");
            return EXIT_FAILURE;
        }
        else if (childpid == 0) { /* This is child pro */
            count++;
            printf("child pro pid = %d, count = %d (addr = %p)\n", getpid(), count, &count);
            printf("child sleeping ...\n");
            sleep(5); /* parent wait() during this period */
            printf("\nchild waking up!\n");
        }
        else { /* This is parent pro */
            terminatedid = wait(0);
            printf("parent pro pid = %d, terminated pid = %d, count = %d (addr = %p)\n",
            getpid(), terminatedid, count, &count);
        }
        printf("\nTesting point by %d\n", getpid()); /* executed by child and parent */
        return EXIT_SUCCESS;
    }

    编译运行结果:

    shell
    node1@vm2:~/Desktop$ ./test
    child pro pid = 197382, count = 2 (addr = 0x7ffc33c444cc)
    child sleeping ...
    # 这里等待了 5 秒钟
    child waking up!
    Testing point by 197382
    parent pro pid = 197381, terminated pid = 197382, count = 1 (addr = 0x7ffc33c444cc)
    Testing point by 197381

    僵尸进程:空有 PCB “外壳”,但是没有用户态的代码,无法执行;

    注意区分孤儿进程

    # abort

    父进程可以使用 abort 系统调用终止子进程的执行。这样做的一些原因:

    • 子进程已超出分配的资源
    • 不再需要分配给子进程的任务
    • 父进程正在退出,如果父进程终止,操作系统不允许子进程继续

    # 进程控制原语

    进程控制由原语实现

    如果不能像原语 “一气呵成”,就有可能导致操作系统中的某些关键数据结构信息不统一的情况,这会影响操作系统进行别的管理工作

    原语的执行具有原子性,即执行过程只能一气呵成,期间不允许被中断。

    可以用 “关中断指令”“开中断指令” 这两个特权指令实现原子性

    无论哪个进程控制原语,要做的无非三类事情:

    • 更新 PCB 中的信息
      • 修改进程状态(state)
      • 保存 / 恢复运行环境
    • 将 PCB 插入合适的队列
    • 分配 / 回收资源

    # 创建

    • 申请空白 PCB
    • 为新进程分配所需资源
    • 初始化 PCB
    • 将 PCB 插入就绪队列(进程由阻塞态转变为就绪态

    引起进程创建的事件可能有:

    • 用户登录 —— 分时系统中,用户登录成功,系统会建立为其建立一个新的进程
    • 作业调度 —— 多道批处理系统中,有新的作业放入内存时,会为其建立一个新的进程
    • 提供服务 —— 用户向操作系统提出某些请求时,会新建一个进程处理该请求
    • 应用请求 —— 由用户进程主动请求创建一个子进程

    所有进程都是以父进程创建子进程的方式创建的:

    • 有些是显式的
      • 用户进程创建的
    • 有些是隐式的
      • 操作系统 “0 号” 根进程创建的(所有进程的祖先)

    # 根进程

    The first process -- The kernel, while it is booting up, creates the firstprocess -- init .

    The init process:

    • has PID = 1
    • and is running the program code “/sbin/init”.
    • Its first task is to create more processes…

    # 孤儿进程

    父进程创建子进程一层一层创建,但若其中某一个父进程 terminate 了,也就是被终止了,那么被终止进程所创建的进程就变成了孤儿进程。孤儿进程会消耗资源,需要被重新管理起来。

    • Linux 中,使用 re-parent operation 处理孤儿进程, The “init” process will become the step-mother of all orphans.
    • Windows maintains a forest-like hierarchy.

    # 撤销

    这一过程进程: 就绪态 / 阻塞态 / 运行态 -> 终止态 -> 无

    • 从 PCB 集合中找到终止进程的 PCB
    • 若进程正在运行,立即剥夺 CPU,将 CPU 分配给其他进程
    • 终止其所有子进程
    • 将该进程拥有的所有资源归还给父进程或操作系统
    • 删除 PCB

    引起进程终止的事件可能有:

    • 正常结束 —— 进程自己请求终止(exit 系统调用)
    • 异常结束 —— 被操作系统强行杀掉
      • 整数除以 O
      • 非法使用特权指令
    • 外界干预 —— 用户选择杀掉进程
      • Ctrl+Alt+delete
      • kill pid

    # 阻塞

    • 找到要阻塞的进程对应的 PCB
    • 保护进程运行现场,将 PCB 状态信息设置为 “阻塞态 ",暂时停止进程运行
    • 将 PCB 插入相应事件的等待队列

    引起进程阻塞的事件可能有:

    • 需要等待系统分配某种资源
    • 需要等待相互合作的其他进程完成工作

    # 唤醒

    • 在事件等待队列中找到 PCB
    • 将 PCB 从等待队列移除,设置进程为就绪态
    • 将 PCB 插入就绪队列,等待被调度

    引起进程唤醒的事件可能有:

    • 等待的事件发生

    # 切换

    • 将运行环境信息存入 PCB
    • PCB 移入相应队列
    • 选择另一个进程执行,并更新其 PCB
    • 根据 PCB 恢复新进程所需的运行环境

    系统切换:

    系统切换

    引起进程切换的事件可能有:

    • 当前进程时间片到
    • 有更高优先级的进程到达
    • 当前进程主动阻塞
    • 当前进程终止

    # 进程通信

    进程间通信(Inter-Process Communication, IPC)是指两个进程之间产生数据交互。

    ⚠️为什么需要进程通信:

    • 进程是分配系统资源的单位(包括内存地址空间),因此各进程拥有的内存地址空间相互独立
    • 为了保证安全,一个进程不能直接访问另一个进程的地址空间
    • 多核处理器系统中,消息传递优于共享内存,因为没有 cache 的一致性问题

    Linux 的 IPC 限制信息可在 /etc/sysctl.conf 中查看

    ipcs 可以查看当前操作系统里有多少消息队列、共享内存块、信号量

    • 低级通信
      • 共享文件,信号量 PV 操作
    • 高级通信 —— 以较高的效率传输大量数据
      • 共享内存
      • 消息传递
      • 管道通信

    # 共享存储

    为避免出错,各个进程对共享空间的访问应该是互斥

    # 基于存储区共享

    操作系统在内存中划出一块共享存储区,数据的形式、存放位置都由通信进程控制,而不是操作系统。这种共享方式速度很快,是一种高级通信方式

    通过 “增加页表项 / 段表项” 即可将同一片共享内存区映射到各个进程的地址空间中

    # Linux 共享内存

    c
    //shm 即 share memory 的缩写
    int shm_open (...);
    // 通过 shm_open 系统调用,申请一片共享内存区
    void * mmap (...);
    // 通过 mmap 系统调用,将共享内存区映射到进程自己的地址空间
    # Key ID
    c
    #include <sys/shm.h>
    key_t ftok(const char *pathname, int id);
    /* key_t is of type int. ftok() convert a pathname and a project identifie
    to an IPC key */
    key_t key = ftok(/home/myshm", 0x27);
    if((key == -1) {
    	perror(ftok()");
    } else
    	printf("key = 0x%x\n", key);
    # Create
    c
    int shmget(key_t key, int size, int shmflg);
    /* shmget() allocates a shared memory segment */
    /* upper bound of size: 1.9G */
    int shmid = shmget(IPC_PRIVATE, 4096, IPC_CREATE|IPC_EXCL|0660);
    if(shmid == -1) {
      perror("shmget()");
    }

    # 基于数据结构的共享

    比如共享空间里只能放一个长度为 10 的数组。这种共享方式速度慢限制多,是一种低级通信方式

    # 消息传递

    进程间的数据交换以格式化的消息(Message)单位。进程通过操作系统提供的两个原语进行数据交换:

    • 发送原语, send(X, msg) ,X 是发送目标
    • 接收消息, receive(X, &msg) ,X 是从哪里接收

    格式化的消息(Message)的结构:

    • 消息头
      • 发送进程 ID
      • 接受进程 ID
      • 消息长度等格式化的信息
    • 消息体

    # 直接通信方式

    P 传递消息给 Q 的过程:

    • P 发送原语 send(Q, msg)
      • msg 数据被封装为格式化消息
      • 格式化消息被从 P 的地址空间转移到操作系统内核的地址空间中进程 Q 的消息队列
    • Q 接收原语 receive(P, &msg)
      • 格式化消息被从操作系统内核的地址空间中进程 Q 的消息队列转移到 Q 的地址空间
      • 解析格式化的消息

    特性:

    • 链接是自动建立的
    • 链路仅与一对通信进程相关联
    • 每对之间只存在一个链接
    • 链路可能是单向的,但通常是双向的

    这种通信方式展示了寻址的对称性,即发送和接收进程必须指定对方,以便通信。

    也可以采用寻址的非对称性,即只要发送者指定接收者,而接收者不需要指定发送者

    # 间接通信方式

    以 “信箱”(或某些特定的端口)作为中间实体进行消息传递,可以多个进程往同一个信箱 send 消息,也可以多个进程从同一个信箱中 receive 消息;

    每个邮箱都有一个唯一的 id ,进程只有在共享邮箱时才能通信;

    允许链接最多与两个进程关联

    一次只允许一个进程执行接收操作

    允许系统任意选择接收器。通知发送方接收者是谁。

    间接通信

    P 传递消息给 Q 的过程:

    • (创建邮箱)
    • P 发送原语 send(A, msg) ,发送给信箱 A
      • msg 数据被封装为格式化消息
      • 格式化消息被从 P 的地址空间转移到操作系统内核的地址空间中信箱 A 中
    • Q 接收原语 receive(A, &msg) ,从信箱 A 接收消息
      • 格式化消息被从操作系统内核的地址空间中信箱 A 转移到 Q 的地址空间
      • 解析格式化的消息
    • (删除邮箱)

    特性:

    • 仅当进程共享公共邮箱时才建立链接
    • 链接可能与许多进程相关联
    • 每对进程可以共享多个通信链路
    • 链路可以是单向的或双向的

    邮箱所有者:

    • 邮箱可以被进程拥有;
    • 邮箱被操作系统拥有;
    • 邮箱可以转移

    # 同步 / 异步

    消息传递可以是阻塞的,也可以是非阻塞的

    • 阻塞被认为是同步的
      • 阻塞发送 — 在接收进程或者邮箱收到消息之前,发送进程阻塞;
      • Blocking receive(阻塞接收)—— 在消息可用之前,接收进程将被阻塞
    • 非阻塞被认为是异步的
      • 非阻塞发送 —— 发送进程发送消息并继续执行;
      • 非阻塞接收 —— 接收进程接收:
        • 有效消息
        • 或空消息
    • 可能有不同的组合
      • 如果发送和接收都是阻塞的,双方之间有一个交会。

    与其他课程的阻塞、同步异步关系可能不太一样

    # 缓存

    附加到链接的消息队列,不管通信是直接还是间接的;以三种方式之一实现队列:

    • 1. 零容量–链路上没有消息排队。发送方必须等待接收方
    • 2. 有限容量–有限长度的 n 条消息,如果链路已满,则发送者必须等待
    • 3. 无限容量–无限长,发送方从不等待

    # Linux 接口实现

    msgdata.h

    c
    #define TEXT_SIZE 512
    /* considering
    ------ Messages Limits --------
    max queues system wide = 32000
    max size of message (bytes) = 8192
    default max size of queue (bytes) = 16384
    ------------------------------------------
    The size of message is set to be 512, the total number of messages is 16384/512 = 32
    If we take the max size 8192, the number would be 16384/8192 = 2. It is not reasonable
    */
    /* message structure */
    struct msg_struct {
    long int msg_type;
    char mtext[TEXT_SIZE]; /* binary data */
    };
    #define PERM S_IRUSR|S_IWUSR|IPC_CREAT
    #define ERR_EXIT(m) \
      do { \
        perror(m); \
        exit(EXIT_FAILURE); \
      } while(0)

    msgsnd.c

    c
    #include <stdio.h>
    #include <stdlib.h>
    #include <string.h>
    #include <unistd.h>
    #include <sys/msg.h>
    #include <sys/stat.h>
    #include <fcntl.h>
    #include "msgdata.h"
    int main(int argc, char *argv[])
    {
      struct msg_struct data;
      long int msg_type;
      char buffer[TEXT_SIZE], pathname[80];
      int msqid, ret, count = 0;
      key_t key;
      FILE *fp;
      struct stat fileattr;
      if(argc < 2) {
        printf("Usage: ./a.out pathname\n");
        return EXIT_FAILURE;
      }
      strcpy(pathname, argv[1]);
      if(stat(pathname, &fileattr) == -1) {
        ret = creat(pathname, O_RDWR);
        if (ret == -1) {
        	ERR_EXIT("creat()");
        }
        printf("shared file object created\n");
      }
      key = ftok(pathname, 0x27); /* project_id can be any nonzero integer */
      if(key < 0) {
      	ERR_EXIT("ftok()");
      }
      printf("\nIPC key = 0x%x\n", key);
      msqid = msgget((key_t)key, 0666 | IPC_CREAT);
      if(msqid == -1) {
      	ERR_EXIT("msgget()");
      }
      fp = fopen("./msgsnd.txt", "rb");
      if(!fp) {
      	ERR_EXIT("source data file: ./msgsnd.txt fopen()");
      }
      struct msqid_ds msqattr;
      ret = msgctl(msqid, IPC_STAT, &msqattr);
      printf("number of messages remainded = %ld, empty slots = %ld\n",
      msqattr.msg_qnum, 16384/TEXT_SIZE-msqattr.msg_qnum);
      printf("Blocking Sending ... \n");
      while (!feof(fp)) {
        ret = fscanf(fp, "%ld %s", &msg_type, buffer);
        if (ret == EOF) break;
        printf("%ld %s\n", msg_type, buffer);
        data.msg_type = msg_type;
        strcpy(data.mtext, buffer);
        ret = msgsnd(msqid, (void *)&data, TEXT_SIZE, 0);
        /* 0: blocking send, waiting when msg queue is full */
        if(ret == -1) {
        	ERR_EXIT("msgsnd()");
        }
        count++;
      }
      printf("number of sent messages = %d\n", count);
      fclose(fp);
      system("ipcs -q");
      exit(EXIT_SUCCESS);
    }

    msgrcv.c

    c
    #include <stdio.h>
    #include <stdlib.h>
    #include <string.h>
    #include <unistd.h>
    #include <sys/msg.h>
    #include <sys/stat.h>
    #include "msgdata.h"
    int main(int argc, char *argv[])
    {
      key_t key;
      struct stat fileattr;
      char pathname[80];
      int msqid, ret, count = 0;
      struct msg_struct data;
      long int msgtype = 0; /* 0 - type of any messages */
      if(argc < 2) {
        printf("Usage: ./msgrcv pathname msg_type\n");
        return EXIT_FAILURE;
      }
      strcpy(pathname, argv[1]);
      if(stat(pathname, &fileattr) == -1) {
      	ERR_EXIT("shared file object stat error");
      }
      if((key = ftok(pathname, 0x27)) < 0) {
      	ERR_EXIT("ftok()");
      }
      printf("\nIPC key = 0x%x\n", key);
      msqid = msgget((key_t)key, 0666); /* do not create a new msg queue */
      if(msqid == -1) {
      	ERR_EXIT("msgget()");
      }
      if(argc < 3)
      	msgtype = 0;
      else {
      	msgtype = atol(argv[2]);
      	if (msgtype < 0)
     			msgtype = 0;
      } /* determin msgtype (class number) */
      printf("Selected message type = %ld\n", msgtype);
      while (1) {
        ret = msgrcv(msqid, (void *)&data, TEXT_SIZE, msgtype, IPC_NOWAIT);
        /* Non_blocking receive */
        if(ret == -1) { /* end of this msgtype */
          printf("number of received messages = %d\n", count);
          break;
        }
        printf("%ld %s\n", data.msg_type, data.mtext);
        count++;
      }
      struct msqid_ds msqattr;
      ret = msgctl(msqid, IPC_STAT, &msqattr);
      printf("number of messages remainding = %ld\n", msqattr.msg_qnum);
      if(msqattr.msg_qnum == 0) {
      	printf("do you want to delete this msg queue?(y/n)");
      if (getchar() == 'y') {
        if(msgctl(msqid, IPC_RMID, 0) == -1)
        	perror("msgctl(IPC_RMID)");
        }
      }
      system("ipcs -q");
      exit(EXIT_SUCCESS);
    }

    # POSIX (pthread) 实现

    c
    #include <mqueue.h>
    # Open, Close and Unlink
    mqd_t mq_open(const char *name, int oflag, mode_t mode, struct mq_attr *attr ); /* return the mqdes, or -1 if failed */
    mqd_t mqID;
    mqID = mq_open("/anonymQueue", O_RDWR | O_CREAT, 0666, NULL);
    mqd_t mq_close(mqd_t mqdes);
    mqd_t mq_unlink(const char *name); /* return -1 if failed */
    # Send and Receive
    mqd_t mq_send(mqd_t mqdes, const char *msg_ptr, size_t msg_len, unsigned msg_prio); /* return 0, or -1 if failed */
    mq_send(mqID, msg, sizeof(msg), i)
    mqd_t mq_receive(mqd_t mqdes, char *msg_ptr, size_t msg_len, unsigned *msg_prio); /* return the number of char received, or -1 if failed */
    mq_attr mqAttr;
    mq_getattr(mqID, &mqAttr);
    mq_receive(mqID, buf, mqAttr.mq_msgsize, NULL)

    Note that the Linux ipcs utility is not fully compatible to the POSIX ipcs utility. The message queues, shared memory and semaphores in POSIX can not be sensed by System V bash command such as ipcs -q

    # Mach

    Mach 作为微内核结构,其通信是基于消息传递的设计,甚至系统调用也是消息传递

    • 每个任务在创建时获得两个端口 - 内核和通知
    • 使用 mach_msg() 函数发送和接收消息
    • 通信所需的端口,通过创建 mach 端口分配
    • 收发灵活;例如,邮箱已满时有四个选项:
      • 无限期地等待
      • 最多等待 n 毫秒
      • 立即返回
      • 临时缓存消息

    # Windows

    通过高级本地过程调用(ALPC)设施以消息传递为中心,仅在同一系统上的两个进程之间工作,类似 RPC;

    • 使用端口(如邮箱)建立和维护通信通道;
    • 使用两种类型的端口:
      • 连接端口
      • 通信端口

    通讯工作如下:

    • 客户端打开子系统连接端口对象的句柄;
    • 客户端发送一个连接请求;
    • 服务器创建一对专用通信端口,并将其中一个端口的句柄返回给客户端;
    • 客户端和服务器使用相应的端口句柄发送消息或回调,并监听回复;

    windows消息传递

    Windows 对不同大小的消息有不同的策略;When an ALPC channel is created, one of three message-passing techniques is chosen:

    1. For small messages (up to 256 bytes), the port’s message queue is used as intermediate storage, and the messages are copied from one process to the other.

    2. Larger messages must be passed through a section object, which is a region of shared memory associated with the channel.

    3. When the amount of data is too large to fit into a section object, an API is available that allows server processes to read and write directly into the address space of a client.

    # 管道通信

    “管道” 是一个特殊的共享文件,又名 pipe 文件。其实就是在内存中开辟一个大小固定的内存缓冲区

    • 普通管道 (匿名管道)—— 无法从创建它的进程外部访问。通常,父进程创建管道并使用它与它创建的子进程通信。是单向的、半双工的。
      普通管道
    • 命名管道 —— 可以在没有父子关系的情况下访问。是双向的全双工的。

    管道通信的规则、限制:

    • 管道只能采用半双工通信,某一时间段内只能实现单向的传输。
    • 如果要实现双向同时通信,则需要设置两个管道。
    • 各进程要互斥地访问管道(由操作系统实现)
    • 当管道写满时,写进程阻塞,直到读进程将管道中的数据取走,即可唤醒写进程。
    • 当管道读空时,读进程阻塞,直到写进程往管道中写入数据,即可唤醒读进程。
    • 管道中的数据一旦被读出,就彻底消失。因此,当多个进程读同一个管道时,可能会错乱。对此,通常有两种解决方案:
      • 一个管道允许多个写进程一个读进程(2014 年 408 真题高教社官方答案);
      • 允许有多个写进程多个读进程,但系统会让各个读进程轮流从管道中读数据(Linux 的方案)。

    管道通信

    写进程往管道写数据,即便管道没被写满,只要管道没空,读进程就可以从管道读数据

    读进程从管道读数据,即便管道没被读空,只要管道没满,写进程就可以往管道写数据

    # Linux 实现

    c
    #include <sys/types.h>
    #include <stdio.h>
    #include <string.h>
    #include <unistd.h>
    #define BUFFER SIZE 25
    #define READ END 0
    #define WRITE END 1
    int main(void)
    {
      char write msg[BUFFER SIZE] = "Greetings";
      char read msg[BUFFER SIZE];
      int fd[2];
      pid t pid;
      /* Program continues in Figure 3.22 */
      /* create the pipe */
      if (pipe(fd) == -1) {
        fprintf(stderr,"Pipe failed");
        return 1;
      }
      /* fork a child process */
      pid = fork();
      if (pid < 0) { /* error occurred */
        fprintf(stderr, "Fork Failed");
        return 1;
      }
      if (pid > 0) { /* parent process */
        /* close the unused end of the pipe */
        close(fd[READ END]);
        /* write to the pipe */
        write(fd[WRITE END], write msg, strlen(write msg)+1);
        /* close the write end of the pipe */
        close(fd[WRITE END]);
      }
      else { /* child process */
        /* close the unused end of the pipe */
        close(fd[WRITE END]);
        /* read from the pipe */
        read(fd[READ END], read msg, BUFFER SIZE);
        printf("read %s",read msg);
        /* close the read end of the pipe */
        close(fd[READ END]);
      }
      return 0;
    }

    利用 libc 创建、操作管道:

    c
    // 普通管道
    int pipefd[2];
    int pipe(int pipefd);
    int fcntl(int pipefd[0|1], int cmd);
    int fcntl(int pipefd[0|1], int cmd, long arg);
    ssize_t write(int pipefd[1], void* buf, size_t count);
    ssize_t read(int pipefd[0], void* buf, size_t count);
    close(pipefd[0]);
    close(pipefd[1]);
    // 命名管道
    #define FIFO pathname /* pathname: "/tmp/my_fifo" */
    unlink(FIFO); /*delete a name and possibly the file it refers to */
    mkfifo(FIFO, 0666);
    int fdw = open(FIFO, O_RDWR);
    mkfifo(FIFO, 0444);
    int fdr = open(FIFO, O_RDONLY);
    ssize_t write(int fdw, void* buf, size_t count);
    /* ssizt_t = signed int, sizt_t = unsigned int */
    ssize_t read(int fdr, void* buf, size_t count);
    close(fdw);
    close(fdr);

    管道缓冲

    c
    #include <stdio.h>
    #include <stdlib.h>
    #include <unistd.h>
    #include <fcntl.h>
    #define ERR_EXIT(m) \
      do { \
        perror(m); \
        exit(EXIT_FAILURE); \
      } while (0)
    int main(int argc, char *argv[])
    {
      int pipefd[2];
      int bufsize;
      char *buffer;
      int flags, ret, lastwritten, count, totalwritten;
      if(pipe(pipefd) == -1) /* create an ordinary pipe */
      	ERR_EXIT("pipe()");
      flags = fcntl(pipefd[1], F_GETFL);
      fcntl(pipefd[1], F_SETFL, flags | O_NONBLOCK); /* set write_end NONBLOCK */
      bufsize = atoi(argv[1]);
      printf("testing buffer size = %d\n", bufsize);
      buffer = (char *)malloc(bufsize*sizeof(char));
      if(buffer == NULL || bufsize == 0)
      	ERR_EXIT("malloc()");
      count = 0;
      while (1) {
        ret = write(pipefd[1], buffer, bufsize);
        /* bufsize is better to be 2^k */
        if(ret == -1) {
          perror("write()");
          break;
        }
        lastwritten = ret;
        count++;
      }
      totalwritten = (count-1)*bufsize +
      lastwritten;
      printf("single pipe buffer count = %d, last written = %d bytes\n", count, lastwritten);
      printf("total written = %d bytes = %d KiB\n", totalwritten, totalwritten/1024); /* pipe buffer */
      return 0;
    }

    # Pipes in UNIX CLI

    A pipe can be constructed on the UNIX command line using the | character. The complete command is ls | less

    The commands ls and less are running as individual processes. The output of ls is delivered as the input to less.

    这里的 shell 和 ls 之间、ls 和 less 之间都是普通管道通信

    # Windows 实现

    c
    #include <stdio.h>
    #include <stdlib.h>
    #include <windows.h>
    #define BUFFER SIZE 25
    int main(VOID)
    {
      HANDLE ReadHandle, WriteHandle;
      STARTUPINFO si;
      PROCESS INFORMATION pi;
      char message[BUFFER SIZE] = "Greetings";
      DWORD written;
      /* Program continues in Figure 3.24 */
      /* set up security attributes allowing pipes to be inherited */
      SECURITY ATTRIBUTES sa = {sizeof(SECURITY ATTRIBUTES),NULL,TRUE};
      /* allocate memory */
      ZeroMemory(&pi, sizeof(pi));
      /* create the pipe */
      if (!CreatePipe(&ReadHandle, &WriteHandle, &sa, 0)) {
        fprintf(stderr, "Create Pipe Failed");
        return 1;
      }
      /* establish the START INFO structure for the child process */
      GetStartupInfo(&si);
      si.hStdOutput = GetStdHandle(STD OUTPUT HANDLE);
      /* redirect standard input to the read end of the pipe */
      si.hStdInput = ReadHandle;
      si.dwFlags = STARTF USESTDHANDLES;
      /* don’t allow the child to inherit the write end of pipe */
      SetHandleInformation(WriteHandle, HANDLE FLAG INHERIT, 0);
      /* create the child process */
      CreateProcess(NULL, "child.exe", NULL, NULL,
                    TRUE, /* inherit handles */
                    0, NULL, NULL, &si, &pi);
      /* close the unused end of the pipe */
      CloseHandle(ReadHandle);
      /* the parent writes to the pipe */
      if (!WriteFile(WriteHandle, message,BUFFER SIZE,&written,NULL))
      	fprintf(stderr, "Error writing to pipe.");
      /* close the write end of the pipe */
      CloseHandle(WriteHandle);
      /* wait for the child to exit */
      WaitForSingleObject(pi.hProcess, INFINITE);
      CloseHandle(pi.hProcess);
      CloseHandle(pi.hThread);
      return 0;
    }

    # 客户机 - 服务器系统中的通信

    • 套接字(Socket)
    • 远程过程调用(RPC)

    # 套间字 (Socket)

    套接字被定义为通信的端点,是很多协议的底层实现方式,通信由一对套接字组成

    IP 地址和端口的串联–包含在消息包开头的数字,用于区分主机上的网络服务;例如:套接字 161.25.19.8:1625 引用主机 161.25.19.8 上的端口 1625 ;

    • 1024 以下的所有端口都是相对固定的,用于标准服务;
    • 特殊 IP 地址 127.0.0.1 (环回),用于表示正在运行进程的系统;

    三种类型的套接字

    • 面向连接(TCP)
    • 无连接(UDP)
    • Multicast Socket 类:数据可以发送到多个收件人

    # 远程过程调用 (RPC)

    远程过程调用(RPC)是抽象网络系统上进程之间的过程调用,也使用端口进行服务区分

    Stubs(桩函数)—— 服务器上实际过程的客户端代理,提供封装好的调用接口:

    • 客户端存根定位服务器并封送参数
    • 服务器端存根接收此消息,解压缩封送的参数,并在服务器上执行该过程

    通过外部数据表示(XDL)格式处理数据表示,以考虑不同的体系结构,例如大端和小端;

    远程通信比本地通信有更多的故障场景,消息可以只传递一次,而不是最多一次

    操作系统通常提供会合(或配对)服务来连接客户端和服务器

    在 Windows 上,存根代码根据用 Microsoft 接口定义语言(MIDL)编写的规范编译

    远程过程调用

    # DCE RPC

    参考:Distributed Computing Environment ,OSF(开放软件基金会)下的一个软件系统框架(慢慢消亡)

    DCE 的构成:

    • 远程过程调用 DCE/RPC;
    • 命名 / 目录服务;
    • 时间服务;
    • 安全服务;
    • 分布式文件系统 DCE/DFS;

    目前 Linux 标配 RPC

    # 进程调度

    Process scheduler 从可用进程中选择下一个在 CPU 核心上执行的进程

    目标 —— 最大限度地利用 CPU,快速将进程切换到 CPU 核心

    维护进程的调度队列

    • 就绪队列 (ready queue)—— 驻留在主内存中、就绪并等待执行的所有进程的集合
    • 等待队列 (wait queue)—— 等待事件(即 I/O)的一组进程
      • I/O wait queue
      • child termination wait queue
      • interrupt wait queue
    • 进程在各种队列之间迁移
    Edited on Views times
    \ No newline at end of file diff --git a/cs/os/scheduling/index.html b/cs/os/scheduling/index.html index cc99c59c..849e7e2e 100644 --- a/cs/os/scheduling/index.html +++ b/cs/os/scheduling/index.html @@ -1 +1 @@ -调度 - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 调度相关概念

    为什么需要调度?

    释放被阻塞、挂起的进程占用的 CPU 资源

    实现 I/O 操作和 CPU 并行执行( I/O 密集型和 CPU 密集型)


    调度的层次、类型

    • 处理器 / CPU 调度⭐️(进程调度的主要内容)

      • 作业调度 /long-term scheduling / 长程调度 / 高级调度

        The decision to add to the pool of processes to be executed

      • 内存调度 /medium-term scheduling / 中程调度 / 中级调度

        The decision to add to the number of processes that are partially or fully in main memory

      • (狭义?) 进程调度 /short-term scheduling / 短程调度 / 初级调度

        The decision as to which available process will be executed by the processor

    • 微调度 —— CPU 内部的指令调度(计组的内容了)

    • I/O scheduling

      The decision as to which process's pending I/O request shall be handled by an available I/O device

    # 三种层次进程调度

    进程调度的层次

    三种层次 CPU 进程调度频繁程度:长程 < 中程 < 短程

    # 长程调度

    控制了系统的并发度

    创建的进程越多,每个进程的执行时间的百分比越小,系统并发度越高;

    长程调度程序可能会限制系统的并发度,为了给当前的进程提供满意的服务;

    多道批处理系统中大多配有该层次进程调度,而其他系统中则通常没有;

    每个进程只会被该层次调度 2 次 :

    • 调入 1 次
    • 调出 1 次

    作业与进程不一定是一一对应的!也可能是一个作业对应多个进程!

    # 中程调度

    中程调度是交换功能的一部分,

    典型的情况下,换入(swapping-in)决定取决于管理系统并发度的需求。

    存储管理是一个问题,换入决策将考虑换出进程的存储需求;

    严格来讲,是进程在内存和系统盘交换区之间的交换

    关掉交换区 swqpoff ,进程就会被锁到内存里执行,可以大幅提高内存占用率高的程序效率

    # 短程调度

    也称为分派程序(dispatcher),是最基本的调度,任何系统都需要配置这种调度

    精确决定 CPU 下次执行哪一个进程;在 Ready、Running、Blocked 之间切换进程状态

    导致当前进程阻塞抢占当前运行进程的事件发生时,触发短程调度程序;

    触发三种层次进程调度的事件:

    Process
    A new process is created.When fork() is invoked and returns successfully. Then, whether the parent or the child is scheduled is up to the scheduler's decision.
    An existing process is terminated.The CPU is freed. The scheduler should choose another process to run.
    A process waits for I/OThe CPU is freed if the process is using CPU. The scheduler should choose another process to run.
    A process finishes waiting for I/OThe interrupt handling routine makes a scheduling request, if necessary.

    # 调度算法的概念

    # 分类

    # Offline

    在提前知道所有进程所有事件的前提下。

    只是用来评测的(做题就是这样的?),实际应用不可能实现。

    # Online

    实际应用的算法都是这种类型

    # 非抢占式

    Non-preemptive

    在非抢占式调度下,一旦 CPU 分配给进程,进程将保持 CPU ,直到通过终止或切换到等待状态释放 CPU 为止。也就调度仅在以下情况发生:

    • 从运行状态切换到等待状态( I/O 请求或 wait() 调用)
    • 终止

    # 抢占式

    Preemptive

    几乎所有现代操作系统,包括 Windows、MacOS、Linux 和 UNIX ,都使用抢占式调度算法。

    CPU 调度决策可能在以下情况下发生:

    • 从运行状态切换到等待状态( I/O 请求或 wait() 调用)
    • 从运行状态切换到就绪状态(出现中断)
    • 从等待切换到就绪 ( I/O 完成)
    • 终止

    当数据在多个进程之间共享时,抢占式调度可能导致竞争条件:考虑两个共享数据的进程的情况,当一个进程更新数据时,它被抢占,以便第二个进程可以运行。然后,第二个进程可能读取处于不一致状态的数据。

    抢占也影响操作系统内核的设计,在上下文切换之前,等待系统调用的完成,或者等待 I/O 阻塞的发生。确保内核简单,但是难以应用于实时系统中。

    # 评价标准与优化

    评价标准也叫做调度准则,包括:

    # CPU 利用率

    CPU utilization

    CPU利用率=CPU有效工作时间CPU有效工作时间+CPU空闲等待时间\text{CPU利用率}=\frac{\text{CPU有效工作时间}}{\text{CPU有效工作时间}+\text{CPU空闲等待时间}}

    # 吞吐量

    Throughput

    吞吐量=单位时间CPU完成的作业数量\text{吞吐量}=\frac{\text{单位时间}}{\text{CPU完成的作业数量}}

    # 周转时间

    Turnaround time

    一个作业的周转时间=作业完成时间作业提交时间n个作业的平均周转时间=作业1的周转时间+作业2的周转时间++作业n的周转时间n一个作业的带权周转时间=作业的周转时间作业实际运行时间n个作业的平均带权周转时间=作业1的带权周转时间+作业2的带权周转时间++作业n的带权周转时间n\begin{aligned} \text{一个作业的周转时间}&=\text{作业完成时间}-\text{作业提交时间}\\ n\text{个作业的平均周转时间}&=\frac{\text{作业1的周转时间}+\text{作业2的周转时间}+\cdots+\text{作业n的周转时间}}{n}\\ \text{一个作业的带权周转时间}&=\frac{\text{作业的周转时间}}{\text{作业实际运行时间}}\\ n\text{个作业的平均带权周转时间}&=\frac{\text{作业1的带权周转时间}+\text{作业2的带权周转时间}+\cdots+\text{作业n的带权周转时间}}{n}\\ \end{aligned}

    # 等待时间

    Waiting time

    一个作业的等待时间=作业处于等待CPU的时间之和\text{一个作业的等待时间}=\text{作业处于等待CPU的时间之和}

    # 响应时间

    Response time

    一个作业的响应时间=系统首次产生响应的时间用户提交请求的时间(作业提交时间?)\text{一个作业的响应时间}=\text{系统首次产生响应的时间}-\text{用户提交请求的时间(作业提交时间?)}

    # 响应比

    一个作业的响应比=作业的周转时间作业的等待时间\text{一个作业的响应比}=\frac{\text{作业的周转时间}}{\text{作业的等待时间}}

    # 优化原则

    • 最大 CPU 利用率
    • 最大吞吐量
    • 最小周转时间
    • 最小等待时间
    • 最小响应时间

    大多数情况下,优化的是平均值;有些情况下优化的是最小值或最大值,也可以最小化方差;

    调度准则的冲突

    # 基本调度算法

    AlgorithmsPreemptive?Target System
    First-come, first-served or First-in, First-out ( FCFS / FIFO )No.Out-of-date
    Shortest-job-first (SJF)Can be both.Out-of-date
    Round-robin (RR)Yes.Modern
    Priority schedulingYes.Modern
    Priority scheduling with multiple queues.The realimplementation!

    # FCFS/FIFO

    先来先服务调度。先请求 CPU 的进程首先分配到 CPU ,通过 FIFO 队列容易实现

    各个评价指标受进程到达就绪队列的时间顺序影响大,也就是对输入敏感

    # 护航效应

    FCFS / FIFO 算法容易引发护航效应,也就是很多后来的短作业被先来的长作业阻塞,导致平均等待时时间、平均周转时间长

    # 非抢占 SJF

    最短作业优先调度的非抢占式版本

    # 抢占 SJF

    ⚠️需要限制发生抢占的条件!在这里只允许有一个新进程到来的时候(被用户提交)才发生抢占

    优点:

    • 平均等待时间降低
    • 平均周转时间降低

    代价是增加了上下文切换的成本

    可以证明 SJF 调度算法是最优的,因为 SJF 的平均等待时间最小;但是这个算法需要预先知道所有进程剩余 CPU 执行时间,这在实际应用中世不可能的

    # RR

    Round Robin

    RR 专门为分时系统设计,类似 FCFS 但是增加了抢占以切换进程。

    RR 是抢占式调度的;(preemptive)

    每个进程给定一个较小的时间单位成为时间片,时间片用完后,

    CPU 选择另外一个进程调度执行;

    CPU 调度程序循环整个就绪队列,一个一个执行;

    每个进程获得一个小的 CPU 时间单位(时间量和时间片),通常为 10-100 毫秒。此时间过后,进程将被抢占并添加到就绪队列的末尾。

    如果就绪队列中有 nn 个进程,且时间片为 qq ,则每个进程一次最多以 qq 个时间单位的块获取 1n\frac{1}{n} 的 CPU 时间。没有进程等待超过 n1n-1 个时间单位。

    计时器中断每个时间片以安排下一个进程

    # 时间片

    quantum—— 时间片(量子化思想?)

    • 时间片 qq 如果非常大, 退化为先进先出(FCFS),性能降低;
    • 时间片 qq 如果非常小,导致大量的上下文切换,性能降低;
      • 所以相对于上下文切换,qq 必须很大,否则开销太高

    现代操作系统往往提供了设置时间片大小的接口

    ⚠️:时间片的大小和平均周转时间等指标没有规律性的关系,实际时间片的大小设置往往是依据实验经验的

    # 性能

    相比 SJF ,RR 的平均等待时间、平均周转时间都更差!但是响应性得到了提升!这就是 RR 的最大优势!

    # 优先级调度

    # 实时系统

    对于实时调度,调度器必须支持抢占式、基于优先级的调度,但只能保证软实时性;

    对于硬实时系统,还必须提供满足截止期限的能力,需要添加新的调度特征;

    进程新的特点:周期性进程需要以固定的时间间隔使用 CPU

    具有处理时间 TT、截止日期 DD、周期 PP0TDP0≤ T≤ D≤ P

    定期任务的速率为 1p\frac{1}{p}

    # 基本优先级调度

    优先级编号(整数)与每个进程相关联;CPU 分配给具有最高优先级(一般默认最小整数,除非题目额外说明)的进程(最高优先级)

    定义优先级

    • 内部定义
    • 外部定义

    ⚠️误区:基于优先级的调度算法并非都是抢占式的:

    • 可以是抢占式的
    • 也可以是非抢占式

    SJF 也是优先级调度,其中优先级是预测的下一个 CPU 执行时间的倒数;

    问题:饥饿 —— 低优先级进程可能永远不会执行;

    解决方案:老化 (aging)—— 随着时间的推移,进程的优先级会增加;

    # 多级队列调度

    仍然是优先级调度,也就是 “多 (优先) 级队列调度”;

    就绪作业队列分成多个单独的队列,每个队列具有不同的调度算法;

    例如分成前台进程(RR)和后台进程(FCFS);

    eg:

    PriorityScheduling algorithm
    Priority class 5Non-preemptive, FIFO
    Priority class 4Non-preemptive, SJF
    Priority class 3RR with quantum = 10 units.
    Priority class 2RR with quantum = 20 units.
    Priority class 1RR with quantum = 40 units.

    不同的队列之间应用调度,通常采用固定优先级抢占调度,如前台队列可以比后台队列具有绝对的优先,只要前台队列非空,后台队列的进程就无法被调度;

    # 多级反馈调度队列

    为了解决多级反馈调度队列中固定化优先级导致的饥饿问题,在调度程序运行时不断调整进程的优先级

    通常,多级反馈队列调度程序可由以下参数定义:

    • 队列数量;
    • 每个队列的调度算法;
    • 用于确定何时升级到高优先级队列的方法;
    • 用于确定何时降级到更低优先级队列的方法;
    • 用于确定当某个进程需要服务时该进程将进入哪个队列的方法;

    成为最通用的调度算法,通过配置适应特定的系统,也是最复杂的调度方法;

    老化可以使用多级反馈队列来实现;

    # 单调速率调度

    单调速率调度算法采用抢占的、静态优先级的策略,调度周期性任务;

    每个周期性任务分配一个优先;

    优先级是根据其周期的倒数来分配的;

    • 周期越短 => 优先级越高;
    • 周期越长 => 优先级越低;

    P1 (周期为 50, 处理时间为 20) 的优先级高于 P2(周期为 100,处理时间为 35);截止时间为下一个周期开始的时间

    单调速率调度可认为是最优的,因为如果一组进程不能由此算法调度,它不能由任何其他分配静态优先级的算法调度

    # 错过截止时间

    单调速率调度有一个限制 ——CPU 利用率是有限的,并不能完全最大化 CPU 资源利用率,调度 N 个进程的最坏情况下的 CPU 利用率为:

    n(21n1)n(2^{\frac{1}{n}}-1)

    # 最早截止期限有限调度 (EDF)

    调度根据截止期限动态分配优先级;截止期限越早,优先级越高;截止期限越晚,优先级越低;

    单调速率调度的优先级是固定的,而 EDF 是动态的;

    此外,EDF 调度不要求进程是周期执行的,也不要求进程的 CPU 时间是固定的,唯一的要求是:进程在变成可运行时,应给出它的截止时间。

    理论上,最佳情况下 CPU 利用率会到 100%。

    # 优先级翻转现象

    低优先的进程占有了高优先级进程需要的资源,特别是一些共享资源,导致高优先级进程被阻塞,这种现象就叫优先级翻转

    • 有界优先级翻转,发生优先级翻转,没有外界干预的情况下也能自己恢复
    • 无界优先级翻转,一旦发生优先级翻转,没有外界干预的情况下将无法恢复,高优先级永远被阻塞

    # 优先级继承

    # 选择函数

    选择函数决定选择哪个就绪进程下次执行;

    这个函数可以根据优先级、资源需求或进程的执行特性来进行选择;

    对于执行特性,可以根据以下三个重要的参数:

    • w (wait) = 目前为止在系统中的等待时间;
    • e (exec) = 目前为止花费的执行时间;
    • s (serv) = 进程所需要的总服务时间,包括 e;这个参数通常须进行估计或由用户提供;
    FCFSRound robinSPNSRTHRRNFeedback
    Selection functionmax[w]\max[w]constantmin[s]\min[s]min[se]\min[s-e]\max \frac{w+s}(see text)
    Decision modeNon-preemptivePreemptive (at time quantum)Non-preemptivePreemptive (at arrival)Non-preemptivePreemptive (at time quantum)
    ThroughPutNot emphasizedMay be low if guantum is too smallHighHighHighNot emphasized
    Response timeMay be high, especially if there is a large variance in process execution timesProvides good response time for short processesProvides good response time for short processesProvides good response timeProvides good response timeNot emphasized
    OverheadMinimumMinimumCan be highCan be highCan be highCan be high
    Effect on processesPenalizes short processes; penalizes I/0 bound processesFair treatmentPenalizes long processesPenalizes long processesGood balanceMay favor I/0 bound processes
    StarvationNoNoPossiblePossibleNoPossible

    TrTs=w+ss=11ρ\frac{T_r}{T_s}=\frac{w+s}{s}=\frac{1}{1-\rho}

    ρ\rho 是处理器利用率(processor utilization)

    # 公平调度算法

    # 传统的 Unix 调度

    调度形式化:

    CPUj(i)=CPUj(i1)2Pj(i)=Basej+CPUj(i)2+nicej\begin{aligned} CPU_j(i)&=\frac{CPU_j(i-1)}{2}\\ P_j(i)&=Base_j+\frac{CPU_j(i)}{2}+nice_j \end{aligned}

    chrt -m 可以用于在 Linux OS 中查看调度相关的信息

    chrt -f -p [priority-level] [pid] 修改进程的调度策略以及优先级

    # 比例分享调度

    调度程序在所有进程之间分配 TT 股(通证);

    应用程序接收 NN 股,其中 N<TN<T

    这确保每个应用程序将收到总处理器时间的 NT\frac{N}{T}

    采用准入控制策略,以确保每个进程能够得到分配时间。

    准入控制策略是:只有客户请求的股数小于可用的股数,才能允许客户进入。

    现代很多网络、磁盘共享等需要共享的情况下往往采用这种调度方式,这是一种比较公平的调度算法

    # 线程调度

    • PCS —— Process Contention Scope ,进程级争用范围、作用域,调度竞争在进程内
      • 通常情况下,PCS 采用优先级调度
    • SCS —— System Contention Scope ,系统级争用范围、作用域,调度到可用 CPU 上的内核线程,系统中所有线程之间的竞争
      • 对于一对一模型的系统,如 Windows、Linux、Solaris 等只采用 SCS 调度

    # 多处理器调度

    前面讲述的内容都是单处理器系统的调度问题,如果有多个处理器,负载分配成为可能,但是调度问题更加复杂

    主要关注同构处理器:(SMP)

    • 非对称多处理器,太复杂,不关注
    • 对称多处理器(SMP)

    对称多处理(SMP)是指每个处理器都是自调度的。

    • 所有线程都可能位于公共就绪队列(a)中
    • 每个处理器可能有自己的线程专用队列(b)

    多线程多核处理器需要两个级别的调度:

    • 一个级别由操作系统调度;
    • 另一个级别是指定每个核心如何运行哪个硬件线程

    # Linux

    # CFS

    完全公平调度,内核在 2.6 以后采用

    Linux 系统的调度基于调度类;

    每个调度类都有特定的优先权;

    调度程序在最高调度类中选择优先级最高的任务;

    而不是基于固定时间分配的时间片,基于 CPU 时间的比例;

    包括两个调度类,可以添加其他调度类;

    • 采用 CFS 调度算法的默认调度类;
    • 实时

    # 调度优先级

    # 调度算法评估

    # 排队模型

    描述进程的到达,以及 CPU 和 I/O 执行的概率

    通常是指数型的,用平均值来描述

    计算平均吞吐量、利用率、等待时间等。

    描述为服务器网络的计算机系统,每个服务器都有等待进程队列

    了解到达率和服务率

    计算利用率、平均队列长度、平均等待时间等

    # Little's Law

    # CPU 调度仿真

    排队模型能力有限

    更精确的模拟

    计算机系统的程序模型

    时钟是一个变量

    收集指示算法性能的统计信息

    用于驱动通过以下方式收集的模拟的数据:

    基于概率的随机数发生器

    数学上或经验上定义的分布

    跟踪磁带记录真实系统中真实事件的序列

    Edited on Views times
    \ No newline at end of file +调度 - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 调度相关概念

    为什么需要调度?

    释放被阻塞、挂起的进程占用的 CPU 资源

    实现 I/O 操作和 CPU 并行执行( I/O 密集型和 CPU 密集型)


    调度的层次、类型

    • 处理器 / CPU 调度⭐️(进程调度的主要内容)

      • 作业调度 /long-term scheduling / 长程调度 / 高级调度

        The decision to add to the pool of processes to be executed

      • 内存调度 /medium-term scheduling / 中程调度 / 中级调度

        The decision to add to the number of processes that are partially or fully in main memory

      • (狭义?) 进程调度 /short-term scheduling / 短程调度 / 初级调度

        The decision as to which available process will be executed by the processor

    • 微调度 —— CPU 内部的指令调度(计组的内容了)

    • I/O scheduling

      The decision as to which process's pending I/O request shall be handled by an available I/O device

    # 三种层次进程调度

    进程调度的层次

    三种层次 CPU 进程调度频繁程度:长程 < 中程 < 短程

    # 长程调度

    控制了系统的并发度

    创建的进程越多,每个进程的执行时间的百分比越小,系统并发度越高;

    长程调度程序可能会限制系统的并发度,为了给当前的进程提供满意的服务;

    多道批处理系统中大多配有该层次进程调度,而其他系统中则通常没有;

    每个进程只会被该层次调度 2 次 :

    • 调入 1 次
    • 调出 1 次

    作业与进程不一定是一一对应的!也可能是一个作业对应多个进程!

    # 中程调度

    中程调度是交换功能的一部分,

    典型的情况下,换入(swapping-in)决定取决于管理系统并发度的需求。

    存储管理是一个问题,换入决策将考虑换出进程的存储需求;

    严格来讲,是进程在内存和系统盘交换区之间的交换

    关掉交换区 swqpoff ,进程就会被锁到内存里执行,可以大幅提高内存占用率高的程序效率

    # 短程调度

    也称为分派程序(dispatcher),是最基本的调度,任何系统都需要配置这种调度

    精确决定 CPU 下次执行哪一个进程;在 Ready、Running、Blocked 之间切换进程状态

    导致当前进程阻塞抢占当前运行进程的事件发生时,触发短程调度程序;

    触发三种层次进程调度的事件:

    Process
    A new process is created.When fork() is invoked and returns successfully. Then, whether the parent or the child is scheduled is up to the scheduler's decision.
    An existing process is terminated.The CPU is freed. The scheduler should choose another process to run.
    A process waits for I/OThe CPU is freed if the process is using CPU. The scheduler should choose another process to run.
    A process finishes waiting for I/OThe interrupt handling routine makes a scheduling request, if necessary.

    # 调度算法的概念

    # 分类

    # Offline

    在提前知道所有进程所有事件的前提下。

    只是用来评测的(做题就是这样的?),实际应用不可能实现。

    # Online

    实际应用的算法都是这种类型

    # 非抢占式

    Non-preemptive

    在非抢占式调度下,一旦 CPU 分配给进程,进程将保持 CPU ,直到通过终止或切换到等待状态释放 CPU 为止。也就调度仅在以下情况发生:

    • 从运行状态切换到等待状态( I/O 请求或 wait() 调用)
    • 终止

    # 抢占式

    Preemptive

    几乎所有现代操作系统,包括 Windows、MacOS、Linux 和 UNIX ,都使用抢占式调度算法。

    CPU 调度决策可能在以下情况下发生:

    • 从运行状态切换到等待状态( I/O 请求或 wait() 调用)
    • 从运行状态切换到就绪状态(出现中断)
    • 从等待切换到就绪 ( I/O 完成)
    • 终止

    当数据在多个进程之间共享时,抢占式调度可能导致竞争条件:考虑两个共享数据的进程的情况,当一个进程更新数据时,它被抢占,以便第二个进程可以运行。然后,第二个进程可能读取处于不一致状态的数据。

    抢占也影响操作系统内核的设计,在上下文切换之前,等待系统调用的完成,或者等待 I/O 阻塞的发生。确保内核简单,但是难以应用于实时系统中。

    # 评价标准与优化

    评价标准也叫做调度准则,包括:

    # CPU 利用率

    CPU utilization

    CPU利用率=CPU有效工作时间CPU有效工作时间+CPU空闲等待时间\text{CPU利用率}=\frac{\text{CPU有效工作时间}}{\text{CPU有效工作时间}+\text{CPU空闲等待时间}}

    # 吞吐量

    Throughput

    吞吐量=单位时间CPU完成的作业数量\text{吞吐量}=\frac{\text{单位时间}}{\text{CPU完成的作业数量}}

    # 周转时间

    Turnaround time

    一个作业的周转时间=作业完成时间作业提交时间n个作业的平均周转时间=作业1的周转时间+作业2的周转时间++作业n的周转时间n一个作业的带权周转时间=作业的周转时间作业实际运行时间n个作业的平均带权周转时间=作业1的带权周转时间+作业2的带权周转时间++作业n的带权周转时间n\begin{aligned} \text{一个作业的周转时间}&=\text{作业完成时间}-\text{作业提交时间}\\ n\text{个作业的平均周转时间}&=\frac{\text{作业1的周转时间}+\text{作业2的周转时间}+\cdots+\text{作业n的周转时间}}{n}\\ \text{一个作业的带权周转时间}&=\frac{\text{作业的周转时间}}{\text{作业实际运行时间}}\\ n\text{个作业的平均带权周转时间}&=\frac{\text{作业1的带权周转时间}+\text{作业2的带权周转时间}+\cdots+\text{作业n的带权周转时间}}{n}\\ \end{aligned}

    # 等待时间

    Waiting time

    一个作业的等待时间=作业处于等待CPU的时间之和\text{一个作业的等待时间}=\text{作业处于等待CPU的时间之和}

    # 响应时间

    Response time

    一个作业的响应时间=系统首次产生响应的时间用户提交请求的时间(作业提交时间?)\text{一个作业的响应时间}=\text{系统首次产生响应的时间}-\text{用户提交请求的时间(作业提交时间?)}

    # 响应比

    一个作业的响应比=作业的周转时间作业的等待时间\text{一个作业的响应比}=\frac{\text{作业的周转时间}}{\text{作业的等待时间}}

    # 优化原则

    • 最大 CPU 利用率
    • 最大吞吐量
    • 最小周转时间
    • 最小等待时间
    • 最小响应时间

    大多数情况下,优化的是平均值;有些情况下优化的是最小值或最大值,也可以最小化方差;

    调度准则的冲突

    # 基本调度算法

    AlgorithmsPreemptive?Target System
    First-come, first-served or First-in, First-out ( FCFS / FIFO )No.Out-of-date
    Shortest-job-first (SJF)Can be both.Out-of-date
    Round-robin (RR)Yes.Modern
    Priority schedulingYes.Modern
    Priority scheduling with multiple queues.The realimplementation!

    # FCFS/FIFO

    先来先服务调度。先请求 CPU 的进程首先分配到 CPU ,通过 FIFO 队列容易实现

    各个评价指标受进程到达就绪队列的时间顺序影响大,也就是对输入敏感

    # 护航效应

    FCFS / FIFO 算法容易引发护航效应,也就是很多后来的短作业被先来的长作业阻塞,导致平均等待时时间、平均周转时间长

    # 非抢占 SJF

    最短作业优先调度的非抢占式版本

    # 抢占 SJF

    ⚠️需要限制发生抢占的条件!在这里只允许有一个新进程到来的时候(被用户提交)才发生抢占

    优点:

    • 平均等待时间降低
    • 平均周转时间降低

    代价是增加了上下文切换的成本

    可以证明 SJF 调度算法是最优的,因为 SJF 的平均等待时间最小;但是这个算法需要预先知道所有进程剩余 CPU 执行时间,这在实际应用中世不可能的

    # RR

    Round Robin

    RR 专门为分时系统设计,类似 FCFS 但是增加了抢占以切换进程。

    RR 是抢占式调度的;(preemptive)

    每个进程给定一个较小的时间单位成为时间片,时间片用完后,

    CPU 选择另外一个进程调度执行;

    CPU 调度程序循环整个就绪队列,一个一个执行;

    每个进程获得一个小的 CPU 时间单位(时间量和时间片),通常为 10-100 毫秒。此时间过后,进程将被抢占并添加到就绪队列的末尾。

    如果就绪队列中有 nn 个进程,且时间片为 qq ,则每个进程一次最多以 qq 个时间单位的块获取 1n\frac{1}{n} 的 CPU 时间。没有进程等待超过 n1n-1 个时间单位。

    计时器中断每个时间片以安排下一个进程

    # 时间片

    quantum—— 时间片(量子化思想?)

    • 时间片 qq 如果非常大, 退化为先进先出(FCFS),性能降低;
    • 时间片 qq 如果非常小,导致大量的上下文切换,性能降低;
      • 所以相对于上下文切换,qq 必须很大,否则开销太高

    现代操作系统往往提供了设置时间片大小的接口

    ⚠️:时间片的大小和平均周转时间等指标没有规律性的关系,实际时间片的大小设置往往是依据实验经验的

    # 性能

    相比 SJF ,RR 的平均等待时间、平均周转时间都更差!但是响应性得到了提升!这就是 RR 的最大优势!

    # 优先级调度

    # 实时系统

    对于实时调度,调度器必须支持抢占式、基于优先级的调度,但只能保证软实时性;

    对于硬实时系统,还必须提供满足截止期限的能力,需要添加新的调度特征;

    进程新的特点:周期性进程需要以固定的时间间隔使用 CPU

    具有处理时间 TT、截止日期 DD、周期 PP0TDP0≤ T≤ D≤ P

    定期任务的速率为 1p\frac{1}{p}

    # 基本优先级调度

    优先级编号(整数)与每个进程相关联;CPU 分配给具有最高优先级(一般默认最小整数,除非题目额外说明)的进程(最高优先级)

    定义优先级

    • 内部定义
    • 外部定义

    ⚠️误区:基于优先级的调度算法并非都是抢占式的:

    • 可以是抢占式的
    • 也可以是非抢占式

    SJF 也是优先级调度,其中优先级是预测的下一个 CPU 执行时间的倒数;

    问题:饥饿 —— 低优先级进程可能永远不会执行;

    解决方案:老化 (aging)—— 随着时间的推移,进程的优先级会增加;

    # 多级队列调度

    仍然是优先级调度,也就是 “多 (优先) 级队列调度”;

    就绪作业队列分成多个单独的队列,每个队列具有不同的调度算法;

    例如分成前台进程(RR)和后台进程(FCFS);

    eg:

    PriorityScheduling algorithm
    Priority class 5Non-preemptive, FIFO
    Priority class 4Non-preemptive, SJF
    Priority class 3RR with quantum = 10 units.
    Priority class 2RR with quantum = 20 units.
    Priority class 1RR with quantum = 40 units.

    不同的队列之间应用调度,通常采用固定优先级抢占调度,如前台队列可以比后台队列具有绝对的优先,只要前台队列非空,后台队列的进程就无法被调度;

    # 多级反馈调度队列

    为了解决多级反馈调度队列中固定化优先级导致的饥饿问题,在调度程序运行时不断调整进程的优先级

    通常,多级反馈队列调度程序可由以下参数定义:

    • 队列数量;
    • 每个队列的调度算法;
    • 用于确定何时升级到高优先级队列的方法;
    • 用于确定何时降级到更低优先级队列的方法;
    • 用于确定当某个进程需要服务时该进程将进入哪个队列的方法;

    成为最通用的调度算法,通过配置适应特定的系统,也是最复杂的调度方法;

    老化可以使用多级反馈队列来实现;

    # 单调速率调度

    单调速率调度算法采用抢占的、静态优先级的策略,调度周期性任务;

    每个周期性任务分配一个优先;

    优先级是根据其周期的倒数来分配的;

    • 周期越短 => 优先级越高;
    • 周期越长 => 优先级越低;

    P1 (周期为 50, 处理时间为 20) 的优先级高于 P2(周期为 100,处理时间为 35);截止时间为下一个周期开始的时间

    单调速率调度可认为是最优的,因为如果一组进程不能由此算法调度,它不能由任何其他分配静态优先级的算法调度

    # 错过截止时间

    单调速率调度有一个限制 ——CPU 利用率是有限的,并不能完全最大化 CPU 资源利用率,调度 N 个进程的最坏情况下的 CPU 利用率为:

    n(21n1)n(2^{\frac{1}{n}}-1)

    # 最早截止期限有限调度 (EDF)

    调度根据截止期限动态分配优先级;截止期限越早,优先级越高;截止期限越晚,优先级越低;

    单调速率调度的优先级是固定的,而 EDF 是动态的;

    此外,EDF 调度不要求进程是周期执行的,也不要求进程的 CPU 时间是固定的,唯一的要求是:进程在变成可运行时,应给出它的截止时间。

    理论上,最佳情况下 CPU 利用率会到 100%。

    # 优先级翻转现象

    低优先的进程占有了高优先级进程需要的资源,特别是一些共享资源,导致高优先级进程被阻塞,这种现象就叫优先级翻转

    • 有界优先级翻转,发生优先级翻转,没有外界干预的情况下也能自己恢复
    • 无界优先级翻转,一旦发生优先级翻转,没有外界干预的情况下将无法恢复,高优先级永远被阻塞

    # 优先级继承

    # 选择函数

    选择函数决定选择哪个就绪进程下次执行;

    这个函数可以根据优先级、资源需求或进程的执行特性来进行选择;

    对于执行特性,可以根据以下三个重要的参数:

    • w (wait) = 目前为止在系统中的等待时间;
    • e (exec) = 目前为止花费的执行时间;
    • s (serv) = 进程所需要的总服务时间,包括 e;这个参数通常须进行估计或由用户提供;
    FCFSRound robinSPNSRTHRRNFeedback
    Selection functionmax[w]\max[w]constantmin[s]\min[s]min[se]\min[s-e]\max \frac{w+s}(see text)
    Decision modeNon-preemptivePreemptive (at time quantum)Non-preemptivePreemptive (at arrival)Non-preemptivePreemptive (at time quantum)
    ThroughPutNot emphasizedMay be low if guantum is too smallHighHighHighNot emphasized
    Response timeMay be high, especially if there is a large variance in process execution timesProvides good response time for short processesProvides good response time for short processesProvides good response timeProvides good response timeNot emphasized
    OverheadMinimumMinimumCan be highCan be highCan be highCan be high
    Effect on processesPenalizes short processes; penalizes I/0 bound processesFair treatmentPenalizes long processesPenalizes long processesGood balanceMay favor I/0 bound processes
    StarvationNoNoPossiblePossibleNoPossible

    TrTs=w+ss=11ρ\frac{T_r}{T_s}=\frac{w+s}{s}=\frac{1}{1-\rho}

    ρ\rho 是处理器利用率(processor utilization)

    # 公平调度算法

    # 传统的 Unix 调度

    调度形式化:

    CPUj(i)=CPUj(i1)2Pj(i)=Basej+CPUj(i)2+nicej\begin{aligned} CPU_j(i)&=\frac{CPU_j(i-1)}{2}\\ P_j(i)&=Base_j+\frac{CPU_j(i)}{2}+nice_j \end{aligned}

    chrt -m 可以用于在 Linux OS 中查看调度相关的信息

    chrt -f -p [priority-level] [pid] 修改进程的调度策略以及优先级

    # 比例分享调度

    调度程序在所有进程之间分配 TT 股(通证);

    应用程序接收 NN 股,其中 N<TN<T

    这确保每个应用程序将收到总处理器时间的 NT\frac{N}{T}

    采用准入控制策略,以确保每个进程能够得到分配时间。

    准入控制策略是:只有客户请求的股数小于可用的股数,才能允许客户进入。

    现代很多网络、磁盘共享等需要共享的情况下往往采用这种调度方式,这是一种比较公平的调度算法

    # 线程调度

    • PCS —— Process Contention Scope ,进程级争用范围、作用域,调度竞争在进程内
      • 通常情况下,PCS 采用优先级调度
    • SCS —— System Contention Scope ,系统级争用范围、作用域,调度到可用 CPU 上的内核线程,系统中所有线程之间的竞争
      • 对于一对一模型的系统,如 Windows、Linux、Solaris 等只采用 SCS 调度

    # 多处理器调度

    前面讲述的内容都是单处理器系统的调度问题,如果有多个处理器,负载分配成为可能,但是调度问题更加复杂

    主要关注同构处理器:(SMP)

    • 非对称多处理器,太复杂,不关注
    • 对称多处理器(SMP)

    对称多处理(SMP)是指每个处理器都是自调度的。

    • 所有线程都可能位于公共就绪队列(a)中
    • 每个处理器可能有自己的线程专用队列(b)

    多线程多核处理器需要两个级别的调度:

    • 一个级别由操作系统调度;
    • 另一个级别是指定每个核心如何运行哪个硬件线程

    # Linux

    # CFS

    完全公平调度,内核在 2.6 以后采用

    Linux 系统的调度基于调度类;

    每个调度类都有特定的优先权;

    调度程序在最高调度类中选择优先级最高的任务;

    而不是基于固定时间分配的时间片,基于 CPU 时间的比例;

    包括两个调度类,可以添加其他调度类;

    • 采用 CFS 调度算法的默认调度类;
    • 实时

    # 调度优先级

    # 调度算法评估

    # 排队模型

    描述进程的到达,以及 CPU 和 I/O 执行的概率

    通常是指数型的,用平均值来描述

    计算平均吞吐量、利用率、等待时间等。

    描述为服务器网络的计算机系统,每个服务器都有等待进程队列

    了解到达率和服务率

    计算利用率、平均队列长度、平均等待时间等

    # Little's Law

    # CPU 调度仿真

    排队模型能力有限

    更精确的模拟

    计算机系统的程序模型

    时钟是一个变量

    收集指示算法性能的统计信息

    用于驱动通过以下方式收集的模拟的数据:

    基于概率的随机数发生器

    数学上或经验上定义的分布

    跟踪磁带记录真实系统中真实事件的序列

    Edited on Views times
    \ No newline at end of file diff --git a/cs/os/sync/index.html b/cs/os/sync/index.html index a26ef443..bcbb83a8 100644 --- a/cs/os/sync/index.html +++ b/cs/os/sync/index.html @@ -1 +1 @@ -经典进程同步问题 - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    This article isn't finished yet...

    # 严格备选

    (Strict Alternation)

    # PetersonSolution

    # 互斥锁

    互斥锁是一种外部强加的控制方案,或者说是全局控制;

    获得锁:

    伪代码
    acquire {
        while (!available)
        ; /* busy wait */
        available = false;
    }

    释放锁:

    伪代码
    release {
      	available = true;
    }

    利用互斥锁的临界区问题解决方案:

    伪代码
    while (true) {
      	acquire lock
          	/* critical section */
        release lock
          	/* others */
    }

    但是这个方案需要繁忙的等待,所以也叫自旋锁(Spinlock)

    存在优点:当进程在等待锁时,没有上下文切换,在使用锁的时间较短时,自旋锁是有用的;自旋锁通常用于多处理器系统

    # 信号量

    信号量是一种内部主动的控制方案,或者说是局部控制;

    # 管程

    管程(monitor)是一种面向对象的高级程序设计语言结构,它提供的功能与信号量相同,但更易于控制;

    管程结构在很多程序设计语言中得以实现,包括:

    • 并发 Pascal
    • Pascal
    • Pascal-Plus
    • Modula-2
    • Modula-3
    • Java

    现在已经被作为一个程序库实现;

    管程是由一个或多个进程、一个初始化序列和局部数据组成的软件模块;(使用信号的管程)

    c
    monitor monitor name
    {
        /* shared variable declarations */
        function P1 ( ... ) {
        		...
        }
        function P2 ( ... ) {
        		...
        }
        .
        .
        .
        function Pn ( ... ) {
        		...
        }
        initialization code ( ... ) {
        		...
        }
    }

    # 条件变量

    定义附加的同步机制,可以由条件(condition)结构来提供;

    当程序员需要编写定制的同步方案时,可以定义一个或者多个类型为 condition 的变量;

    对于条件变量,只有操作 wait()signal 可以调用;

    利用条件变量实现互斥访问以及控制执行顺序,这里 S1 必须在 S2 之前执行:

    c
    F1(){
        S1;
        		done = true;
        x.signal();
    }
    F2(){
        if done == false
        		x.wait()
        S2;
    }

    # 信号量实现

    c
    // Variables
        semaphore mutex; // (initially = 1)
        semaphore next; // (initially = 0)
        int next_count = 0; // number of processes waiting
    // inside the monitor
    // Each function P will be replaced by
        wait(mutex);
            ...
            body of P;
            ...
        if (next_count > 0)
            signal(next)
        else
            signal(mutex);
    // Mutual exclusion within a monitor is ensured

    # 条件变量实现

    c
    //For each condition variable x, we have:
        semaphore x_sem; // (initially = 0)
        int x_count = 0;
    //The operation x.wait() can be implemented as:
        x_count++;
        if (next_count > 0)
            signal(next);
        else
            signal(mutex);
        wait(x_sem);
        x_count--;

    其中 signal 方法实现参考:

    c
    if (x_count > 0) {
        next_count++;
        signal(x_sem);
        wait(next);
        next_count--;
    }

    # 进程恢复

    • 唤醒并等待(signal and wait)
      • 进程 P 等待进程直到进程 Q 离开管程,或者等待另一个条件;
      • 等待 Q 的时候若 P 想再次执行,需要测试 P 是否还具有执行的环境条件
    • 唤醒并继续(signal and continue)
      • 进程 Q 等待直到进程 P 离开管程或者等待另一个条件;

    # java

    java
    public class BlockedQueue<T> (
        final Lock lock = new ReentrantLock () ;
        // 条件变量:队列不满
        final Condition notFull = lock.newCondition () ;
        // 条件变量:队列不空
        final Condition notEmpty = lock.newCondition ();
        // 入队
        void enq (T x) {
            lock. lock() ;
            try {
                whileStack已满){
                    // 等待队列不满
                    notFull.await () ;
                }
                // 省略入队操作
                // 入队后,通知可出队
                notEmpty.signal ();
            } finally {
            		lock.unlock () ;
            }
        }
        // 出队
        void deq() {
            lock.lock () ;
            try {
                while(队列已空){
                // 等待队列不空
                notEmpty.await () ;
                }
                // 省略出队操作.
                // 出队后,通知可入队
                notFull.signal () ;
            } finally {
            		lock.unlock () ;
        }
    }

    # 管程模型 (补充)

    Hasen、Hoare 和 MESA 模型区别

    Hasen 模型、Hoare 模型和 MESA 模型的一个核心区别就是当条件满足后,如何通知相关线程。管程要求同一时刻只允许一个线程执行,那当线程 T2 的操作使线程 T1 等待的条件满足时,T1 和 T2 究竟谁可以执行呢?

    • Hasen 模型里面,要求 notify() 放在代码的最后,这样 T2 通知完 T1 后,T2 就结束了,然后 T1 再执行,这样就能保证同一时刻只有一个线程执行。
    • Hoare 模型里面,T2 通知完 T1 后,T2 阻塞,T1 马上执行;等 T1 执行完,再唤醒 T2,也能保证同一时刻只有一个线程执行。但是相比 Hasen 模型,T2 多了一次阻塞唤醒操作。
    • MESA 管程里面,T2 通知完 T1 后,T2 还是会接着执行,T1 并不立即执行,仅仅是从条件变量的等待队列进到入口等待队列里面。这样做的好处是 notify() 不用放到代码的最后,T2 也没有多余的阻塞唤醒操作。但是也有个副作用,就是当 T1 再次执行的时候,可能曾经满足的条件,现在已经不满足了,所以需要以循环方式检验条件变量。

    对于 MESA 管程来说,有一个编程范式,就是需要在一个 while 循环里面调用 wait() 。这个是 MESA 管程特有的!

    # 生产者 - 消费者问题

    # 信号量实现

    producer
    while (true) {
        ...
        /* produce an item in next produced */
        ...
        wait(empty);
        wait(mutex);  // 对应的 signal 不是本进程的,而一定是 consumer 进程的 signal !
        ...
        /* add next produced to the buffer */
        ...
        signal(mutex);  // 对应的 wait 不是本进程的,而一定是 consumer 进程的 wait !
        signal(full);
    }
    consumer
    while (true) {
        wait(full);
        wait(mutex);  // 对应的 signal 不是本进程的,而一定是 producer 进程的 signal !
        ...
        /* remove an item from buffer to next consumed */
        ...
        signal(mutex);  // 对应的 wait 不是本进程的,而一定是 producer 进程的 wait !
        signal(empty);
        ...
        /* consume the item in next consumed */
        ...
    }

    # 错误案例

    c
    int n;
    binary_semaphore s = 1, delay = 0;
    void producer(){
        while (true) {
            produce();
            semWaitB(s);
            append();
            n++;
            if (n==1) semSignalB(delay);
            semSignalB(s);
        }
    }
    void consumer(){
        semWaitB(delay);
        while (true) {
            semWaitB(s);
            take();
            n--;
            semSignalB(s);
            consume();  /* 这里没有互斥,producer 打断则会导致后续 n 出现负值 */
            if (n==0) semWaitB(delay);
        }
    }
    void main(){
        n = 0;
        parbegin (producer, consumer);
    }

    consumer 的正确实现:

    c
    void consumer (){
        int m; /* a local variable */
      	semWaitB(delay) ;
        while (true) {
            semWaitB(s);
            take ( );
            n--;
            m = ni
            semSignalB(s) ;
            consume () ;
            if (m==0) semWaitB(delay);
        }
    }

    # 无限队列生产者消费者问题实现

    缓冲区无大小限制

    c
    /* program producerconsumer */
    semaphore n = 0, s = 1;
    void producer(){
        while (true) {
            produce();
            semWait(s);
            append();
            semSignal(s);
            semSignal(n);
        }
    }
    void consumer(){
        while (true) {
            semWait(n);
            semWait(s);
            take();
            semSignal(s);
            consume();
        }
    }
    void main(){
    		parbegin (producer, consumer);
    }

    # 管程实现

    c
    void append (char x){
        while (count == N) wait(notfull);   /* buffer is full; avoid overflow */
        buffer [nextin] = x;
        nextin = (nextin + 1) % N;
        count++                             /* one more item in buffer */
        cnotify (notempty) ;                /* notify any waiting consumer */
    }
    void take (char x){
        while (count == 0) wait(notempty) ; /* buffer is empty; avoid underflow */
        x = buffer[nextout];
        nextout = (nextout + 1) % N;
        count --;                           /* one fewer item in buffer */
      	cnotify(notfull) ;                  /* notify any waiting producer */
    }

    # 共享内存方式 (pthread)

    利用 POSIX API (c 的 pthread) 实现共享内存。The API is supported by Linux 2.4 and later, FreeBSD, … 编译时需要: gcc -lrt filename.c

    shmpthreadcon.c

    c
    #include <stdio.h>
    #include <stdlib.h>
    #include <unistd.h>
    #include <sys/stat.h>
    #include <sys/wait.h>
    #include <fcntl.h>
    #include <sys/mman.h>
    #include "shmdata.h"
    /* gcc -lrt */
    int main(int argc, char *argv[])
    {
      char pathname[80], cmd_str[80];
      struct stat fileattr;
      int fd, shmsize, ret;
      pid_t childpid1, childpid2;
      if(argc < 2) {
        printf("Usage: ./a.out filename\n");
        return EXIT_FAILURE;
      }
      fd = shm_open(argv[1], O_CREAT|O_RDWR, 0666);
      /* /dev/shm/filename as the shared object, creating if not exist */
      if(fd == -1) {
      	ERR_EXIT("con: shm_open()");
      }
      system("ls -l /dev/shm/");
      shmsize = TEXT_NUM*sizeof(struct shared_struct);
      ret = ftruncate(fd, shmsize);
      if(ret == -1) {
        ERR_EXIT"con: ftruncate()");
        char *argv1[] = {" ", argv[1], 0};
        childpid1 = vfork();
        if(childpid1 < 0) {
        	ERR_EXIT("shmpthreadcon: 1st vfork()");
        }
        else if(childpid1 == 0) {
        	execv("./shmproducer", argv1); /* call producer with filename */
        }
        else {
        	childpid2 = vfork();
        if(childpid2 < 0)
        	ERR_EXIT("shmpthreadcon: 2nd vfork()");
        else if (childpid2 == 0)
        	execv("shmconsumer.o", argv1); /* call consumer with filename */
        else {
          wait(&childpid1);
          wait(&childpid2);
          ret = shm_unlink(argv[1]);
          if(ret == -1) {
          	ERR_EXIT("con: shm_unlink()");
          } /* shared object can be removed by any process knew the filename */
          system("ls -l /dev/shm/");
        }
      }
    	exit(EXIT_SUCCESS);
    }

    # 读者 - 写者问题

    一个数据集在多个并发进程之间共享

    • Readers—— 仅读取数据集;它们不执行任何更新
    • Writer—— 既能读又能写

    🤔问题:允许多个读者同时读取

    • 只有一个写者程序可以同时访问共享数据;
    • 写者在访问数据时,不允许读者访问数据;

    读者和作者的考虑方式有几种不同 —— 都涉及某种形式的

    优先权

    • 第一读者写者问题:读者优先;会导致写者饥饿
    • 第二读者写者问题:写者优先;会导致读者饥饿(相对复杂一些)

    # 信号量实现读者优先

    c
    int readcount;
    semaphore x = 1, wsem = 1;
    void reader (){
        while (true) {
            semWait (x);
            readcount++;
            if (readcount == 1) semWait(wsem);
          	semSignal(x);
            READUNIT();
            semWait(x);
            readcount--;
            if (readcount == 0) semSignal(wsem);
          	semSignal (x);
        }
    }
    void writer (){
        while (true) {
            semWait (wsem) ;
            WRITEUNIT () ;
            semSignal (wsem) ;
        }
    }
    void main (){
        readcount = 0;
        parbegin (reader, writer);
    }

    # 单读者 - 单写者

    Single-writer-single-reader problem

    # 共享内存方式

    利用 Linux 的接口实现

    shmdata.h

    c
    #define TEXT_SIZE 4*1024 /* = PAGE_SIZE, size of each message */
    #define TEXT_NUM 1 /* maximal number of messages */
    /* total size can not exceed current shmmax, or an 'invalid argument' error occurs when shmget */
    #define PERM S_IRUSR|S_IWUSR|IPC_CREAT
    #define ERR_EXIT(m) \
      do { \
        perror(m); \
        exit(EXIT_FAILURE); \
      } while(0)
    /* a demo structure, modified as needed */
    struct shared_struct {
    	int written; /* flag = 0: buffer writable; others: readable */
    	char mtext[TEXT_SIZE]; /* buffer for message reading and writing */
    };

    shmcon.c

    c
    #include <stdio.h>
    #include <stdlib.h>
    #include <unistd.h>
    #include <sys/stat.h>
    #include <sys/wait.h>
    #include <sys/shm.h>
    #include <fcntl.h>
    #include "alg.8-0-shmdata.h"
    int main(int argc, char *argv[]){
      struct stat fileattr;
      key_t key; /* of type int */
      int shmid; /* shared memory ID */
      void *shmptr;
      struct shared_struct *shared; /* structured shm */
      pid_t childpid1, childpid2;
      char pathname[80], key_str[10], cmd_str[80];
      int shmsize, ret;
      shmsize = TEXT_NUM*sizeof(struct shared_struct);
      printf("max record number = %d, shm size = %d\n", TEXT_NUM, shmsize);
      if(argc <2) {
        printf("Usage: ./a.out pathname\n");
        return EXIT_FAILURE;
      }
      strcpy(pathname, argv[1]);
      if(stat(pathname, &fileattr) == -1) {
          ret = creat(pathname, O_RDWR);
        if (ret == -1) {
          ERR_EXIT("creat()");
        }
        printf("shared file object created\n");
      }
      key = ftok(pathname, 0x27); /* 0x27 a pro_id 0x0001 - 0xffff, 8 least bits used */
      if(key == -1) {
      	ERR_EXIT("shmcon: ftok()");
      }
      printf("key generated: IPC key = 0x%x\n", key); /* set any key>0 without ftok() */
      shmid = shmget((key_t)key, shmsize, 0666|PERM);
      if(shmid == -1) {
      	ERR_EXIT("shmcon: shmget()");
      }
      printf("shmcon: shmid = %d\n", shmid);
      shmptr = shmat(shmid, 0, 0); /* returns the virtual base address mapping to the
      shared memory, *shmaddr=0 decided by kernel */
      if(shmptr == (void *)-1) {
      	ERR_EXIT("shmcon: shmat()");
      }
      printf("shmcon: shared Memory attached at %p\n", shmptr);
      shared = (struct shared_struct *)shmptr;
      shared->written = 0;
      sprintf(cmd_str, "ipcs -m | grep '%d'\n", shmid);
      printf("\n------ Shared Memory Segments ------\n");
      system(cmd_str);
      if(shmdt(shmptr) == -1) {
      	ERR_EXIT("shmcon: shmdt()");
      }
      printf("\n------ Shared Memory Segments ------\n");
      system(cmd_str);
      sprintf(key_str, "%x", key);
      char *argv1[] = {" ", key_str, 0};
      childpid1 = vfork();
      if(childpid1 < 0) {
      	ERR_EXIT("shmcon: 1st vfork()");
      }
      else if(childpid1 == 0) {
      	execv("./alg.8-2-shmread.o", argv1); /* call shm_read with IPC key */
      }
      else {
      	childpid2 = vfork();
        if(childpid2 < 0) {
          ERR_EXIT("shmcon: 2nd vfork()");
        }
        else if (childpid2 == 0) {
          execv("./alg.8-3-shmwrite.o", argv1); /* call shmwrite with IPC key */
        } else {
          wait(&childpid1);
          wait(&childpid2);
          /* shmid can be removed by any process known the
          IPC key */
          if (shmctl(shmid, IPC_RMID, 0) == -1) {
            ERR_EXIT("shmcon: shmctl(IPC_RMID)");
          } else {
            printf("shmcon: shmid = %d removed \n", shmid);
            printf("\n------ Shared Memory Segments ------\n");
            system(cmd_str);
            printf("nothing found ...\n");
            return EXIT_SUCCESS;
          }
        }
      }
    }

    shmread.c

    c
    #include <stdio.h>
    #include <stdlib.h>
    #include <unistd.h>
    #include <sys/stat.h>
    #include <string.h>
    #include <sys/shm.h>
    #include "alg.8-0-shmdata.h"
    int main(int argc, char *argv[])
    {
      void *shmptr = NULL;
      struct shared_struct *shared;
      int shmid;
      key_t key;
      sscanf(argv[1], "%x", &key);
      printf("%*sshmread: IPC key = 0x%x\n", 30, " ", key);
      shmid = shmget((key_t)key, TEXT_NUM*sizeof(struct shared_struct), 0666|PERM);
      if (shmid == -1) {
        ERR_EXIT("shread: shmget()");
      }
      shmptr = shmat(shmid, 0, 0);
      if(shmptr == (void *)-1) {
      	ERR_EXIT("shread: shmat()");
      }
      printf("%*sshmread: shmid = %d\n", 30, " ", shmid);
      printf("%*sshmread: shared memory attached at %p\n", 30, " ", shmptr);
      printf("%*sshmread process ready ...\n", 30, " ");
      shared = (struct shared_struct *)shmptr;
      while (1) {
        while (shared->written == 0) {
          sleep(1); /* message not ready, waiting ... */
        }
        printf("%*sYou wrote: %s\n", 30, " ", shared->mtext);
        shared->written = 0;
        if (strncmp(shared->mtext, "end", 3) == 0) {
          break;
        }
      } /* it is not reliable to use shared->written for process synchronization */
      if (shmdt(shmptr) == -1) {
      	ERR_EXIT("shmread: shmdt()");
      }
      sleep(1);
      exit(EXIT_SUCCESS);
    }

    shmwrite.c

    c
    #include <stdio.h>
    #include <stdlib.h>
    #include <unistd.h>
    #include <sys/stat.h>
    #include <string.h>
    #include <sys/shm.h>
    #include "alg.8-0-shmdata.h"
    int main(int argc, char *argv[])
    {
      void *shmptr = NULL;
      struct shared_struct *shared = NULL;
      int shmid;
      key_t key;
      char buffer[BUFSIZ + 1]; /* 8192bytes, saved from stdin */
      sscanf(argv[1], "%x", &key);
      printf("shmwrite: IPC key = 0x%x\n", key);
      shmid = shmget((key_t)key, TEXT_NUM*sizeof(struct shared_struct), 0666|PERM);
      if (shmid == -1) {
      	ERR_EXIT("shmwite: shmget()");
      }
      shmptr = shmat(shmid, 0, 0);
      if(shmptr == (void *)-1) {
      	ERR_EXIT("shmwrite: shmat()");
      }
      printf("shmwrite: shmid = %d\n", shmid);
      printf("shmwrite: shared memory attached at %p\n", shmptr);
      printf("shmwrite precess ready ...\n");
      shared = (struct shared_struct *)shmptr;
      while (1) {
        while (shared->written == 1) {
          sleep(1); /* message not read yet, waiting ... */
        }
        printf("Enter some text: ");
        fgets(buffer, BUFSIZ, stdin);
        strncpy(shared->mtext, buffer, TEXT_SIZE);
        printf("shared buffer: %s\n",shared->mtext);
        shared->written = 1; /* message prepared */
        if(strncmp(buffer, "end", 3) == 0) {
          break;
        }
      }
      /* detach the shared memory */
      if(shmdt(shmptr) == -1) {
      	ERR_EXIT("shmwrite: shmdt()");
      }
      sleep(1);
      exit(EXIT_SUCCESS);
    }

    ipcs 查看当前操作系统里有多少消息队列、共享内存块、信号量。

    分别编译三个 .c 文件得到三个可执行文件: shmcon,shmread,shmwrite ,然后运行 ./shmcon myshm

    # 哲学家问题

    对称解决方案会导致死锁问题,可以考虑非对称解决方案

    经典的非对称方案:

    • 单号哲学家先拿起左边的筷子,再拿起右边的筷子;而双号的哲学家先拿起右边的筷子,再拿起左边的筷子;

    限制哲学家拿起筷子:

    • 只有哲学家的两根筷子都可用时,才能拿起筷子(哲学家可能饿死)

    # 补充

    被历史抛弃的事务型内存

    Edited on Views times
    \ No newline at end of file +经典进程同步问题 - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    This article isn't finished yet...

    # 严格备选

    (Strict Alternation)

    # PetersonSolution

    # 互斥锁

    互斥锁是一种外部强加的控制方案,或者说是全局控制;

    获得锁:

    伪代码
    acquire {
        while (!available)
        ; /* busy wait */
        available = false;
    }

    释放锁:

    伪代码
    release {
      	available = true;
    }

    利用互斥锁的临界区问题解决方案:

    伪代码
    while (true) {
      	acquire lock
          	/* critical section */
        release lock
          	/* others */
    }

    但是这个方案需要繁忙的等待,所以也叫自旋锁(Spinlock)

    存在优点:当进程在等待锁时,没有上下文切换,在使用锁的时间较短时,自旋锁是有用的;自旋锁通常用于多处理器系统

    # 信号量

    信号量是一种内部主动的控制方案,或者说是局部控制;

    # 管程

    管程(monitor)是一种面向对象的高级程序设计语言结构,它提供的功能与信号量相同,但更易于控制;

    管程结构在很多程序设计语言中得以实现,包括:

    • 并发 Pascal
    • Pascal
    • Pascal-Plus
    • Modula-2
    • Modula-3
    • Java

    现在已经被作为一个程序库实现;

    管程是由一个或多个进程、一个初始化序列和局部数据组成的软件模块;(使用信号的管程)

    c
    monitor monitor name
    {
        /* shared variable declarations */
        function P1 ( ... ) {
        		...
        }
        function P2 ( ... ) {
        		...
        }
        .
        .
        .
        function Pn ( ... ) {
        		...
        }
        initialization code ( ... ) {
        		...
        }
    }

    # 条件变量

    定义附加的同步机制,可以由条件(condition)结构来提供;

    当程序员需要编写定制的同步方案时,可以定义一个或者多个类型为 condition 的变量;

    对于条件变量,只有操作 wait()signal 可以调用;

    利用条件变量实现互斥访问以及控制执行顺序,这里 S1 必须在 S2 之前执行:

    c
    F1(){
        S1;
        		done = true;
        x.signal();
    }
    F2(){
        if done == false
        		x.wait()
        S2;
    }

    # 信号量实现

    c
    // Variables
        semaphore mutex; // (initially = 1)
        semaphore next; // (initially = 0)
        int next_count = 0; // number of processes waiting
    // inside the monitor
    // Each function P will be replaced by
        wait(mutex);
            ...
            body of P;
            ...
        if (next_count > 0)
            signal(next)
        else
            signal(mutex);
    // Mutual exclusion within a monitor is ensured

    # 条件变量实现

    c
    //For each condition variable x, we have:
        semaphore x_sem; // (initially = 0)
        int x_count = 0;
    //The operation x.wait() can be implemented as:
        x_count++;
        if (next_count > 0)
            signal(next);
        else
            signal(mutex);
        wait(x_sem);
        x_count--;

    其中 signal 方法实现参考:

    c
    if (x_count > 0) {
        next_count++;
        signal(x_sem);
        wait(next);
        next_count--;
    }

    # 进程恢复

    • 唤醒并等待(signal and wait)
      • 进程 P 等待进程直到进程 Q 离开管程,或者等待另一个条件;
      • 等待 Q 的时候若 P 想再次执行,需要测试 P 是否还具有执行的环境条件
    • 唤醒并继续(signal and continue)
      • 进程 Q 等待直到进程 P 离开管程或者等待另一个条件;

    # java

    java
    public class BlockedQueue<T> (
        final Lock lock = new ReentrantLock () ;
        // 条件变量:队列不满
        final Condition notFull = lock.newCondition () ;
        // 条件变量:队列不空
        final Condition notEmpty = lock.newCondition ();
        // 入队
        void enq (T x) {
            lock. lock() ;
            try {
                whileStack已满){
                    // 等待队列不满
                    notFull.await () ;
                }
                // 省略入队操作
                // 入队后,通知可出队
                notEmpty.signal ();
            } finally {
            		lock.unlock () ;
            }
        }
        // 出队
        void deq() {
            lock.lock () ;
            try {
                while(队列已空){
                // 等待队列不空
                notEmpty.await () ;
                }
                // 省略出队操作.
                // 出队后,通知可入队
                notFull.signal () ;
            } finally {
            		lock.unlock () ;
        }
    }

    # 管程模型 (补充)

    Hasen、Hoare 和 MESA 模型区别

    Hasen 模型、Hoare 模型和 MESA 模型的一个核心区别就是当条件满足后,如何通知相关线程。管程要求同一时刻只允许一个线程执行,那当线程 T2 的操作使线程 T1 等待的条件满足时,T1 和 T2 究竟谁可以执行呢?

    • Hasen 模型里面,要求 notify() 放在代码的最后,这样 T2 通知完 T1 后,T2 就结束了,然后 T1 再执行,这样就能保证同一时刻只有一个线程执行。
    • Hoare 模型里面,T2 通知完 T1 后,T2 阻塞,T1 马上执行;等 T1 执行完,再唤醒 T2,也能保证同一时刻只有一个线程执行。但是相比 Hasen 模型,T2 多了一次阻塞唤醒操作。
    • MESA 管程里面,T2 通知完 T1 后,T2 还是会接着执行,T1 并不立即执行,仅仅是从条件变量的等待队列进到入口等待队列里面。这样做的好处是 notify() 不用放到代码的最后,T2 也没有多余的阻塞唤醒操作。但是也有个副作用,就是当 T1 再次执行的时候,可能曾经满足的条件,现在已经不满足了,所以需要以循环方式检验条件变量。

    对于 MESA 管程来说,有一个编程范式,就是需要在一个 while 循环里面调用 wait() 。这个是 MESA 管程特有的!

    # 生产者 - 消费者问题

    # 信号量实现

    producer
    while (true) {
        ...
        /* produce an item in next produced */
        ...
        wait(empty);
        wait(mutex);  // 对应的 signal 不是本进程的,而一定是 consumer 进程的 signal !
        ...
        /* add next produced to the buffer */
        ...
        signal(mutex);  // 对应的 wait 不是本进程的,而一定是 consumer 进程的 wait !
        signal(full);
    }
    consumer
    while (true) {
        wait(full);
        wait(mutex);  // 对应的 signal 不是本进程的,而一定是 producer 进程的 signal !
        ...
        /* remove an item from buffer to next consumed */
        ...
        signal(mutex);  // 对应的 wait 不是本进程的,而一定是 producer 进程的 wait !
        signal(empty);
        ...
        /* consume the item in next consumed */
        ...
    }

    # 错误案例

    c
    int n;
    binary_semaphore s = 1, delay = 0;
    void producer(){
        while (true) {
            produce();
            semWaitB(s);
            append();
            n++;
            if (n==1) semSignalB(delay);
            semSignalB(s);
        }
    }
    void consumer(){
        semWaitB(delay);
        while (true) {
            semWaitB(s);
            take();
            n--;
            semSignalB(s);
            consume();  /* 这里没有互斥,producer 打断则会导致后续 n 出现负值 */
            if (n==0) semWaitB(delay);
        }
    }
    void main(){
        n = 0;
        parbegin (producer, consumer);
    }

    consumer 的正确实现:

    c
    void consumer (){
        int m; /* a local variable */
      	semWaitB(delay) ;
        while (true) {
            semWaitB(s);
            take ( );
            n--;
            m = ni
            semSignalB(s) ;
            consume () ;
            if (m==0) semWaitB(delay);
        }
    }

    # 无限队列生产者消费者问题实现

    缓冲区无大小限制

    c
    /* program producerconsumer */
    semaphore n = 0, s = 1;
    void producer(){
        while (true) {
            produce();
            semWait(s);
            append();
            semSignal(s);
            semSignal(n);
        }
    }
    void consumer(){
        while (true) {
            semWait(n);
            semWait(s);
            take();
            semSignal(s);
            consume();
        }
    }
    void main(){
    		parbegin (producer, consumer);
    }

    # 管程实现

    c
    void append (char x){
        while (count == N) wait(notfull);   /* buffer is full; avoid overflow */
        buffer [nextin] = x;
        nextin = (nextin + 1) % N;
        count++                             /* one more item in buffer */
        cnotify (notempty) ;                /* notify any waiting consumer */
    }
    void take (char x){
        while (count == 0) wait(notempty) ; /* buffer is empty; avoid underflow */
        x = buffer[nextout];
        nextout = (nextout + 1) % N;
        count --;                           /* one fewer item in buffer */
      	cnotify(notfull) ;                  /* notify any waiting producer */
    }

    # 共享内存方式 (pthread)

    利用 POSIX API (c 的 pthread) 实现共享内存。The API is supported by Linux 2.4 and later, FreeBSD, … 编译时需要: gcc -lrt filename.c

    shmpthreadcon.c

    c
    #include <stdio.h>
    #include <stdlib.h>
    #include <unistd.h>
    #include <sys/stat.h>
    #include <sys/wait.h>
    #include <fcntl.h>
    #include <sys/mman.h>
    #include "shmdata.h"
    /* gcc -lrt */
    int main(int argc, char *argv[])
    {
      char pathname[80], cmd_str[80];
      struct stat fileattr;
      int fd, shmsize, ret;
      pid_t childpid1, childpid2;
      if(argc < 2) {
        printf("Usage: ./a.out filename\n");
        return EXIT_FAILURE;
      }
      fd = shm_open(argv[1], O_CREAT|O_RDWR, 0666);
      /* /dev/shm/filename as the shared object, creating if not exist */
      if(fd == -1) {
      	ERR_EXIT("con: shm_open()");
      }
      system("ls -l /dev/shm/");
      shmsize = TEXT_NUM*sizeof(struct shared_struct);
      ret = ftruncate(fd, shmsize);
      if(ret == -1) {
        ERR_EXIT"con: ftruncate()");
        char *argv1[] = {" ", argv[1], 0};
        childpid1 = vfork();
        if(childpid1 < 0) {
        	ERR_EXIT("shmpthreadcon: 1st vfork()");
        }
        else if(childpid1 == 0) {
        	execv("./shmproducer", argv1); /* call producer with filename */
        }
        else {
        	childpid2 = vfork();
        if(childpid2 < 0)
        	ERR_EXIT("shmpthreadcon: 2nd vfork()");
        else if (childpid2 == 0)
        	execv("shmconsumer.o", argv1); /* call consumer with filename */
        else {
          wait(&childpid1);
          wait(&childpid2);
          ret = shm_unlink(argv[1]);
          if(ret == -1) {
          	ERR_EXIT("con: shm_unlink()");
          } /* shared object can be removed by any process knew the filename */
          system("ls -l /dev/shm/");
        }
      }
    	exit(EXIT_SUCCESS);
    }

    # 读者 - 写者问题

    一个数据集在多个并发进程之间共享

    • Readers—— 仅读取数据集;它们不执行任何更新
    • Writer—— 既能读又能写

    🤔问题:允许多个读者同时读取

    • 只有一个写者程序可以同时访问共享数据;
    • 写者在访问数据时,不允许读者访问数据;

    读者和作者的考虑方式有几种不同 —— 都涉及某种形式的

    优先权

    • 第一读者写者问题:读者优先;会导致写者饥饿
    • 第二读者写者问题:写者优先;会导致读者饥饿(相对复杂一些)

    # 信号量实现读者优先

    c
    int readcount;
    semaphore x = 1, wsem = 1;
    void reader (){
        while (true) {
            semWait (x);
            readcount++;
            if (readcount == 1) semWait(wsem);
          	semSignal(x);
            READUNIT();
            semWait(x);
            readcount--;
            if (readcount == 0) semSignal(wsem);
          	semSignal (x);
        }
    }
    void writer (){
        while (true) {
            semWait (wsem) ;
            WRITEUNIT () ;
            semSignal (wsem) ;
        }
    }
    void main (){
        readcount = 0;
        parbegin (reader, writer);
    }

    # 单读者 - 单写者

    Single-writer-single-reader problem

    # 共享内存方式

    利用 Linux 的接口实现

    shmdata.h

    c
    #define TEXT_SIZE 4*1024 /* = PAGE_SIZE, size of each message */
    #define TEXT_NUM 1 /* maximal number of messages */
    /* total size can not exceed current shmmax, or an 'invalid argument' error occurs when shmget */
    #define PERM S_IRUSR|S_IWUSR|IPC_CREAT
    #define ERR_EXIT(m) \
      do { \
        perror(m); \
        exit(EXIT_FAILURE); \
      } while(0)
    /* a demo structure, modified as needed */
    struct shared_struct {
    	int written; /* flag = 0: buffer writable; others: readable */
    	char mtext[TEXT_SIZE]; /* buffer for message reading and writing */
    };

    shmcon.c

    c
    #include <stdio.h>
    #include <stdlib.h>
    #include <unistd.h>
    #include <sys/stat.h>
    #include <sys/wait.h>
    #include <sys/shm.h>
    #include <fcntl.h>
    #include "alg.8-0-shmdata.h"
    int main(int argc, char *argv[]){
      struct stat fileattr;
      key_t key; /* of type int */
      int shmid; /* shared memory ID */
      void *shmptr;
      struct shared_struct *shared; /* structured shm */
      pid_t childpid1, childpid2;
      char pathname[80], key_str[10], cmd_str[80];
      int shmsize, ret;
      shmsize = TEXT_NUM*sizeof(struct shared_struct);
      printf("max record number = %d, shm size = %d\n", TEXT_NUM, shmsize);
      if(argc <2) {
        printf("Usage: ./a.out pathname\n");
        return EXIT_FAILURE;
      }
      strcpy(pathname, argv[1]);
      if(stat(pathname, &fileattr) == -1) {
          ret = creat(pathname, O_RDWR);
        if (ret == -1) {
          ERR_EXIT("creat()");
        }
        printf("shared file object created\n");
      }
      key = ftok(pathname, 0x27); /* 0x27 a pro_id 0x0001 - 0xffff, 8 least bits used */
      if(key == -1) {
      	ERR_EXIT("shmcon: ftok()");
      }
      printf("key generated: IPC key = 0x%x\n", key); /* set any key>0 without ftok() */
      shmid = shmget((key_t)key, shmsize, 0666|PERM);
      if(shmid == -1) {
      	ERR_EXIT("shmcon: shmget()");
      }
      printf("shmcon: shmid = %d\n", shmid);
      shmptr = shmat(shmid, 0, 0); /* returns the virtual base address mapping to the
      shared memory, *shmaddr=0 decided by kernel */
      if(shmptr == (void *)-1) {
      	ERR_EXIT("shmcon: shmat()");
      }
      printf("shmcon: shared Memory attached at %p\n", shmptr);
      shared = (struct shared_struct *)shmptr;
      shared->written = 0;
      sprintf(cmd_str, "ipcs -m | grep '%d'\n", shmid);
      printf("\n------ Shared Memory Segments ------\n");
      system(cmd_str);
      if(shmdt(shmptr) == -1) {
      	ERR_EXIT("shmcon: shmdt()");
      }
      printf("\n------ Shared Memory Segments ------\n");
      system(cmd_str);
      sprintf(key_str, "%x", key);
      char *argv1[] = {" ", key_str, 0};
      childpid1 = vfork();
      if(childpid1 < 0) {
      	ERR_EXIT("shmcon: 1st vfork()");
      }
      else if(childpid1 == 0) {
      	execv("./alg.8-2-shmread.o", argv1); /* call shm_read with IPC key */
      }
      else {
      	childpid2 = vfork();
        if(childpid2 < 0) {
          ERR_EXIT("shmcon: 2nd vfork()");
        }
        else if (childpid2 == 0) {
          execv("./alg.8-3-shmwrite.o", argv1); /* call shmwrite with IPC key */
        } else {
          wait(&childpid1);
          wait(&childpid2);
          /* shmid can be removed by any process known the
          IPC key */
          if (shmctl(shmid, IPC_RMID, 0) == -1) {
            ERR_EXIT("shmcon: shmctl(IPC_RMID)");
          } else {
            printf("shmcon: shmid = %d removed \n", shmid);
            printf("\n------ Shared Memory Segments ------\n");
            system(cmd_str);
            printf("nothing found ...\n");
            return EXIT_SUCCESS;
          }
        }
      }
    }

    shmread.c

    c
    #include <stdio.h>
    #include <stdlib.h>
    #include <unistd.h>
    #include <sys/stat.h>
    #include <string.h>
    #include <sys/shm.h>
    #include "alg.8-0-shmdata.h"
    int main(int argc, char *argv[])
    {
      void *shmptr = NULL;
      struct shared_struct *shared;
      int shmid;
      key_t key;
      sscanf(argv[1], "%x", &key);
      printf("%*sshmread: IPC key = 0x%x\n", 30, " ", key);
      shmid = shmget((key_t)key, TEXT_NUM*sizeof(struct shared_struct), 0666|PERM);
      if (shmid == -1) {
        ERR_EXIT("shread: shmget()");
      }
      shmptr = shmat(shmid, 0, 0);
      if(shmptr == (void *)-1) {
      	ERR_EXIT("shread: shmat()");
      }
      printf("%*sshmread: shmid = %d\n", 30, " ", shmid);
      printf("%*sshmread: shared memory attached at %p\n", 30, " ", shmptr);
      printf("%*sshmread process ready ...\n", 30, " ");
      shared = (struct shared_struct *)shmptr;
      while (1) {
        while (shared->written == 0) {
          sleep(1); /* message not ready, waiting ... */
        }
        printf("%*sYou wrote: %s\n", 30, " ", shared->mtext);
        shared->written = 0;
        if (strncmp(shared->mtext, "end", 3) == 0) {
          break;
        }
      } /* it is not reliable to use shared->written for process synchronization */
      if (shmdt(shmptr) == -1) {
      	ERR_EXIT("shmread: shmdt()");
      }
      sleep(1);
      exit(EXIT_SUCCESS);
    }

    shmwrite.c

    c
    #include <stdio.h>
    #include <stdlib.h>
    #include <unistd.h>
    #include <sys/stat.h>
    #include <string.h>
    #include <sys/shm.h>
    #include "alg.8-0-shmdata.h"
    int main(int argc, char *argv[])
    {
      void *shmptr = NULL;
      struct shared_struct *shared = NULL;
      int shmid;
      key_t key;
      char buffer[BUFSIZ + 1]; /* 8192bytes, saved from stdin */
      sscanf(argv[1], "%x", &key);
      printf("shmwrite: IPC key = 0x%x\n", key);
      shmid = shmget((key_t)key, TEXT_NUM*sizeof(struct shared_struct), 0666|PERM);
      if (shmid == -1) {
      	ERR_EXIT("shmwite: shmget()");
      }
      shmptr = shmat(shmid, 0, 0);
      if(shmptr == (void *)-1) {
      	ERR_EXIT("shmwrite: shmat()");
      }
      printf("shmwrite: shmid = %d\n", shmid);
      printf("shmwrite: shared memory attached at %p\n", shmptr);
      printf("shmwrite precess ready ...\n");
      shared = (struct shared_struct *)shmptr;
      while (1) {
        while (shared->written == 1) {
          sleep(1); /* message not read yet, waiting ... */
        }
        printf("Enter some text: ");
        fgets(buffer, BUFSIZ, stdin);
        strncpy(shared->mtext, buffer, TEXT_SIZE);
        printf("shared buffer: %s\n",shared->mtext);
        shared->written = 1; /* message prepared */
        if(strncmp(buffer, "end", 3) == 0) {
          break;
        }
      }
      /* detach the shared memory */
      if(shmdt(shmptr) == -1) {
      	ERR_EXIT("shmwrite: shmdt()");
      }
      sleep(1);
      exit(EXIT_SUCCESS);
    }

    ipcs 查看当前操作系统里有多少消息队列、共享内存块、信号量。

    分别编译三个 .c 文件得到三个可执行文件: shmcon,shmread,shmwrite ,然后运行 ./shmcon myshm

    # 哲学家问题

    对称解决方案会导致死锁问题,可以考虑非对称解决方案

    经典的非对称方案:

    • 单号哲学家先拿起左边的筷子,再拿起右边的筷子;而双号的哲学家先拿起右边的筷子,再拿起左边的筷子;

    限制哲学家拿起筷子:

    • 只有哲学家的两根筷子都可用时,才能拿起筷子(哲学家可能饿死)

    # 补充

    被历史抛弃的事务型内存

    Edited on Views times
    \ No newline at end of file diff --git a/cs/os/thread/index.html b/cs/os/thread/index.html index 014bcd18..7027bc8b 100644 --- a/cs/os/thread/index.html +++ b/cs/os/thread/index.html @@ -1 +1 @@ -线程 - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 线程的基本概念

    # 为什么需要线程

    有的进程可能需要 “同时” 做很多事,而传统的进程只能串行地执行一系列程序。为此,引入了 “线程”,来增加并发度。

    大多数现代应用程序都是多线程的,线程在应用程序中运行,应用程序的多个任务可以由单独的线程实现

    如 Web 浏览器:

    • 更新显示
    • 获取数据
    • 拼写检查
    • 响应网络请求

    进程创建很重载,而线程创建很轻量

    可以简化代码,提高效率;

    内核 (作为内核程序进程) 通常是多线程的,多个线程在内核中运行,每个线程执行一个特定的任务;

    现代服务器 —— 多线程服务器,可以同时处理多个并发请求

    # 线程进程对比

    系统资源分配与处理器调度

    引入线程后,进程只作为除 CPU 之外的系统资源 的 基本分配单元 (最小单位)(如打印机、内存地址空间等都是分配给进程的);CPU 的 分配单元 则是线程


    并发性

    引入线程之后,不仅是进程之间可以并发,进程内的各线程之间也可以并发,从而进一步提升了系统的并发度,使得一个进程内也可以并发处理各种任务。


    系统开销

    • 传统的进程间并发,需要切换进程的运行环境,系统开销很大
    • 线程间并发,如果是同一进程内的线程切换,则不需要切换进程环境,系统开销小

    独立性

    某个进程中的线程对其他进程是不可见的

    每个进程都拥有独立的地址空间和资源,除了共享全局变量不允许其他进程访问


    支持多处理器系统

    • 传统单线程进程不管系统有多少 CPU ,进程只能运行在一个 CPU 上
    • 对于多线程进程,进程多个线程可以分配到多个 CPU 上执行

    # 线程的组成

    线程也有叫做轻量级进程(LWP),线程只包含运行时的状态

    • (用户空间)静态部分由进程提供,与其他线程共享:
      • 代码段 text
      • 数据段 data
      • 堆 heap
      • 其他操作系统资源
    • (内核空间)TCB—— 包括了执行所需的最小状态 (主要是寄存器和栈);
      • TID
      • 程序计数器 PC
      • 寄存器
      • 栈 stack(的指针)(每个线程拥有自己的栈,内核中也有为线程准备的内核栈)
      • 指向父进程 / 线程的指针
      • 指向子进程 / 线程的指针

    一个进程可以包含多个线程

    • 每个线程共享同一地址空间 (方便数据共享和交互);
    • 允许进程内并行;

    # 单线程

    早期操作系统多为单线程进程的结构,实际上也没有形成线程的概念。例如 MS-DOS

    单线程进程

    # 多线程

    多线程是指操作系统在单个进程内支持多个并发执行路径的能力。

    多线程进程

    多线程的优点:

    • 响应性 —— 如果部分线程被阻塞,仍可以继续执行,这对于 用户界面设计尤其重要,快速响应;
    • 资源共享 —— 线程共享进程资源,比进程共享内存或消息传 递更容易,允许一个应用程序在同一地址空间内有多个不 同的活跃线程;
    • 经济 —— 比进程创建更便宜,线程切换比上下文切换开销更 低。进程创建是线程创建的几十倍,进程切换比线程切换 慢 5 倍;
    • 可扩展性 (可伸缩性)—— 线程可以利用多核体系结构;

    # 多核并行

    单核下只能并发执行的任务,现在可以通过多核(多线程)并行执行

    # 两类并行

    • 数据并行性 —— 将相同数据的子集分布在多个核上,每个核上的操作相同;

      数据并行

    • 任务并行性 —— 将任务而不是数据分配到多个计算核心, 每个线程执行唯一的操作,不同线程可以操作相同的数据,也可以操作不同的数据;

      任务并行

    OpenAI 采用的就是数据并行、任务并行混合使用

    # Amdahl 定律

    N 条并行,串行占比 S :

    speedup1S+(1S)N+speedup\leq\frac{1}{S+\frac{(1-S)}{N}+\cdots}

    定律是否考虑了当代多核系统?—— 分母部分会增加 “通信项”,代表核间通信的开销

    # TCB

    Linux 中进程与线程使用的是同一种数据结构 task_struct ,大多数操作系统也都是线程进程共享这样的数据结构

    # 线程的实现方式

    “用户级线程” 就是 “从用户视角看能看到的线程”,早期操作系统只有用户级线程

    “内核级线程” 就是 “从操作系统内核视角看能看到的线程” ,现代操作系统支持内核级线程

    # 用户级线程

    User-Level Thread,ULT

    用户级线程

    1. 用户级线程由应用程序通过线程库(例如 pthreads )实现,所有的线程管理工作都由应用程序负责(包括线程切换)
    2. 用户级线程中,线程切换可以在用户态下即可完成,无需操作系统干预。
    3. 在用户看来,是有多个线程。但是从内核、操作系统的角度来看,并意识不到线程的存在。

    线程不存在挂起态,因为线程不拥有资源!进程才拥有资源,才有 “挂起态”。

    三状态线程模型:

    三状态线程模型

    优缺点

    • 优点:

      • 用户级线程的切换在用户空间即可完成,不需要切换到核心态,线程管理的系统开销小效率高
      • 可以由用户程序自己的调度算法来调度线程
      • 跨平台,在任何操作系统都可以运行,可迁移性强
    • 缺点:

      • 当一个用户级线程被阻塞后,整个进程都会被阻塞,并发度不高。多个线程不可在多核处理机上并行运行。

        • 没有内核线程在阻塞时提供并行能力
        • 解决方案:
          • 套管(Jacketing)把一个产生阻塞的系统调用转化为一个非阻塞的系统调用;
          • 把应用程序写成一个多进程程序而非多线程程序,每次切换都变成进程间的切换而非线程间切换,例如 Python
      • 在纯 ULT 策略中,多线程应用程序不能利用多处理技术。内核一次只把一个进程分配给一个处理器,因此一个进程中只有一个线程可以执行。

    # 内核级线程

    Kernel-Level Thread, KLT,又称 “内核支持的线程”

    1. 内核级线程管理工作由操作系统内核完成。
    2. 线程调度、切换等工作都由内核负责,因此内核级线程的切换必然需要在核心态下才能完成。
    3. 操作系统会为每个内核级线程建立相应的 TCB (Thread Control Block, 线程控制块),通过 TCB 对线程进行管理。
    4. 应用级没有线程管理代码,只有一个内核线程设施的应用编程接口(API),例如 Windows

    优缺点

    • 优点:

      • 内核可以同时把同一个进程中的多个线程调度到多个处理器;
      • 当一个线程被阻塞后,别的线程还可以继续执行,并发能力强。多线程可在多核处理机上并行执行。
      • 内核线程自身也可以是多线程
    • 缺点:

      • 一个用户进程会占用多个内核级线程,线程切换由操作系统内核完成,需要切换到核心态,因此线程管理能成本高开销大

    # 多线程模型

    在支持内核级线程的系统中,根据用户级线程和内核级线程的映射关系,可以划分为几种多线程模型

    # 多对一模型

    个人理解就是只支持用户级线程,而不支持内核级线程的情况,早期操作系统的类型

    多个用户级线程映射到一个内核级线程。且一个进程只被分配一个内核级线程。

    例如 UNIX

    多对一模型

    • 优点:
      • 用户级线程的切换在用户空间即可完成,不需要切换到核心态,线程管理的系统开销小,效率高
    • 缺点:
      • 当一个用户级线程被阻塞后,整个进程都会被阻塞
      • 并发度不高。多个线程不可在多核处理机上并行运行

    操作系统只 “看得见” 内核级线程,因此只有内核级线程才是处理机分配的单位。

    # 一对一模型

    一个用户级线程映射到一个内核级线程。每个用户进程有与用户级线程同数量的内核级线程。

    例如 Linux 、Windows

    一对一模型

    • 优点:
      • 当一个线程被阻塞后,别的线程还可以继续执行
      • 并发能力强。多线程可在多核处理机上并行执行
    • 缺点:
      • 一个用户进程会占用多个内核级线程, 线程切换由操作系统内核完成,需要切换到核心态,开销大
      • 由于内核存储空间的限制,无法创建太多的线程
      • 每创建一个用户线程就要创建一个内核线程,一个用户进程可能占用很多内核线程,开销大

    # 多对多模型

    ULT 和 KLT 混合的模式,例如 Solaris 。实际上非常复杂,目前没有操作系统能够很好地实现这种模型

    多对多模型:n 用户及线程映射到 m 个内核级线程(nmn\geq m)。每个用户进程对应 m 个内核级线程。

    多对多模型

    • 克服了多对一模型并发度不高的缺点(一个阻塞全体阻塞)
    • 克服了一对一模型中一个用户进程占用太多内核级线程,开销太大的缺点

    可以这么理解:

    • 用户级线程是 “代码逻辑” 的载体
    • 内核级线程是 “运行机会” 的载体

    内核级线程中可以运行任意一个有映射关系的用户级线程代码,一个进程只有所有与其有映射关系的内核级线程中正在运行的代码逻辑都阻塞时,这个进程才会阻塞

    • 优点:
      • 可以创建很多线程
      • 并发度高
    • 缺点:
      • 复杂、难以管理、难实现

    # 一对多模型?

    一个线程对应多个进程

    执行过程中发生状态的迁移,就需要将线程从一个进程迁移到另一个进程

    例如银行系统,一个请求会多个有一模一样的进程处理,如果其中一个出现了错误,就会将其他正确的进程迁移过去

    # 现代多线程

    • 异步线程 Asynchronous threading

      父线程创建子线程后继续运行,父子线程并发、独立运行。

    • 同步线程 Synchronous threading

      父线程创建子线程后等待子线程结束后再继续运行,采用 Fork-join strategy ,类似于 Linux 进程中的 fork+wait+exit 。

    # 线程库

    线程库程序员提供了创建和管理线程的 API;

    两种主要的实施方式:

    • 第一种方式:在用户空间中提供一个没有内核支持的库;
    • 第二种方法:实现操作系统支持的内核级库;

    三种主要的线程库:

    • POSIX Pthreads(用户层次或者内核层次);

      • #include<pthread>
    • Windows API(内核层次);

      • 特色: WaitForSingleObject(...) / WaitForMultipleObject(...)
      • 支持线程池
    • Java(利用 Windows API 或者 pthreads 来实现);

      • 由 JVM 管理,通常使用底层操作系统提供的线程模型实现

      • Java 线程可以通过以下方式创建:

        • 扩展线程类
        • 实现可运行接口
        • 标准实践是实现可运行接口
      • 允许围绕 Executor 接口创建线程,典型的线程池

        • static ExecutorService newSingleThreadExecutor()
        • static ExecutorService newFixedThreadPool (int size)
        • static ExecutorService newCachedThreadPool()

    # Linux 线程

    Linux 将它们称为任务(task)而不是线程

    线程创建是通过 clone() 系统调用完成的(Linux 上用 pthread 库的创建线程也是使用这个系统调用)

    clone() 允许子任务共享父任务 (进程) 的地址空间

    标记控件行为...

    结构 task_struct 指向流程数据结构 (共享或唯一)

    # 隐式多线程

    随着线程数量的增加,程序员来控制多线程程序变得越来越困难,于是诞生了由编译器和运行时库而不是程序员创建和管理线程的隐式多线程

    探索了五种方法

    • 线程池

    • Fork-Join

      • 适用于超算、高性能计算场景,利用嵌套创建子线程可以短时间创建数百万子线程
    • OpenMP

      • 让编译器实现程序的多线程并行化
      • 指令指定代码编译为多线程并行程序
        • #pragma omp parallel {...}
        • #pragma omp for ...
    • 中央大调度 (GCD) (eg: Apple...)

      • 目标是自动识别程序的并行部分并实现并行化
      • C、C++ 和 Obj-C 语言、API 和运行库的扩展
    • 英特尔线程构件 (TBB)

      • 设计并行 C++ 程序的模板库
      • parallel_forfor 的并行化版本

    # 多线程引入的问题

    • fork()exec() 系统调用的语义
      • fork() 是只复制调用线程,还是复制该进程的所有线程?
        • 有些 UNIX 有 2 种 fork 版本
      • exec() 通常正常工作,替换正在运行的进程,包括所有线程

    UNIX 中的信号 —— 用于通知进程某个特定事件已发生。

    信号处理器用于处理信号

    1. 信号由特定事件产生

    2. 信号被传送到进程

    3. 信号由两个信号处理程序之一处理:

      • 缺省的信号处理程序
      • 用户定义

    每个信号都有内核在处理信号时运行的默认处理程序

    • 用户定义的信号处理程序可以覆盖默认值
    • 对于单线程,信号传输到进程
    • 信号处理

      • 同步 / 异步
      • 多线程的信号应该在哪里传递?
        • 将信号传递到应用该信号的线程(一对一)
        • 将信号传递给进程中的每个线程(一对多)
        • 将信号传递给进程中的某些线程(一对一)
        • 指定一个特定线程来接收进程的所有信号(多对一)
    • 线程取消(要终止的线程一般叫做目标线程)

      • 异步 —— 立即终止目标线程
      • 延迟 —— 允许目标线程定期 —— 也就是执行到代码中的 “取消点” 的时候 —— 检查是否应该取消,缺省情况下都是这种情况
      • Linux 由信号处理,而且应该是一对一的
      • Java 用 interrupt 方法
    • 线程本地存储

      • 线程本地存储 (TLS) 允许每个线程拥有自己的数据副本,有这些好处:

        • 当无法控制线程创建过程时 (即,当使用线程池时),此功能非常有用
        • 可以避免不同线程之间访问同一互斥变量带来的性能损失
      • 与局部变量不同

        • 局部变量仅在单个函数调用期间可见
        • TLS 在函数调用中可见
      • 类似于静态数据

        • TLS 对于每个线程都是唯一的
    • 调度程序激活

      • 多对多模型和一对一模型都需要通信来维护分配给应用程序的适当数量的内核线程

      • 通常在用户线程和内核线程之间使用中间数据结构 —— 轻量级进程 (LWP, Light Weight Process)

        • 似乎是一个虚拟处理器,进程可以 在该处理器上调度用户线程运行
        • 每个 LWP 都连接到内核线程
        • 要创建多少 LWP?
      • 调度器激活提供了 upcall —— 一种从内核到线程库中 upcall 处理程序的通信机制

      • 这种通信允许应用程序保持正确的内核线程数

    Edited on Views times
    \ No newline at end of file +线程 - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 线程的基本概念

    # 为什么需要线程

    有的进程可能需要 “同时” 做很多事,而传统的进程只能串行地执行一系列程序。为此,引入了 “线程”,来增加并发度。

    大多数现代应用程序都是多线程的,线程在应用程序中运行,应用程序的多个任务可以由单独的线程实现

    如 Web 浏览器:

    • 更新显示
    • 获取数据
    • 拼写检查
    • 响应网络请求

    进程创建很重载,而线程创建很轻量

    可以简化代码,提高效率;

    内核 (作为内核程序进程) 通常是多线程的,多个线程在内核中运行,每个线程执行一个特定的任务;

    现代服务器 —— 多线程服务器,可以同时处理多个并发请求

    # 线程进程对比

    系统资源分配与处理器调度

    引入线程后,进程只作为除 CPU 之外的系统资源 的 基本分配单元 (最小单位)(如打印机、内存地址空间等都是分配给进程的);CPU 的 分配单元 则是线程


    并发性

    引入线程之后,不仅是进程之间可以并发,进程内的各线程之间也可以并发,从而进一步提升了系统的并发度,使得一个进程内也可以并发处理各种任务。


    系统开销

    • 传统的进程间并发,需要切换进程的运行环境,系统开销很大
    • 线程间并发,如果是同一进程内的线程切换,则不需要切换进程环境,系统开销小

    独立性

    某个进程中的线程对其他进程是不可见的

    每个进程都拥有独立的地址空间和资源,除了共享全局变量不允许其他进程访问


    支持多处理器系统

    • 传统单线程进程不管系统有多少 CPU ,进程只能运行在一个 CPU 上
    • 对于多线程进程,进程多个线程可以分配到多个 CPU 上执行

    # 线程的组成

    线程也有叫做轻量级进程(LWP),线程只包含运行时的状态

    • (用户空间)静态部分由进程提供,与其他线程共享:
      • 代码段 text
      • 数据段 data
      • 堆 heap
      • 其他操作系统资源
    • (内核空间)TCB—— 包括了执行所需的最小状态 (主要是寄存器和栈);
      • TID
      • 程序计数器 PC
      • 寄存器
      • 栈 stack(的指针)(每个线程拥有自己的栈,内核中也有为线程准备的内核栈)
      • 指向父进程 / 线程的指针
      • 指向子进程 / 线程的指针

    一个进程可以包含多个线程

    • 每个线程共享同一地址空间 (方便数据共享和交互);
    • 允许进程内并行;

    # 单线程

    早期操作系统多为单线程进程的结构,实际上也没有形成线程的概念。例如 MS-DOS

    单线程进程

    # 多线程

    多线程是指操作系统在单个进程内支持多个并发执行路径的能力。

    多线程进程

    多线程的优点:

    • 响应性 —— 如果部分线程被阻塞,仍可以继续执行,这对于 用户界面设计尤其重要,快速响应;
    • 资源共享 —— 线程共享进程资源,比进程共享内存或消息传 递更容易,允许一个应用程序在同一地址空间内有多个不 同的活跃线程;
    • 经济 —— 比进程创建更便宜,线程切换比上下文切换开销更 低。进程创建是线程创建的几十倍,进程切换比线程切换 慢 5 倍;
    • 可扩展性 (可伸缩性)—— 线程可以利用多核体系结构;

    # 多核并行

    单核下只能并发执行的任务,现在可以通过多核(多线程)并行执行

    # 两类并行

    • 数据并行性 —— 将相同数据的子集分布在多个核上,每个核上的操作相同;

      数据并行

    • 任务并行性 —— 将任务而不是数据分配到多个计算核心, 每个线程执行唯一的操作,不同线程可以操作相同的数据,也可以操作不同的数据;

      任务并行

    OpenAI 采用的就是数据并行、任务并行混合使用

    # Amdahl 定律

    N 条并行,串行占比 S :

    speedup1S+(1S)N+speedup\leq\frac{1}{S+\frac{(1-S)}{N}+\cdots}

    定律是否考虑了当代多核系统?—— 分母部分会增加 “通信项”,代表核间通信的开销

    # TCB

    Linux 中进程与线程使用的是同一种数据结构 task_struct ,大多数操作系统也都是线程进程共享这样的数据结构

    # 线程的实现方式

    “用户级线程” 就是 “从用户视角看能看到的线程”,早期操作系统只有用户级线程

    “内核级线程” 就是 “从操作系统内核视角看能看到的线程” ,现代操作系统支持内核级线程

    # 用户级线程

    User-Level Thread,ULT

    用户级线程

    1. 用户级线程由应用程序通过线程库(例如 pthreads )实现,所有的线程管理工作都由应用程序负责(包括线程切换)
    2. 用户级线程中,线程切换可以在用户态下即可完成,无需操作系统干预。
    3. 在用户看来,是有多个线程。但是从内核、操作系统的角度来看,并意识不到线程的存在。

    线程不存在挂起态,因为线程不拥有资源!进程才拥有资源,才有 “挂起态”。

    三状态线程模型:

    三状态线程模型

    优缺点

    • 优点:

      • 用户级线程的切换在用户空间即可完成,不需要切换到核心态,线程管理的系统开销小效率高
      • 可以由用户程序自己的调度算法来调度线程
      • 跨平台,在任何操作系统都可以运行,可迁移性强
    • 缺点:

      • 当一个用户级线程被阻塞后,整个进程都会被阻塞,并发度不高。多个线程不可在多核处理机上并行运行。

        • 没有内核线程在阻塞时提供并行能力
        • 解决方案:
          • 套管(Jacketing)把一个产生阻塞的系统调用转化为一个非阻塞的系统调用;
          • 把应用程序写成一个多进程程序而非多线程程序,每次切换都变成进程间的切换而非线程间切换,例如 Python
      • 在纯 ULT 策略中,多线程应用程序不能利用多处理技术。内核一次只把一个进程分配给一个处理器,因此一个进程中只有一个线程可以执行。

    # 内核级线程

    Kernel-Level Thread, KLT,又称 “内核支持的线程”

    1. 内核级线程管理工作由操作系统内核完成。
    2. 线程调度、切换等工作都由内核负责,因此内核级线程的切换必然需要在核心态下才能完成。
    3. 操作系统会为每个内核级线程建立相应的 TCB (Thread Control Block, 线程控制块),通过 TCB 对线程进行管理。
    4. 应用级没有线程管理代码,只有一个内核线程设施的应用编程接口(API),例如 Windows

    优缺点

    • 优点:

      • 内核可以同时把同一个进程中的多个线程调度到多个处理器;
      • 当一个线程被阻塞后,别的线程还可以继续执行,并发能力强。多线程可在多核处理机上并行执行。
      • 内核线程自身也可以是多线程
    • 缺点:

      • 一个用户进程会占用多个内核级线程,线程切换由操作系统内核完成,需要切换到核心态,因此线程管理能成本高开销大

    # 多线程模型

    在支持内核级线程的系统中,根据用户级线程和内核级线程的映射关系,可以划分为几种多线程模型

    # 多对一模型

    个人理解就是只支持用户级线程,而不支持内核级线程的情况,早期操作系统的类型

    多个用户级线程映射到一个内核级线程。且一个进程只被分配一个内核级线程。

    例如 UNIX

    多对一模型

    • 优点:
      • 用户级线程的切换在用户空间即可完成,不需要切换到核心态,线程管理的系统开销小,效率高
    • 缺点:
      • 当一个用户级线程被阻塞后,整个进程都会被阻塞
      • 并发度不高。多个线程不可在多核处理机上并行运行

    操作系统只 “看得见” 内核级线程,因此只有内核级线程才是处理机分配的单位。

    # 一对一模型

    一个用户级线程映射到一个内核级线程。每个用户进程有与用户级线程同数量的内核级线程。

    例如 Linux 、Windows

    一对一模型

    • 优点:
      • 当一个线程被阻塞后,别的线程还可以继续执行
      • 并发能力强。多线程可在多核处理机上并行执行
    • 缺点:
      • 一个用户进程会占用多个内核级线程, 线程切换由操作系统内核完成,需要切换到核心态,开销大
      • 由于内核存储空间的限制,无法创建太多的线程
      • 每创建一个用户线程就要创建一个内核线程,一个用户进程可能占用很多内核线程,开销大

    # 多对多模型

    ULT 和 KLT 混合的模式,例如 Solaris 。实际上非常复杂,目前没有操作系统能够很好地实现这种模型

    多对多模型:n 用户及线程映射到 m 个内核级线程(nmn\geq m)。每个用户进程对应 m 个内核级线程。

    多对多模型

    • 克服了多对一模型并发度不高的缺点(一个阻塞全体阻塞)
    • 克服了一对一模型中一个用户进程占用太多内核级线程,开销太大的缺点

    可以这么理解:

    • 用户级线程是 “代码逻辑” 的载体
    • 内核级线程是 “运行机会” 的载体

    内核级线程中可以运行任意一个有映射关系的用户级线程代码,一个进程只有所有与其有映射关系的内核级线程中正在运行的代码逻辑都阻塞时,这个进程才会阻塞

    • 优点:
      • 可以创建很多线程
      • 并发度高
    • 缺点:
      • 复杂、难以管理、难实现

    # 一对多模型?

    一个线程对应多个进程

    执行过程中发生状态的迁移,就需要将线程从一个进程迁移到另一个进程

    例如银行系统,一个请求会多个有一模一样的进程处理,如果其中一个出现了错误,就会将其他正确的进程迁移过去

    # 现代多线程

    • 异步线程 Asynchronous threading

      父线程创建子线程后继续运行,父子线程并发、独立运行。

    • 同步线程 Synchronous threading

      父线程创建子线程后等待子线程结束后再继续运行,采用 Fork-join strategy ,类似于 Linux 进程中的 fork+wait+exit 。

    # 线程库

    线程库程序员提供了创建和管理线程的 API;

    两种主要的实施方式:

    • 第一种方式:在用户空间中提供一个没有内核支持的库;
    • 第二种方法:实现操作系统支持的内核级库;

    三种主要的线程库:

    • POSIX Pthreads(用户层次或者内核层次);

      • #include<pthread>
    • Windows API(内核层次);

      • 特色: WaitForSingleObject(...) / WaitForMultipleObject(...)
      • 支持线程池
    • Java(利用 Windows API 或者 pthreads 来实现);

      • 由 JVM 管理,通常使用底层操作系统提供的线程模型实现

      • Java 线程可以通过以下方式创建:

        • 扩展线程类
        • 实现可运行接口
        • 标准实践是实现可运行接口
      • 允许围绕 Executor 接口创建线程,典型的线程池

        • static ExecutorService newSingleThreadExecutor()
        • static ExecutorService newFixedThreadPool (int size)
        • static ExecutorService newCachedThreadPool()

    # Linux 线程

    Linux 将它们称为任务(task)而不是线程

    线程创建是通过 clone() 系统调用完成的(Linux 上用 pthread 库的创建线程也是使用这个系统调用)

    clone() 允许子任务共享父任务 (进程) 的地址空间

    标记控件行为...

    结构 task_struct 指向流程数据结构 (共享或唯一)

    # 隐式多线程

    随着线程数量的增加,程序员来控制多线程程序变得越来越困难,于是诞生了由编译器和运行时库而不是程序员创建和管理线程的隐式多线程

    探索了五种方法

    • 线程池

    • Fork-Join

      • 适用于超算、高性能计算场景,利用嵌套创建子线程可以短时间创建数百万子线程
    • OpenMP

      • 让编译器实现程序的多线程并行化
      • 指令指定代码编译为多线程并行程序
        • #pragma omp parallel {...}
        • #pragma omp for ...
    • 中央大调度 (GCD) (eg: Apple...)

      • 目标是自动识别程序的并行部分并实现并行化
      • C、C++ 和 Obj-C 语言、API 和运行库的扩展
    • 英特尔线程构件 (TBB)

      • 设计并行 C++ 程序的模板库
      • parallel_forfor 的并行化版本

    # 多线程引入的问题

    • fork()exec() 系统调用的语义
      • fork() 是只复制调用线程,还是复制该进程的所有线程?
        • 有些 UNIX 有 2 种 fork 版本
      • exec() 通常正常工作,替换正在运行的进程,包括所有线程

    UNIX 中的信号 —— 用于通知进程某个特定事件已发生。

    信号处理器用于处理信号

    1. 信号由特定事件产生

    2. 信号被传送到进程

    3. 信号由两个信号处理程序之一处理:

      • 缺省的信号处理程序
      • 用户定义

    每个信号都有内核在处理信号时运行的默认处理程序

    • 用户定义的信号处理程序可以覆盖默认值
    • 对于单线程,信号传输到进程
    • 信号处理

      • 同步 / 异步
      • 多线程的信号应该在哪里传递?
        • 将信号传递到应用该信号的线程(一对一)
        • 将信号传递给进程中的每个线程(一对多)
        • 将信号传递给进程中的某些线程(一对一)
        • 指定一个特定线程来接收进程的所有信号(多对一)
    • 线程取消(要终止的线程一般叫做目标线程)

      • 异步 —— 立即终止目标线程
      • 延迟 —— 允许目标线程定期 —— 也就是执行到代码中的 “取消点” 的时候 —— 检查是否应该取消,缺省情况下都是这种情况
      • Linux 由信号处理,而且应该是一对一的
      • Java 用 interrupt 方法
    • 线程本地存储

      • 线程本地存储 (TLS) 允许每个线程拥有自己的数据副本,有这些好处:

        • 当无法控制线程创建过程时 (即,当使用线程池时),此功能非常有用
        • 可以避免不同线程之间访问同一互斥变量带来的性能损失
      • 与局部变量不同

        • 局部变量仅在单个函数调用期间可见
        • TLS 在函数调用中可见
      • 类似于静态数据

        • TLS 对于每个线程都是唯一的
    • 调度程序激活

      • 多对多模型和一对一模型都需要通信来维护分配给应用程序的适当数量的内核线程

      • 通常在用户线程和内核线程之间使用中间数据结构 —— 轻量级进程 (LWP, Light Weight Process)

        • 似乎是一个虚拟处理器,进程可以 在该处理器上调度用户线程运行
        • 每个 LWP 都连接到内核线程
        • 要创建多少 LWP?
      • 调度器激活提供了 upcall —— 一种从内核到线程库中 upcall 处理程序的通信机制

      • 这种通信允许应用程序保持正确的内核线程数

    Edited on Views times
    \ No newline at end of file diff --git a/cs/os/ysosv2/lab0/index.html b/cs/os/ysosv2/lab0/index.html index 3e1ec436..f2d5653a 100644 --- a/cs/os/ysosv2/lab0/index.html +++ b/cs/os/ysosv2/lab0/index.html @@ -1 +1 @@ -YSOSv2: lab0 实验报告 - YSOSv2 - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party
    Edited on Views times
    \ No newline at end of file +YSOSv2: lab0 实验报告 - YSOSv2 - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party
    Edited on Views times
    \ No newline at end of file diff --git a/cs/os/ysosv2/lab1/index.html b/cs/os/ysosv2/lab1/index.html index acb94927..22f4f3a5 100644 --- a/cs/os/ysosv2/lab1/index.html +++ b/cs/os/ysosv2/lab1/index.html @@ -1 +1 @@ -YSOSv2: lab1 实验报告 - YSOSv2 - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party
    Edited on Views times
    \ No newline at end of file +YSOSv2: lab1 实验报告 - YSOSv2 - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party
    Edited on Views times
    \ No newline at end of file diff --git a/cs/os/ysosv2/lab2/index.html b/cs/os/ysosv2/lab2/index.html index 16d29df1..7054103b 100644 --- a/cs/os/ysosv2/lab2/index.html +++ b/cs/os/ysosv2/lab2/index.html @@ -1 +1 @@ -YSOSv2: lab2 实验报告 - YSOSv2 - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party
    Edited on Views times
    \ No newline at end of file +YSOSv2: lab2 实验报告 - YSOSv2 - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party
    Edited on Views times
    \ No newline at end of file diff --git a/cs/os/ysosv2/lab3/index.html b/cs/os/ysosv2/lab3/index.html index 5fbc77d6..5db36d55 100644 --- a/cs/os/ysosv2/lab3/index.html +++ b/cs/os/ysosv2/lab3/index.html @@ -1 +1 @@ -YSOSv2: lab3 实验报告 - YSOSv2 - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party
    Edited on Views times
    \ No newline at end of file +YSOSv2: lab3 实验报告 - YSOSv2 - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party
    Edited on Views times
    \ No newline at end of file diff --git a/cs/os/ysosv2/lab4/index.html b/cs/os/ysosv2/lab4/index.html index 1a7ce15f..9c067cc3 100644 --- a/cs/os/ysosv2/lab4/index.html +++ b/cs/os/ysosv2/lab4/index.html @@ -1 +1 @@ -YSOSv2: lab4 实验报告 - YSOSv2 - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party
    Edited on Views times
    \ No newline at end of file +YSOSv2: lab4 实验报告 - YSOSv2 - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party
    Edited on Views times
    \ No newline at end of file diff --git a/cs/os/ysosv2/lab5/index.html b/cs/os/ysosv2/lab5/index.html index 39b1a9e9..cbd370e3 100644 --- a/cs/os/ysosv2/lab5/index.html +++ b/cs/os/ysosv2/lab5/index.html @@ -1 +1 @@ -YSOSv2: lab5 实验报告 - YSOSv2 - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party
    Edited on Views times
    \ No newline at end of file +YSOSv2: lab5 实验报告 - YSOSv2 - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party
    Edited on Views times
    \ No newline at end of file diff --git a/cs/os/ysosv2/lab6/index.html b/cs/os/ysosv2/lab6/index.html index 07f00c7a..5fae3e47 100644 --- a/cs/os/ysosv2/lab6/index.html +++ b/cs/os/ysosv2/lab6/index.html @@ -1 +1 @@ -YSOSv2: lab6 实验报告 - YSOSv2 - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party
    Edited on Views times
    \ No newline at end of file +YSOSv2: lab6 实验报告 - YSOSv2 - 操作系统 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party
    Edited on Views times
    \ No newline at end of file diff --git a/cs/parallel-programing/pthread/index.html b/cs/parallel-programing/pthread/index.html index 2c6a5ab8..a7e9fc62 100644 --- a/cs/parallel-programing/pthread/index.html +++ b/cs/parallel-programing/pthread/index.html @@ -1 +1 @@ -线程 - 并行程序设计 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # pthread 库

    # 简介

    pthread 是 POSIX 线程库,它是一种跨平台的线程库,可以在不同的操作系统上使用。它是 C 语言的标准库,提供了一组函数来创建、管理和同步线程。"pthread" 库相对于 "thread" 库来说,更加底层和与操作系统紧密相关。

    thread 是 C11 标准提供的线程库,它是 C 的一部分。它提供了一种面向对象的方式来创建、管理和同步线程。"thread" 库提供了 std::thread 类,可以用于创建和管理线程,并提供了一些成员函数和工具函数来操作线程。

    pthread 相关函数需要引入标准库头文件: #include<pthread.h>

    # 参考资料

    开源教程:https://hpc-tutorials.llnl.gov/posix/

    pthread 库在不同的操作系统上有不同的实现,因此,官方的 pthread 库文档可能由各个操作系统的文档或开发者文档提供。以下是一些常见操作系统的 pthread 官方文档链接:

    • Linux: 在 Linux 系统上,pthread 库的官方文档可以在 GNU C 库(glibc)的文档中找到。您可以访问以下链接获取详细的 pthread 文档:https://www.gnu.org/software/libc/manual/html_node/Threads.html

    • macOS: 在 macOS 上,pthread 库的官方文档可以在 Apple 开发者文档中找到。您可以访问以下链接获取有关 pthread 的详细信息:https://developer.apple.com/library/archive/documentation/Cocoa/Conceptual/Multithreading/Introduction/Introduction.html

    • Windows: 在 Windows 操作系统上,Microsoft 提供了一种名为 "Windows Threads" 的线程库,它是 Windows 对 POSIX 线程的实现。您可以在 Microsoft 的官方文档中找到有关 "Windows Threads" 的详细信息:https://docs.microsoft.com/en-us/windows/win32/procthread/creating-threads

    • IBM:

      • z/OS:

        z/OS V2R4 POSIX Threads Programming: https://www.ibm.com/docs/en/zos/3.1.0?topic=programming-posix-threads-programming

      • IBM i:

        IBM i 7.4 PASE Programming and Reference: https://www.ibm.com/docs/en/i/7.4?topic=programming-pase-programming-reference

      • AIX:

        AIX 7.2 Library Reference - Threads and Concurrency: https://www.ibm.com/docs/en/aix/7.2?topic=library-threads-concurrency

    不同操作系统上的 pthread 库在语法和使用方式上可能会有一些差异。这是因为 pthread 库是根据 POSIX 线程标准实现的,而各个操作系统可能会在标准的基础上进行一些定制和扩展。

    # 多线程程序举例分析

    # 线程间接执行的函数

    c
    float *func1(float a, float b) {
        int i;
        float *result;
        result = (float *)malloc(sizeof(float) * 2);
        for (i = 0; i < 1000000000; ++i) {
            result[0] = a + b;
            result[1] = a - b;
        }
        return result;
    }
    float *func2(float a, float b) {
        int i;
        float *result;
        result = (float *)malloc(sizeof(float) * 2);
        for (i = 0; i < 1000000000; ++i) {
            result[0] = a * b;
            result[1] = a / b;
        }
        return result;
    }

    # 给间接函数传递参数用的结构体

    传递给子线程的将是这个结构体变量的地址,这个地址对父线程和子线程都是一样的,可见的,所以其内容是共享的,互相可以看到对方的修改。

    c
    typedef struct {
        float a;
        float b;
    } Args;

    # 线程直接执行的函数

    只能返回 void 指针类型,即返回 void*

    return 返回的地址内容将被 pthread_join 读出

    c
    void *thread_func1(void *arg) {
        Args *args = (Args *)arg; // 传递参数结构体变量的地址
        return func1(args->a, args->b);
    }
    void *thread_func2(void *arg) {
        Args *args = (Args *)arg; // 传递参数结构体变量的地址
        return func2(args->a, args->b);
    }

    # pthread_exit 结束线程

    除了 return 返回数据,还可以用 pthread_exit 函数返回数据并直接结束线程,而不必等到主线程 pthread_join 连接该线程。提前结束进程可以释放进程占用的内存、处理器,节省计算机资源 (真的能吗?)

    线程终止函数 pthread_exit 传入 1 个参数

    • void * 类型:非 void 类型指针,指向返回数据的地址
    c
    void *thread_func1(void *arg) {
        Args *args = (Args *)arg;
        pthread_exit(func1(args->a, args->b));
    }
    void *thread_func2(void *arg) {
        Args *args = (Args *)arg;
        pthread_exit(func1(args->a, args->b));
    }

    # 线程返回数据存储的有类型地址

    应当是数据的类型

    c
    result1 = (float *)malloc(sizeof(float) * 2);
    result2 = (float *)malloc(sizeof(float) * 2);

    # 线程返回数据直接存储的无类型地址

    只能是 void 指针类型

    c
    void *r1, *r2;

    # 定义 pthread_t 类型变量

    pthread_t 类型变量用于存储线程句柄 (TID) , t 表示数据类型 (type)

    c
    pthread_t tid1, tid2;

    # pthread_create 创建线程

    线程创建函数 pthread_create 传入 4 个参数:

    • pthread_t 类型:新线程的线程句柄 (TID) 要存储的地址 (pthread_t 变量指针)
    • const pthread_attr_t * 类型:属性
    • 执行函数 (只能是 void 类型,即无返回)
    • 传给执行函数的参数 (一次只能传一个参数,所以一般用结构体传参数)
    c
    pthread_create(&tid1, NULL, thread_func1, &ab);
    pthread_create(&tid2, NULL, thread_func2, &ab);

    函数返回结果:代表线程连接的状态,成功返回 0 ,否则返回错误码

    错误码:

    • EAGAIN : The system lacks the necessary resources to create another thread.
    • EINVAL : The value specified by thread is null.
    • ELEMULTITHREADFORK : pthread_create() was invoked from a child process created by calling fork() from a multi-threaded process. This child process is restricted from becoming multi-threaded.
    • ENOMEM : There is not enough memory to create the thread.

    # ⚠️区别进程的创建

    c
    for(i = 0; i < 5; i++)
        pthread_join(tid[i], NULL);

    这里与 Linux 创建进程(线程)的 fork 不同,后者会创建 252^5 个进程(线程),而这里是 5 个线程!因为子线程、父线程是并行的!

    # pthread_join 连接并读出线程执行结果

    线程数据读出函数 pthread_join 传入 2 个参数:

    • pthread_t 类型:线程句柄,即 TID
    • void ** 类型:执行函数的返回数据要存储的地址 (地址只能是 void 类型,一次只能返回一个地址)
    c
    pthread_join(tid1, &r1);
    pthread_join(tid2, &r2);

    函数返回结果:代表线程连接的状态,成功返回 0 ,否则返回错误码

    错误码:

    • EDEADLK : A deadlock was detected (e.g., two threads tried to join with each other); or thread specifies the calling thread.
    • EINVAL : hread is not a joinable thread.
    • EINVAL : Another thread is already waiting to join with this thread.
    • ESRCH : No thread with the ID thread could be found.

    一个线程只能同时连接一个线程,子线程自创建开始执行,若连接子线程时子线程还没结束,主线程会等到子线程执行结束后再读出数据

    Windows 中使用的是 WaitForSingleObject(... , ...) / WaitForMultipleObject(... , ...) ,因为 Windows 将所有资源对象命名为 Object

    # 无类型地址数据存储到有类型地址

    c
    result1 = (float *)r1;
    result2 = (float *)r2;

    # 完整程序

    c
    #include <pthread.h>
    #include <stdio.h>
    #include <stdlib.h>
    #include <sys/time.h>
    float *func1(float a, float b) {
        int i;
        float *result;
        result = (float *)malloc(sizeof(float) * 2);
        for (i = 0; i < 1000000000; ++i) {
            result[0] = a + b;
            result[1] = a - b;
        }
        return result;
    }
    float *func2(float a, float b) {
        int i;
        float *result;
        result = (float *)malloc(sizeof(float) * 2);
        for (i = 0; i < 1000000000; ++i) {
            result[0] = a * b;
            result[1] = a / b;
        }
        return result;
    }
    typedef struct {
        float a;
        float b;
    } Args;
    void *thread_func1(void *arg) {
        Args *args = (Args *)arg;
        return func1(args->a, args->b);
    }
    void *thread_func2(void *arg) {
        Args *args = (Args *)arg;
        return func1(args->a, args->b);
    }
    int main() {
        float a = 1.14, b = 5.14, *result1, *result2, during;
        Args ab;
        ab.a = a;
        ab.b = b;
        result1 = (float *)malloc(sizeof(float) * 2);
        result2 = (float *)malloc(sizeof(float) * 2);
        struct timeval start, end;
        gettimeofday(&start, NULL);
        void *r1, *r2;
        pthread_t tid1, tid2;
        pthread_create(&tid1, NULL, thread_func1, &ab);
        pthread_create(&tid2, NULL, thread_func2, &ab);
        pthread_join(tid1, &r1);
        pthread_join(tid2, &r2);
        result1 = (float *)r1;
        result2 = (float *)r2;
        gettimeofday(&end, NULL);
        during = (end.tv_sec - start.tv_sec) + (end.tv_usec - start.tv_usec) / 1000000.0;
        printf("result1[0] = %f\nresult1[1] = %f\n", result1[0], result1[1]);
        printf("result2[0] = %f\nresult2[1] = %f\n", result2[0], result2[1]);
        printf("during time = %f s\n", during);
        free(result1);
        free(result2);
        return 0;
    }

    # 线程间互动

    # 互斥锁 pthread_mutex

    对应数据类型 (type) 为 pthred_mutex_t ,以及相关函数 pthread_mutex_function

    互斥锁的作用主要是每个线程都可以在进入某个区域时 把这个区域锁起来, 不让其他的线程进入。 当自己的工作完成时或者达成某种条件时,则打开互斥锁。互斥锁存在的目的就是为了避免很多线程同时进入一个事件时发生冲突。

    # pthread_cond 条件变量

    对应数据类型 (type) 为 pthred_cond_t ,以及相关函数 pthread_cond_function

    条件变量主要和互斥锁一同使用,也就是当满足某一个条件时打开互斥锁。这个条件可以通过 pthread_cond_signal 或者是 pthread_cond_broadcast 发出

    # pthread_cond_signal

    • 一次只能激活一个条件变量
    • 不容易出现 “线程打架”

    # pthread_cond_broadcast

    • 一次性激活所有等待的条件变量
    • 容易出现 “线程打架”,造成 “死锁”

    # 线程互动程序举例分析

    # 初始化锁 pthread_mutex_t

    c
    pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;

    # 初始化条件变量 pthread_cond_t

    c
    pthread_cond_t cond_catchfish_finish = PTHREAD_COND_INITIALIZER;
    pthread_cond_t cond_eatfish_finish = PTHREAD_COND_INITIALIZER;

    # 线程函数

    # pthread_mutex_lock 上锁

    c
    pthread_mutex_lock (&mutex) ;

    上锁之后线程将无法收到来自其他线程的信号

    # 输入

    c
    int pthread_mutex_lock(pthread_mutex_t *mutex);
    //SUSv3:
    int pthread_mutex_lock(pthread_mutex_t *mutex);

    # 返回值

    • If successful, pthread_mutex_lock() returns 0.

    • If unsuccessful, pthread_mutex_lock() returns -1 and sets errno to one of the following values:

      • Error Code

        • Description*
      • EAGAIN

        • The mutex could not be acquired because the maximum number of recursive locks for mutex has been exceeded. This errno will only occur in the shared path.
      • EDEADLK

        • The current thread already owns the mutex, and the mutex has a kind attribute of __MUTEX_NONRECURSIVE.
      • EINVAL

        • The value specified by mutex is not valid.

    Special behavior for Single UNIX Specification, Version 3: If unsuccessful, pthread_mutex_lock () returns an error number to indicate the error. SUSv3 操作系统可以直接返回错误码

    # pthred_cond_wait 使线程进入等待状态

    c
    pthread_cond_wait(&cond_eatfish_finish, &mutex);

    会原子地释放之前获取的互斥锁,使当前线程进入等待状态,并等待条件变量满足特定的条件

    # 输入

    c
    int pthread_cond_wait(pthread_cond_t *cond,
                          pthread_mutex_t *mutex);
    //SUSv3:
    int pthread_cond_wait(pthread_cond_t * __restrict__cond, 
                          pthread_mutex_t * __restrict__mutex);

    # 返回值

    • If successful, pthread_cond_wait() returns 0.

    • If unsuccessful, pthread_cond_wait() returns -1 and sets errno to one of the following values:

      • Error Code

        • Description
      • EINVAL

        • Different mutexes were specified for concurrent operations on the same condition variable.
      • EPERM

        • The mutex was not owned by the current thread at the time of the call.

    Special behavior for Single UNIX Specification, Version 3: If unsuccessful, pthread_cond_wait () returns an error number to indicate the error. SUSv3 操作系统可以直接返回错误码

    # 完整程序

    c
    #include<stdio.h>
    #include<stdlib.h>
    #include<pthread.h>
    int countFish = 0;// 计数器,记录容器内鱼的数量
    const int numFish = 10;// 一筐鱼的数量
    pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;// 初始化一把叫做 mutex 的锁
    pthread_cond_t cond_catchfish_finish = PTHREAD_COND_INITIALIZER;// 初始化一个条件变量,这个条件变量为捕鱼操作是否完成
    pthread_cond_t cond_eatfish_finish = PTHREAD_COND_INITIALIZER;// 初始化一个条件变量,这个条件变量为吃鱼操作是否完成
    void* fishman (){
        int i, j, k;
        for(i= 0;i < numFish * 2; i++){
            pthread_mutex_lock (&mutex) ;
            // 渔夫在捕鱼时,先加一把锁,防止猫咪把容器抢走
            while (countFish == numFish){
                pthread_cond_wait(&cond_eatfish_finish, &mutex);
                // 如果容器中鱼满了,渔夫就等待猫咪吃完鱼,然后再继续捕鱼
                // 当没有接收到 cond_eatfish_finish 信号时,就会一直等待,并且会释放 mutex 锁
            }
            countFish++;
            // 捕到一条鱼
            printf("Catching Fish! Now there are %d fish, total fish are %d\n", countFish, i + 1);
            pthread_cond_signal (&cond_catchfish_finish);
            // 发送信号,告诉猫咪可以吃鱼了,但是如果没有捕够鱼,猫咪还是会等待,因为锁还没有释放
            pthread_mutex_unlock(&mutex) ;
            // 解锁
        }
        pthread_exit (NULL);
    }
    // 渔夫函数
    void* cat (){
        int i,j,k;
        for(i= 0; i < numFish * 2; i++){
            pthread_mutex_lock (&mutex) ;
            // 猫咪在吃鱼时,先加一把锁,防止渔夫把容器抢走
            while (countFish == 0){
                pthread_cond_wait (&cond_catchfish_finish, &mutex);
                // 如果容器中鱼没有了,猫咪就等待渔夫捕鱼的操作完成,然后继续吃鱼
                // 当没有接收到 cond_catchfish_finish 信号时,就会一直等待,并且会释放 mutex 锁
            }
            countFish--;
            // 吃掉一只鱼
            printf("Eating Fish! Now there are %d fish\n", countFish);
            pthread_cond_signal (&cond_eatfish_finish);
            // 发送信号,告诉渔夫可以继续捕鱼了,但是如果没有吃完鱼,渔夫还是会等待,因为锁没有释放
            pthread_mutex_unlock(&mutex) ;
            // 解锁
        }
        pthread_exit (NULL);
    }
    int main() {
        pthread_t fishmanThread, catThread;
        pthread_create(&fishmanThread, NULL, fishman, NULL);
        pthread_create (&catThread, NULL, cat, NULL);
        pthread_join (fishmanThread, NULL);
        pthread_join (catThread, NULL);
        printf ("The cat is full!\n");
        return 0;
    }
    \ No newline at end of file +线程 - 并行程序设计 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # pthread 库

    # 简介

    pthread 是 POSIX 线程库,它是一种跨平台的线程库,可以在不同的操作系统上使用。它是 C 语言的标准库,提供了一组函数来创建、管理和同步线程。"pthread" 库相对于 "thread" 库来说,更加底层和与操作系统紧密相关。

    thread 是 C11 标准提供的线程库,它是 C 的一部分。它提供了一种面向对象的方式来创建、管理和同步线程。"thread" 库提供了 std::thread 类,可以用于创建和管理线程,并提供了一些成员函数和工具函数来操作线程。

    pthread 相关函数需要引入标准库头文件: #include<pthread.h>

    # 参考资料

    开源教程:https://hpc-tutorials.llnl.gov/posix/

    pthread 库在不同的操作系统上有不同的实现,因此,官方的 pthread 库文档可能由各个操作系统的文档或开发者文档提供。以下是一些常见操作系统的 pthread 官方文档链接:

    • Linux: 在 Linux 系统上,pthread 库的官方文档可以在 GNU C 库(glibc)的文档中找到。您可以访问以下链接获取详细的 pthread 文档:https://www.gnu.org/software/libc/manual/html_node/Threads.html

    • macOS: 在 macOS 上,pthread 库的官方文档可以在 Apple 开发者文档中找到。您可以访问以下链接获取有关 pthread 的详细信息:https://developer.apple.com/library/archive/documentation/Cocoa/Conceptual/Multithreading/Introduction/Introduction.html

    • Windows: 在 Windows 操作系统上,Microsoft 提供了一种名为 "Windows Threads" 的线程库,它是 Windows 对 POSIX 线程的实现。您可以在 Microsoft 的官方文档中找到有关 "Windows Threads" 的详细信息:https://docs.microsoft.com/en-us/windows/win32/procthread/creating-threads

    • IBM:

      • z/OS:

        z/OS V2R4 POSIX Threads Programming: https://www.ibm.com/docs/en/zos/3.1.0?topic=programming-posix-threads-programming

      • IBM i:

        IBM i 7.4 PASE Programming and Reference: https://www.ibm.com/docs/en/i/7.4?topic=programming-pase-programming-reference

      • AIX:

        AIX 7.2 Library Reference - Threads and Concurrency: https://www.ibm.com/docs/en/aix/7.2?topic=library-threads-concurrency

    不同操作系统上的 pthread 库在语法和使用方式上可能会有一些差异。这是因为 pthread 库是根据 POSIX 线程标准实现的,而各个操作系统可能会在标准的基础上进行一些定制和扩展。

    # 多线程程序举例分析

    # 线程间接执行的函数

    c
    float *func1(float a, float b) {
        int i;
        float *result;
        result = (float *)malloc(sizeof(float) * 2);
        for (i = 0; i < 1000000000; ++i) {
            result[0] = a + b;
            result[1] = a - b;
        }
        return result;
    }
    float *func2(float a, float b) {
        int i;
        float *result;
        result = (float *)malloc(sizeof(float) * 2);
        for (i = 0; i < 1000000000; ++i) {
            result[0] = a * b;
            result[1] = a / b;
        }
        return result;
    }

    # 给间接函数传递参数用的结构体

    传递给子线程的将是这个结构体变量的地址,这个地址对父线程和子线程都是一样的,可见的,所以其内容是共享的,互相可以看到对方的修改。

    c
    typedef struct {
        float a;
        float b;
    } Args;

    # 线程直接执行的函数

    只能返回 void 指针类型,即返回 void*

    return 返回的地址内容将被 pthread_join 读出

    c
    void *thread_func1(void *arg) {
        Args *args = (Args *)arg; // 传递参数结构体变量的地址
        return func1(args->a, args->b);
    }
    void *thread_func2(void *arg) {
        Args *args = (Args *)arg; // 传递参数结构体变量的地址
        return func2(args->a, args->b);
    }

    # pthread_exit 结束线程

    除了 return 返回数据,还可以用 pthread_exit 函数返回数据并直接结束线程,而不必等到主线程 pthread_join 连接该线程。提前结束进程可以释放进程占用的内存、处理器,节省计算机资源 (真的能吗?)

    线程终止函数 pthread_exit 传入 1 个参数

    • void * 类型:非 void 类型指针,指向返回数据的地址
    c
    void *thread_func1(void *arg) {
        Args *args = (Args *)arg;
        pthread_exit(func1(args->a, args->b));
    }
    void *thread_func2(void *arg) {
        Args *args = (Args *)arg;
        pthread_exit(func1(args->a, args->b));
    }

    # 线程返回数据存储的有类型地址

    应当是数据的类型

    c
    result1 = (float *)malloc(sizeof(float) * 2);
    result2 = (float *)malloc(sizeof(float) * 2);

    # 线程返回数据直接存储的无类型地址

    只能是 void 指针类型

    c
    void *r1, *r2;

    # 定义 pthread_t 类型变量

    pthread_t 类型变量用于存储线程句柄 (TID) , t 表示数据类型 (type)

    c
    pthread_t tid1, tid2;

    # pthread_create 创建线程

    线程创建函数 pthread_create 传入 4 个参数:

    • pthread_t 类型:新线程的线程句柄 (TID) 要存储的地址 (pthread_t 变量指针)
    • const pthread_attr_t * 类型:属性
    • 执行函数 (只能是 void 类型,即无返回)
    • 传给执行函数的参数 (一次只能传一个参数,所以一般用结构体传参数)
    c
    pthread_create(&tid1, NULL, thread_func1, &ab);
    pthread_create(&tid2, NULL, thread_func2, &ab);

    函数返回结果:代表线程连接的状态,成功返回 0 ,否则返回错误码

    错误码:

    • EAGAIN : The system lacks the necessary resources to create another thread.
    • EINVAL : The value specified by thread is null.
    • ELEMULTITHREADFORK : pthread_create() was invoked from a child process created by calling fork() from a multi-threaded process. This child process is restricted from becoming multi-threaded.
    • ENOMEM : There is not enough memory to create the thread.

    # ⚠️区别进程的创建

    c
    for(i = 0; i < 5; i++)
        pthread_join(tid[i], NULL);

    这里与 Linux 创建进程(线程)的 fork 不同,后者会创建 252^5 个进程(线程),而这里是 5 个线程!因为子线程、父线程是并行的!

    # pthread_join 连接并读出线程执行结果

    线程数据读出函数 pthread_join 传入 2 个参数:

    • pthread_t 类型:线程句柄,即 TID
    • void ** 类型:执行函数的返回数据要存储的地址 (地址只能是 void 类型,一次只能返回一个地址)
    c
    pthread_join(tid1, &r1);
    pthread_join(tid2, &r2);

    函数返回结果:代表线程连接的状态,成功返回 0 ,否则返回错误码

    错误码:

    • EDEADLK : A deadlock was detected (e.g., two threads tried to join with each other); or thread specifies the calling thread.
    • EINVAL : hread is not a joinable thread.
    • EINVAL : Another thread is already waiting to join with this thread.
    • ESRCH : No thread with the ID thread could be found.

    一个线程只能同时连接一个线程,子线程自创建开始执行,若连接子线程时子线程还没结束,主线程会等到子线程执行结束后再读出数据

    Windows 中使用的是 WaitForSingleObject(... , ...) / WaitForMultipleObject(... , ...) ,因为 Windows 将所有资源对象命名为 Object

    # 无类型地址数据存储到有类型地址

    c
    result1 = (float *)r1;
    result2 = (float *)r2;

    # 完整程序

    c
    #include <pthread.h>
    #include <stdio.h>
    #include <stdlib.h>
    #include <sys/time.h>
    float *func1(float a, float b) {
        int i;
        float *result;
        result = (float *)malloc(sizeof(float) * 2);
        for (i = 0; i < 1000000000; ++i) {
            result[0] = a + b;
            result[1] = a - b;
        }
        return result;
    }
    float *func2(float a, float b) {
        int i;
        float *result;
        result = (float *)malloc(sizeof(float) * 2);
        for (i = 0; i < 1000000000; ++i) {
            result[0] = a * b;
            result[1] = a / b;
        }
        return result;
    }
    typedef struct {
        float a;
        float b;
    } Args;
    void *thread_func1(void *arg) {
        Args *args = (Args *)arg;
        return func1(args->a, args->b);
    }
    void *thread_func2(void *arg) {
        Args *args = (Args *)arg;
        return func1(args->a, args->b);
    }
    int main() {
        float a = 1.14, b = 5.14, *result1, *result2, during;
        Args ab;
        ab.a = a;
        ab.b = b;
        result1 = (float *)malloc(sizeof(float) * 2);
        result2 = (float *)malloc(sizeof(float) * 2);
        struct timeval start, end;
        gettimeofday(&start, NULL);
        void *r1, *r2;
        pthread_t tid1, tid2;
        pthread_create(&tid1, NULL, thread_func1, &ab);
        pthread_create(&tid2, NULL, thread_func2, &ab);
        pthread_join(tid1, &r1);
        pthread_join(tid2, &r2);
        result1 = (float *)r1;
        result2 = (float *)r2;
        gettimeofday(&end, NULL);
        during = (end.tv_sec - start.tv_sec) + (end.tv_usec - start.tv_usec) / 1000000.0;
        printf("result1[0] = %f\nresult1[1] = %f\n", result1[0], result1[1]);
        printf("result2[0] = %f\nresult2[1] = %f\n", result2[0], result2[1]);
        printf("during time = %f s\n", during);
        free(result1);
        free(result2);
        return 0;
    }

    # 线程间互动

    # 互斥锁 pthread_mutex

    对应数据类型 (type) 为 pthred_mutex_t ,以及相关函数 pthread_mutex_function

    互斥锁的作用主要是每个线程都可以在进入某个区域时 把这个区域锁起来, 不让其他的线程进入。 当自己的工作完成时或者达成某种条件时,则打开互斥锁。互斥锁存在的目的就是为了避免很多线程同时进入一个事件时发生冲突。

    # pthread_cond 条件变量

    对应数据类型 (type) 为 pthred_cond_t ,以及相关函数 pthread_cond_function

    条件变量主要和互斥锁一同使用,也就是当满足某一个条件时打开互斥锁。这个条件可以通过 pthread_cond_signal 或者是 pthread_cond_broadcast 发出

    # pthread_cond_signal

    • 一次只能激活一个条件变量
    • 不容易出现 “线程打架”

    # pthread_cond_broadcast

    • 一次性激活所有等待的条件变量
    • 容易出现 “线程打架”,造成 “死锁”

    # 线程互动程序举例分析

    # 初始化锁 pthread_mutex_t

    c
    pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;

    # 初始化条件变量 pthread_cond_t

    c
    pthread_cond_t cond_catchfish_finish = PTHREAD_COND_INITIALIZER;
    pthread_cond_t cond_eatfish_finish = PTHREAD_COND_INITIALIZER;

    # 线程函数

    # pthread_mutex_lock 上锁

    c
    pthread_mutex_lock (&mutex) ;

    上锁之后线程将无法收到来自其他线程的信号

    # 输入

    c
    int pthread_mutex_lock(pthread_mutex_t *mutex);
    //SUSv3:
    int pthread_mutex_lock(pthread_mutex_t *mutex);

    # 返回值

    • If successful, pthread_mutex_lock() returns 0.

    • If unsuccessful, pthread_mutex_lock() returns -1 and sets errno to one of the following values:

      • Error Code

        • Description*
      • EAGAIN

        • The mutex could not be acquired because the maximum number of recursive locks for mutex has been exceeded. This errno will only occur in the shared path.
      • EDEADLK

        • The current thread already owns the mutex, and the mutex has a kind attribute of __MUTEX_NONRECURSIVE.
      • EINVAL

        • The value specified by mutex is not valid.

    Special behavior for Single UNIX Specification, Version 3: If unsuccessful, pthread_mutex_lock () returns an error number to indicate the error. SUSv3 操作系统可以直接返回错误码

    # pthred_cond_wait 使线程进入等待状态

    c
    pthread_cond_wait(&cond_eatfish_finish, &mutex);

    会原子地释放之前获取的互斥锁,使当前线程进入等待状态,并等待条件变量满足特定的条件

    # 输入

    c
    int pthread_cond_wait(pthread_cond_t *cond,
                          pthread_mutex_t *mutex);
    //SUSv3:
    int pthread_cond_wait(pthread_cond_t * __restrict__cond, 
                          pthread_mutex_t * __restrict__mutex);

    # 返回值

    • If successful, pthread_cond_wait() returns 0.

    • If unsuccessful, pthread_cond_wait() returns -1 and sets errno to one of the following values:

      • Error Code

        • Description
      • EINVAL

        • Different mutexes were specified for concurrent operations on the same condition variable.
      • EPERM

        • The mutex was not owned by the current thread at the time of the call.

    Special behavior for Single UNIX Specification, Version 3: If unsuccessful, pthread_cond_wait () returns an error number to indicate the error. SUSv3 操作系统可以直接返回错误码

    # 完整程序

    c
    #include<stdio.h>
    #include<stdlib.h>
    #include<pthread.h>
    int countFish = 0;// 计数器,记录容器内鱼的数量
    const int numFish = 10;// 一筐鱼的数量
    pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;// 初始化一把叫做 mutex 的锁
    pthread_cond_t cond_catchfish_finish = PTHREAD_COND_INITIALIZER;// 初始化一个条件变量,这个条件变量为捕鱼操作是否完成
    pthread_cond_t cond_eatfish_finish = PTHREAD_COND_INITIALIZER;// 初始化一个条件变量,这个条件变量为吃鱼操作是否完成
    void* fishman (){
        int i, j, k;
        for(i= 0;i < numFish * 2; i++){
            pthread_mutex_lock (&mutex) ;
            // 渔夫在捕鱼时,先加一把锁,防止猫咪把容器抢走
            while (countFish == numFish){
                pthread_cond_wait(&cond_eatfish_finish, &mutex);
                // 如果容器中鱼满了,渔夫就等待猫咪吃完鱼,然后再继续捕鱼
                // 当没有接收到 cond_eatfish_finish 信号时,就会一直等待,并且会释放 mutex 锁
            }
            countFish++;
            // 捕到一条鱼
            printf("Catching Fish! Now there are %d fish, total fish are %d\n", countFish, i + 1);
            pthread_cond_signal (&cond_catchfish_finish);
            // 发送信号,告诉猫咪可以吃鱼了,但是如果没有捕够鱼,猫咪还是会等待,因为锁还没有释放
            pthread_mutex_unlock(&mutex) ;
            // 解锁
        }
        pthread_exit (NULL);
    }
    // 渔夫函数
    void* cat (){
        int i,j,k;
        for(i= 0; i < numFish * 2; i++){
            pthread_mutex_lock (&mutex) ;
            // 猫咪在吃鱼时,先加一把锁,防止渔夫把容器抢走
            while (countFish == 0){
                pthread_cond_wait (&cond_catchfish_finish, &mutex);
                // 如果容器中鱼没有了,猫咪就等待渔夫捕鱼的操作完成,然后继续吃鱼
                // 当没有接收到 cond_catchfish_finish 信号时,就会一直等待,并且会释放 mutex 锁
            }
            countFish--;
            // 吃掉一只鱼
            printf("Eating Fish! Now there are %d fish\n", countFish);
            pthread_cond_signal (&cond_eatfish_finish);
            // 发送信号,告诉渔夫可以继续捕鱼了,但是如果没有吃完鱼,渔夫还是会等待,因为锁没有释放
            pthread_mutex_unlock(&mutex) ;
            // 解锁
        }
        pthread_exit (NULL);
    }
    int main() {
        pthread_t fishmanThread, catThread;
        pthread_create(&fishmanThread, NULL, fishman, NULL);
        pthread_create (&catThread, NULL, cat, NULL);
        pthread_join (fishmanThread, NULL);
        pthread_join (catThread, NULL);
        printf ("The cat is full!\n");
        return 0;
    }
    \ No newline at end of file diff --git a/cs/principles-of-computer-composition/basic-indicators-of-computer-performance/index.html b/cs/principles-of-computer-composition/basic-indicators-of-computer-performance/index.html index f6c0e1c1..3651cf83 100644 --- a/cs/principles-of-computer-composition/basic-indicators-of-computer-performance/index.html +++ b/cs/principles-of-computer-composition/basic-indicators-of-computer-performance/index.html @@ -1,4 +1,4 @@ -计算机性能的基本指标 - 计算机组成原理 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 思维导图

    计算机性能的基本指标

    # 完成单个任务的时间

    # 响应时间 (response time):

    完成单个任务所需的总时间,由 2 部分组成

    • 响应时间短的场合:
      事务处理系统 (存 / 取款的速度要快)

    • CPU 执行时间 (CPU time)
      执行时间 (Execution time)
      CPU 真正花费在程序执行上的时间

      • 用户 CPU 时间:
        用来运行用户代码的时间

      • 系统 CPU 时间:
        为执行用户程序而需运行一些操作系统代码的时间

    • 其他时间

      • 等待 I/O 操作完成

      • CPU 花在其他用户程序的时间

    # 单位时间完成的任务量

    # 吞吐率 (throughput)

    • 吞吐率高的场合:
      多媒体应用 (音 / 视频播放要流畅)

    • 不同层次上对吞吐率性能的度量

      • ISA

        • MIPS
          Million Instructions Per Seconds
          因为每条指令执行时间不同,所以 MIPS 是一个平均值,
          指令使用的频度动态变化

          • MIPS
            = Instruction Counts / ( Execution Time ×10^6 )
            = Clock Rate / ( CPI × 10^6 )
            = 1 / ( Clock Cycle × CPI × 10^6 )

          • 用 MIPS 数表示性能有局限

            • 不同机器的指令集不同

            • 程序由不同的指令混合而成

            • 指令使用的频度动态变化

        • MFLOPS
          Million Floating-point Operations Per Second

          • MFLOPS
            = FP Operations / ( Execution Time × 10^6 )

            • Peak MIPS
              取一组指令组合,使得得到的平均 CPI 最小,由此得到的 MIPS 就是峰值 MIPS (Peak MIPS)
              (不实用)
        • Datapath Control

          • Megabytes per second (带宽)
        • Function Units

          • Transistors

          • Wires

          • Pins

          • Cycles per second (频率)

    # 带宽 (bandwidth)

    # 系统性能

    (System performance)
    表示系统响应时间,与 CPU 之外的其他部分都有关系

    # 影响系统性能的硬件技术指标

    • 主频

      • 定义

        • CPU 的工作节拍是由时钟控制的,时钟不断产生固定频率的时钟脉冲,这个时钟的频率就是 CPU 的主频
      • 主频越高,CPU 的工作节拍就越快,运算速度就越高

      • 主频通常用一秒钟内处理器所能发出电子脉冲数来表示,单位一般为 GHz

      • 芯片的功耗与频率成正比

    • 运算速度

      • 定义

        • 每秒钟所能执行的指令条数
      • 计量单位

        • MIPS

        • MFLOPS

      • 计算方法

        • 吉布森混合法:各类指令根据出现频率加权求平均

        • 计算各种指令的执行速度

        • 计算典型程序的运算速度

        • 模型分析和模拟等其他方法

    • 运算精度

      • 定义

        • 计算机能直接处理的二进制位数
      • 一般和 CPU 中存储的数据寄存器的位数是相同。位数越多,精度越高

      • 参与运算的操作数的基本位数称之为基本字长 。早期的微机字长为 8 位或 16 位,现为 32 位或 64 位

    • 存储容量

      • 主存容量

        • 存储单元个数 × 存储字长

        • 主存越大,处理问题的速度越快

        • 与辅存交换次数越少,访存效率越高

      • 辅存容量

    # CPU 性能

    (CPU performance)
    表示用户 CPU 时间
    衡量指标是: CPU time

    # 一个程序的 CPU 执行时间 (CPU time)

    执行时间 (Execution time)
    = 一个程序的 CPU 时钟周期数 × 时钟周期时间
    = 一个程序的 CPU 时钟周期数 ÷ 时钟频率
    = 程序的指令数 × CPI × 时钟周期时间
    = 程序的指令数 × CPI ÷ 时钟频率
    = Instruction Counts / (MIPS × 10^6)

    • 一个程序的 CPU 时钟周期数
      =程序的指令数 × CPI

    Clock Cycle Counts = Instructions Counts × CPI

    - CPI (avg. clock Cycles Per Instruction)
    +计算机性能的基本指标 - 计算机组成原理 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 思维导图

    计算机性能的基本指标

    # 完成单个任务的时间

    # 响应时间 (response time):

    完成单个任务所需的总时间,由 2 部分组成

    • 响应时间短的场合:
      事务处理系统 (存 / 取款的速度要快)

    • CPU 执行时间 (CPU time)
      执行时间 (Execution time)
      CPU 真正花费在程序执行上的时间

      • 用户 CPU 时间:
        用来运行用户代码的时间

      • 系统 CPU 时间:
        为执行用户程序而需运行一些操作系统代码的时间

    • 其他时间

      • 等待 I/O 操作完成

      • CPU 花在其他用户程序的时间

    # 单位时间完成的任务量

    # 吞吐率 (throughput)

    • 吞吐率高的场合:
      多媒体应用 (音 / 视频播放要流畅)

    • 不同层次上对吞吐率性能的度量

      • ISA

        • MIPS
          Million Instructions Per Seconds
          因为每条指令执行时间不同,所以 MIPS 是一个平均值,
          指令使用的频度动态变化

          • MIPS
            = Instruction Counts / ( Execution Time ×10^6 )
            = Clock Rate / ( CPI × 10^6 )
            = 1 / ( Clock Cycle × CPI × 10^6 )

          • 用 MIPS 数表示性能有局限

            • 不同机器的指令集不同

            • 程序由不同的指令混合而成

            • 指令使用的频度动态变化

        • MFLOPS
          Million Floating-point Operations Per Second

          • MFLOPS
            = FP Operations / ( Execution Time × 10^6 )

            • Peak MIPS
              取一组指令组合,使得得到的平均 CPI 最小,由此得到的 MIPS 就是峰值 MIPS (Peak MIPS)
              (不实用)
        • Datapath Control

          • Megabytes per second (带宽)
        • Function Units

          • Transistors

          • Wires

          • Pins

          • Cycles per second (频率)

    # 带宽 (bandwidth)

    # 系统性能

    (System performance)
    表示系统响应时间,与 CPU 之外的其他部分都有关系

    # 影响系统性能的硬件技术指标

    • 主频

      • 定义

        • CPU 的工作节拍是由时钟控制的,时钟不断产生固定频率的时钟脉冲,这个时钟的频率就是 CPU 的主频
      • 主频越高,CPU 的工作节拍就越快,运算速度就越高

      • 主频通常用一秒钟内处理器所能发出电子脉冲数来表示,单位一般为 GHz

      • 芯片的功耗与频率成正比

    • 运算速度

      • 定义

        • 每秒钟所能执行的指令条数
      • 计量单位

        • MIPS

        • MFLOPS

      • 计算方法

        • 吉布森混合法:各类指令根据出现频率加权求平均

        • 计算各种指令的执行速度

        • 计算典型程序的运算速度

        • 模型分析和模拟等其他方法

    • 运算精度

      • 定义

        • 计算机能直接处理的二进制位数
      • 一般和 CPU 中存储的数据寄存器的位数是相同。位数越多,精度越高

      • 参与运算的操作数的基本位数称之为基本字长 。早期的微机字长为 8 位或 16 位,现为 32 位或 64 位

    • 存储容量

      • 主存容量

        • 存储单元个数 × 存储字长

        • 主存越大,处理问题的速度越快

        • 与辅存交换次数越少,访存效率越高

      • 辅存容量

    # CPU 性能

    (CPU performance)
    表示用户 CPU 时间
    衡量指标是: CPU time

    # 一个程序的 CPU 执行时间 (CPU time)

    执行时间 (Execution time)
    = 一个程序的 CPU 时钟周期数 × 时钟周期时间
    = 一个程序的 CPU 时钟周期数 ÷ 时钟频率
    = 程序的指令数 × CPI × 时钟周期时间
    = 程序的指令数 × CPI ÷ 时钟频率
    = Instruction Counts / (MIPS × 10^6)

    • 一个程序的 CPU 时钟周期数
      =程序的指令数 × CPI

    Clock Cycle Counts = Instructions Counts × CPI

    - CPI (avg. clock Cycles Per Instruction)
     

    每条指令的平均时钟周期数

    	- 对于某一条特定的指令而言,其CPI是一个确定的值 
     						
     	- 对于某一类指令、或一个程序、或一台机器而言,其CPI是一个平均值 
    @@ -6,4 +6,4 @@
     - 程序包含的指令数量 (Instruction Counts) 
     						
     - CPI
    -
    • 时钟周期 × 时钟频率 = 1

      • 时钟周期 (Clock Cycle)

      • 时钟频率 (Clock Rate)

    # 改善性能的措施

    • 采用更复杂的指令

      • 减少指令数

      • 可能导致 CPI 升高

      • 可能导致时钟周期增大

    • 采用更简单的指令

      • 可降低 CPI

      • 可能导致指令数增加

      • 可能缩短时钟周期

    • 减少每个指令执行的周期数

      • 可降低 CPI

      • 每个时钟周期需完成的操作更多,时钟周期可能需要增大

    • 缩短时钟周期

      • 每个周期内完成的操作有限,可能导致 CPI 提高

    # 影响因素

    • 算法

      • 指令数

        • 算法决定源程序的指令条数,因此决定了处理器执行的指令条数
      • CPI

        • 算法由于对慢速或快速指令的不同倾向性而同样影响了 CPI
    • 编程语言

      • 指令数

        • 由于编程语言中的语句被翻译成 CPU 指令,而后者决定了指令条数,因此编程语言势必会影响指令条数
      • CPI

        • 由于编程语言的自身特点,它可能同样影响 CPI。例:强支持数据抽象的语言 (如 Java) 要求间接调用,而这往往会用到具有高 CPI 的指令
    • 编译器

      • 指令数

      • CPI

    • IS / 指令集 / 体系结构
      / ISA

      • 指令数

      • CPI

      • 时钟频率 / 时钟周期

    # Amdahl 定律

    #

    # 吞吐率高且响应时间短的场合:

    ATM、文件服务器、Web 服务器等

    \ No newline at end of file +
    • 时钟周期 × 时钟频率 = 1

      • 时钟周期 (Clock Cycle)

      • 时钟频率 (Clock Rate)

    # 改善性能的措施

    • 采用更复杂的指令

      • 减少指令数

      • 可能导致 CPI 升高

      • 可能导致时钟周期增大

    • 采用更简单的指令

      • 可降低 CPI

      • 可能导致指令数增加

      • 可能缩短时钟周期

    • 减少每个指令执行的周期数

      • 可降低 CPI

      • 每个时钟周期需完成的操作更多,时钟周期可能需要增大

    • 缩短时钟周期

      • 每个周期内完成的操作有限,可能导致 CPI 提高

    # 影响因素

    • 算法

      • 指令数

        • 算法决定源程序的指令条数,因此决定了处理器执行的指令条数
      • CPI

        • 算法由于对慢速或快速指令的不同倾向性而同样影响了 CPI
    • 编程语言

      • 指令数

        • 由于编程语言中的语句被翻译成 CPU 指令,而后者决定了指令条数,因此编程语言势必会影响指令条数
      • CPI

        • 由于编程语言的自身特点,它可能同样影响 CPI。例:强支持数据抽象的语言 (如 Java) 要求间接调用,而这往往会用到具有高 CPI 的指令
    • 编译器

      • 指令数

      • CPI

    • IS / 指令集 / 体系结构
      / ISA

      • 指令数

      • CPI

      • 时钟频率 / 时钟周期

    # Amdahl 定律

    #

    # 吞吐率高且响应时间短的场合:

    ATM、文件服务器、Web 服务器等

    \ No newline at end of file diff --git a/cs/principles-of-computer-composition/cache/index.html b/cs/principles-of-computer-composition/cache/index.html index 9437be60..80cb8804 100644 --- a/cs/principles-of-computer-composition/cache/index.html +++ b/cs/principles-of-computer-composition/cache/index.html @@ -1 +1 @@ -Cache - 计算机组成原理 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 思维导图

    Cache

    # Cache 的结构

    # Cache 是小容量、高速缓冲存储器,由 SRAM 组成

    # 一般将 Cache 和主存的存储空间都划分为若干大小相同的块并映射

    • 把主存划分成大小相等的主存块 (Block)

    • Cache 中存放一个主存块的对应单位称为行 (line) 或槽 (Slot) 或项 (Entry) 或块 (Block),一段 Cache 行包括:

      • Valid 位,1 位

      • LRU 位

      • Dirty 位,1 位,采用 Write Back (写回、一次性写、回写) 时需要

      • Tag 位

      • Data 数据

    • 主存块 (Block) 与 Cache 中的 行 / 槽 / 项 / 块 映射

    # Cache 中存放最近访问的数据

    # Cache 工作原理

    # 程序运行时,CPU 使用的一部分数据 / 指令会预先成批拷贝到 Cache 中,Cache 的内容是主存储器中部分内容的映象 (副本)

    # 当 CPU 需从主存读 (写) 数据或指令时,先查看 Cache

    • 失效或缺失 (miss)
      若被访问信息不在 cache 中,则直接从 Cache 中取,不用访问主存

    • 命中 (hit)
      若被访问信息在 cache 中,则直接访问主存

    # Cache 对程序员 (编译器) 是透明的

    # 对操作系统程序员也是基本透明的

    # 对一些内核高级程序员不是透明的

    # 程序访问局部性

    # 时间局部性 (Temporal Locality) :

    刚被访问过的存储单元很可能不久又被访问
    所以让最近被访问过的信息保留在靠近 CPU 的存储器中

    # 空间局部性 (Spatial Locality) :

    刚被访问过的存储单元的邻近单元很可能不久被访问,单个数据不考虑空间局部性
    所以将刚被访问过的存储单元的邻近单元调到靠近 CPU 的存储器中

    # 块大小的设置需利用空间局部性

    • 块太小,无法利用空间局部性,发生缺失的概率增大

    # Cache 在计算机存储层次结构中的位置

    #

    # Cache 位于 CPU 内,速度几乎与 CPU 一样快

    # 块 (Block) 是一个定长块,是两层存储器之间存储信息的最小单元。Cache 是主存一部分的副本

    # Cache 替换算法 / 淘汰策略问题

    # 当一个新的主存块需要复制到 Cache 中时,如果 Cache 中的对应行已经全部被占满,如何选择被替换掉的 Cache 行?

    • 直接映射(Direct Mapped)Cache
      映射唯一,无条件用新信息替换老信息

    • N 路组相联(N-way Set Associative)Cache
      每个主存数据有 N 个 Cache 行可选择,需考虑替换哪一行

    • 全相联(Fully Associative)Cache
      每个主存数据可存放到 Cache 任意行中,需考虑替换哪一行

    # 先进先出(First In First Out,FIFO)

    总是把最近最少用的那一块淘汰掉,利用时间局部性

    • 加一位标记位标记当前哪一行是最先进入的

    • 替换掉之后按顺序将下一行标记为最先进入的,最后一行被替换掉则标记第一行

    # 最近最少用(Least Recently Used,LRU)

    总是把最近最少用的那一块淘汰掉,利用时间局部性

    • 具体实现:通过给每个 Cache 行设定一个计数器,根据计数值来记录这些主存块的使用情况。这个计数值称为 LRU 位。

      • 组相联每组 n 行时,LRU 位长度 = log_2 (n)

      • 全相联相当于组相联只有一组,LRU 位长度 = log_2 (Cache 的行数)

      • 命中时,被访问行的 LRU 位置 0,其他行 LRU 位加 1,其余不变

      • 未命中且该组未满时,新行 LRU 位置为 0,其余全加 1

      • 未命中且该组已满时,LRU 位最大的那一行中的主存块被淘汰,新行 LRU 位置为 0,其余加 1

      • LRU 位的最大值 = n-1 (从 0 开始计数)

    • LRU 算法的命中率随组中行数的增大而提高

    # 随机替换(Random)

    • 随机地从候选的槽中选取一个淘汰,与使用情况无关

    • 模拟试验表明,随机替换算法在性能上只稍逊于 LRU 算法,而且代价低!

    # 最不经常使用 LFU

    # 相关指标与术语

    # 命中(Hit):要访问的信息在 Cache 中

    • Hit Rate (命中率 p): 在 Cache 中的概率

    • Hit Time (命中时间 Tc) :访问 Cache 所需时间,包括:判断时间 + Cache 访问

    # 失效(Miss):要访问的信息不在 Cache 中

    • Miss Rate (缺失率 / 失效率) = 1 - (Hit Rate) = 1-p

    • Miss Penalty (失效损失 Tm):从主存将一块数据读取到 Cache 所需时间,包括访问主存块,向上逐层传输块直至将数据块放入发生缺失的那一层所需时间。

    # 平均访问时间 = p × Tc + (1-p) × (Tm + Tc) = Tc + (1-p) × Tm

    【失效损失 Tm】 不包括 【命中时间 Tc】!考点

    # Cache 抖动

    由于内存中不同主存块都映射到 Cache 同一行,某些块在主存和 Cache 之间频繁传送,产生了频繁的 Cache 替换,称之为 Cache 抖动

    • 可能引起 Cache 抖动的 Cache 失效类型

      • 容量失效

      • 冲突失效

    • 增加容量和相联性,有助于缓解这种现象

    # 逻辑地址

    CPU 给出的虚拟地址

    # 物理地址 / 主存地址

    在 Cache 和主存中查询所用的地址

    • Tag 位,与 Cache 中的 Tag 位比较,相等则命中

    • Index 位,组号位,只有组相联才有

    • Offset 位,在 Cache 行或主存块中的偏移量,即最低位块内地址

    # 关联度

    主存块映射到 Cache 时,可能存放的位置个数

    • 关联度最低?直接映射(关联度为 1)

    • 关联度居中?N - 路组相联映射(关联度为 N)

    • 关联度最高?全相联映射(为 Cache 行数)

    • Cache 大小和块大小一定的情况下直观结论

      • 提高关联度通常能够降低缺失率 (miss rate)

      • 提高关联度通常会增加命中时间

    # Cache 的失效率(Miss Rate)

    # Miss Rate 与 Cache 大小、Block 大小、映射方式、Cache 级数等有关

    • Cache 大小:

      • Cache 越大,Miss Rate 越低

      • 但成本越高!

    • Block 大小:

      • Block 越大,Miss Rate 越低,因为空间局部性充分发掘

      • Block 大小在 Cache 大小中所占的比例增加到一定程度时, Miss Rate 也会随之增加

      • Cache 不扩大时 Cache Block 的总数变少了,冲突变大

    • Cache 映射方式

      • Cache 容量小时, Cache 映射方式对 Miss Rate 有影响

      • Cache 容量大时,Cache 映射方式对 Miss Rate 影响不大

        • 映射到同一组的概率降低

    # Cache 失效类型

    • 强制失效 (Compulsory misses)

      • 首次访问某数据块时,必然引起的 Cache 失效

      • 增加 Block 大小,有利于减少此类不命中

    • 容量失效 (Capacity misses)

      • Cache 不能存放程序运行所需的所有块,替换后再次被使用所引起的 Cache 失效

      • 增加 Cache 大小,有利于减少此类不命中

    • 冲突失效 (Conflict misses)

      • 映射到同一组的数据块个数超过组内可容纳的块时,竞争所引起的 Cache 失效

      • 全相联没有此类失效,但价格贵且访问速度慢

    # Cache 的一致性问题

    # 导致 Cache 和主存数据不一致的情况

    • 情况 1:当 Cache 中的内容进行更新时,而没有改变主存中的相应内容时,Cache 和主存之间产生了不一致 (inconsistent)

    • 情况 2:当多个设备都允许访问主存时

      • 例:I/O 设备可通过 DMA 方式直接读写内存时,如果 Cache 中的内容被修改,则 I/O 设备读出的对应主存单元的内容无效;若 I/O 设备修改了主存单元的内容,则对应 Cache 行中的内容无效。
    • 情况 3:当多个 CPU 都有各自私有的 Cache 并且共享主存时

      • 例:某个 CPU 修改了自身 Cache 中的内容,则对应的主存单元和其他 CPU 中对应的 Cache 行的内容都要变为无效。

    # Cache 写机制

    • 写命中 (Write Hit)
      要写的单元已经在 Cache 中时

      • Write Through (写直达、写通过、直写、写穿透)
        当一个写操作产生时,无论写的是 Cache 还是主存,新值同时写到 Cache 和主存的块中

        • 问题:内存速度太慢,所以会导致 CPU “被拖累”,CPU 性能降低

        • 一种改进措施:在 Cache 和主存之间使用写缓冲 (Write Buffer)

          • 当一个数据等待被写入主存时,先将其存入写缓冲;

          • 在把数据写入 Cache 和写缓冲后,处理器继续执行命令;

          • 当写主存操作结束后,写缓冲里的数据释放

        • 写穿透 / 直达 Cache 可用写分配或写不分配

      • Write Back (写回、一次性写、回写)
        当一个写操作产生时,新值仅被写到 Cache 中,而不被写入主存

        • 特点

          • 大大降低主存带宽需求

          • 提高系统性能

          • 控制可能很复杂

        • 每个 Cache 行都设置一个修改位 (“dirty bit - 脏位”)

          • 如果修改位为 “0”,则说明对应主存块没有被修改过

          • 如果对应 Cache 行中的主存块被修改,就同时置修改位为 “1”

          • 只有当修改位为 “1” 的块从 cache 中替换出去时,才把它写回主存
            (写不命中时 Cache 里没有有效的对应块,即使写分配也不需要考虑 Cache 中的数据。但是写分配写入 Cache 新的行时,被替换掉的行不能忘了考虑写回)

          • 修改位为 “0” 的块从 cache 中替换出去时,不写回主存

        • 写回 Cache 通常用写分配

    • 写不命中 (Write Miss)
      要写的单元不在 Cache 中时

      • Allocate-on-miss (写分配)

        • 更新主存块中相应单元,再将该主存块装入 Cache;

        • 试图利用空间局部性,但增加了从主存读入一个块的开销

      • No-allocate-on-write (写不分配)

        • 直接写主存,不把被写数据放入 Cache,同时将 Cache 中该行 Valid 位置 0;

        • 减少了从主存读一个块的开销,但没有利用空间局部性

    # Cache 和主存之间的映射

    # 直接映射 / 模映射 (Direct Mapped)

    • 原理:

      • 把主存的每一块映射到一个固定的 Cache 行中。
        Cache 总行数为 N,将存储空间总块数为 M 的主存分成 K = M/N 个组群,每个组群有 N 个块;
        即每个主存地址对应于高速缓存中唯一的地址,也称模映射(本质是哈希)
    • 映射关系为:Cache 行号 = 主存块号 mod Cache 行数

    • 物理地址

      • Tag 位,标记位
        长度 = log _2 (主存中 块 的 数量 / Cache 中 块 的 数量) =log_2 (K)

      • Index 位,Cache 槽号
        长度 = log _2 (Cache 中 块 的 数量) = Log_2 (N)

      • Offset 位,块内地址
        长度 = log _2 (块 的 大小) (按编址单位算)

    • 特点

      • 关联度 = 1

      • 易实现 (求低地址就行了) 电路简单,命中时间短

      • 淘汰 / 替换策略简单

      • 不灵活,Cache 存储空间得不到充分利用,命中率低,抖动多

    # 全相联映射(Fully Associative)

    • 原理:
      主存块可装到 Cache 任一行 / 槽中,称为全相联映射
      Cache 有几个 slot 就最多需要查找多少次 slot,查找时需要一个 slot 一个 slot 比较 Tag 位查找(除非并行同时比较所有 Cache 行的 Tag,硬件实现更复杂)

    • 物理地址

      • Tag 位,标记位
        长度 = log _2 (主存中 块 的 数量)

      • Offset 位,块内地址
        长度 = log _2 (块 的 大小) (按编址单位算)

    • 特点:

      • 关联度 = Cache 的行的数量

      • 块冲突概率低:只要有空闲 Cache 块,都不会发生冲突

      • 实现复杂 (比较逻辑的硬件代价大)、速度慢

    # 组相联映射(N-way Set Associative)

    • n - 路组相联将总行数为 N 的 Cache 分成 S = N/n 组,每一组有 n 行;
      将存储空间总块数为 M 的主存分成 K = M/S = M*n/N 个组群,每个组群有 S 个块;
      把主存块映射到 Cache 固定组的任一行中。即:组间模映射、组内全映射

    • 物理地址

      • Tag 位,标记位
        长度 = log _2 (主存中 块 的 数量 / 组的数量) = log _2 (K)

      • 组号位
        长度 = log _2 (组 的 数量) = log _2 (S)

      • Offset 位,块内地址
        长度 = log _2 (块 的 大小) (按编址单位算)

    • 特点

      • 关联度 = n

      • 结合直接映射和全相联映射的优点。当 Cache 的组数为 1 时,则为全相联映射;

      • 每组两个行(2 路组相联)较常用。在较大容量的 L2 Cahce 和 L3 Cahce 中使用 4 路以上

      • 当每组只有一行时,则为直接映射

      • 当只有一组时,则为全相联映射

    \ No newline at end of file +Cache - 计算机组成原理 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 思维导图

    Cache

    # Cache 的结构

    # Cache 是小容量、高速缓冲存储器,由 SRAM 组成

    # 一般将 Cache 和主存的存储空间都划分为若干大小相同的块并映射

    • 把主存划分成大小相等的主存块 (Block)

    • Cache 中存放一个主存块的对应单位称为行 (line) 或槽 (Slot) 或项 (Entry) 或块 (Block),一段 Cache 行包括:

      • Valid 位,1 位

      • LRU 位

      • Dirty 位,1 位,采用 Write Back (写回、一次性写、回写) 时需要

      • Tag 位

      • Data 数据

    • 主存块 (Block) 与 Cache 中的 行 / 槽 / 项 / 块 映射

    # Cache 中存放最近访问的数据

    # Cache 工作原理

    # 程序运行时,CPU 使用的一部分数据 / 指令会预先成批拷贝到 Cache 中,Cache 的内容是主存储器中部分内容的映象 (副本)

    # 当 CPU 需从主存读 (写) 数据或指令时,先查看 Cache

    • 失效或缺失 (miss)
      若被访问信息不在 cache 中,则直接从 Cache 中取,不用访问主存

    • 命中 (hit)
      若被访问信息在 cache 中,则直接访问主存

    # Cache 对程序员 (编译器) 是透明的

    # 对操作系统程序员也是基本透明的

    # 对一些内核高级程序员不是透明的

    # 程序访问局部性

    # 时间局部性 (Temporal Locality) :

    刚被访问过的存储单元很可能不久又被访问
    所以让最近被访问过的信息保留在靠近 CPU 的存储器中

    # 空间局部性 (Spatial Locality) :

    刚被访问过的存储单元的邻近单元很可能不久被访问,单个数据不考虑空间局部性
    所以将刚被访问过的存储单元的邻近单元调到靠近 CPU 的存储器中

    # 块大小的设置需利用空间局部性

    • 块太小,无法利用空间局部性,发生缺失的概率增大

    # Cache 在计算机存储层次结构中的位置

    #

    # Cache 位于 CPU 内,速度几乎与 CPU 一样快

    # 块 (Block) 是一个定长块,是两层存储器之间存储信息的最小单元。Cache 是主存一部分的副本

    # Cache 替换算法 / 淘汰策略问题

    # 当一个新的主存块需要复制到 Cache 中时,如果 Cache 中的对应行已经全部被占满,如何选择被替换掉的 Cache 行?

    • 直接映射(Direct Mapped)Cache
      映射唯一,无条件用新信息替换老信息

    • N 路组相联(N-way Set Associative)Cache
      每个主存数据有 N 个 Cache 行可选择,需考虑替换哪一行

    • 全相联(Fully Associative)Cache
      每个主存数据可存放到 Cache 任意行中,需考虑替换哪一行

    # 先进先出(First In First Out,FIFO)

    总是把最近最少用的那一块淘汰掉,利用时间局部性

    • 加一位标记位标记当前哪一行是最先进入的

    • 替换掉之后按顺序将下一行标记为最先进入的,最后一行被替换掉则标记第一行

    # 最近最少用(Least Recently Used,LRU)

    总是把最近最少用的那一块淘汰掉,利用时间局部性

    • 具体实现:通过给每个 Cache 行设定一个计数器,根据计数值来记录这些主存块的使用情况。这个计数值称为 LRU 位。

      • 组相联每组 n 行时,LRU 位长度 = log_2 (n)

      • 全相联相当于组相联只有一组,LRU 位长度 = log_2 (Cache 的行数)

      • 命中时,被访问行的 LRU 位置 0,其他行 LRU 位加 1,其余不变

      • 未命中且该组未满时,新行 LRU 位置为 0,其余全加 1

      • 未命中且该组已满时,LRU 位最大的那一行中的主存块被淘汰,新行 LRU 位置为 0,其余加 1

      • LRU 位的最大值 = n-1 (从 0 开始计数)

    • LRU 算法的命中率随组中行数的增大而提高

    # 随机替换(Random)

    • 随机地从候选的槽中选取一个淘汰,与使用情况无关

    • 模拟试验表明,随机替换算法在性能上只稍逊于 LRU 算法,而且代价低!

    # 最不经常使用 LFU

    # 相关指标与术语

    # 命中(Hit):要访问的信息在 Cache 中

    • Hit Rate (命中率 p): 在 Cache 中的概率

    • Hit Time (命中时间 Tc) :访问 Cache 所需时间,包括:判断时间 + Cache 访问

    # 失效(Miss):要访问的信息不在 Cache 中

    • Miss Rate (缺失率 / 失效率) = 1 - (Hit Rate) = 1-p

    • Miss Penalty (失效损失 Tm):从主存将一块数据读取到 Cache 所需时间,包括访问主存块,向上逐层传输块直至将数据块放入发生缺失的那一层所需时间。

    # 平均访问时间 = p × Tc + (1-p) × (Tm + Tc) = Tc + (1-p) × Tm

    【失效损失 Tm】 不包括 【命中时间 Tc】!考点

    # Cache 抖动

    由于内存中不同主存块都映射到 Cache 同一行,某些块在主存和 Cache 之间频繁传送,产生了频繁的 Cache 替换,称之为 Cache 抖动

    • 可能引起 Cache 抖动的 Cache 失效类型

      • 容量失效

      • 冲突失效

    • 增加容量和相联性,有助于缓解这种现象

    # 逻辑地址

    CPU 给出的虚拟地址

    # 物理地址 / 主存地址

    在 Cache 和主存中查询所用的地址

    • Tag 位,与 Cache 中的 Tag 位比较,相等则命中

    • Index 位,组号位,只有组相联才有

    • Offset 位,在 Cache 行或主存块中的偏移量,即最低位块内地址

    # 关联度

    主存块映射到 Cache 时,可能存放的位置个数

    • 关联度最低?直接映射(关联度为 1)

    • 关联度居中?N - 路组相联映射(关联度为 N)

    • 关联度最高?全相联映射(为 Cache 行数)

    • Cache 大小和块大小一定的情况下直观结论

      • 提高关联度通常能够降低缺失率 (miss rate)

      • 提高关联度通常会增加命中时间

    # Cache 的失效率(Miss Rate)

    # Miss Rate 与 Cache 大小、Block 大小、映射方式、Cache 级数等有关

    • Cache 大小:

      • Cache 越大,Miss Rate 越低

      • 但成本越高!

    • Block 大小:

      • Block 越大,Miss Rate 越低,因为空间局部性充分发掘

      • Block 大小在 Cache 大小中所占的比例增加到一定程度时, Miss Rate 也会随之增加

      • Cache 不扩大时 Cache Block 的总数变少了,冲突变大

    • Cache 映射方式

      • Cache 容量小时, Cache 映射方式对 Miss Rate 有影响

      • Cache 容量大时,Cache 映射方式对 Miss Rate 影响不大

        • 映射到同一组的概率降低

    # Cache 失效类型

    • 强制失效 (Compulsory misses)

      • 首次访问某数据块时,必然引起的 Cache 失效

      • 增加 Block 大小,有利于减少此类不命中

    • 容量失效 (Capacity misses)

      • Cache 不能存放程序运行所需的所有块,替换后再次被使用所引起的 Cache 失效

      • 增加 Cache 大小,有利于减少此类不命中

    • 冲突失效 (Conflict misses)

      • 映射到同一组的数据块个数超过组内可容纳的块时,竞争所引起的 Cache 失效

      • 全相联没有此类失效,但价格贵且访问速度慢

    # Cache 的一致性问题

    # 导致 Cache 和主存数据不一致的情况

    • 情况 1:当 Cache 中的内容进行更新时,而没有改变主存中的相应内容时,Cache 和主存之间产生了不一致 (inconsistent)

    • 情况 2:当多个设备都允许访问主存时

      • 例:I/O 设备可通过 DMA 方式直接读写内存时,如果 Cache 中的内容被修改,则 I/O 设备读出的对应主存单元的内容无效;若 I/O 设备修改了主存单元的内容,则对应 Cache 行中的内容无效。
    • 情况 3:当多个 CPU 都有各自私有的 Cache 并且共享主存时

      • 例:某个 CPU 修改了自身 Cache 中的内容,则对应的主存单元和其他 CPU 中对应的 Cache 行的内容都要变为无效。

    # Cache 写机制

    • 写命中 (Write Hit)
      要写的单元已经在 Cache 中时

      • Write Through (写直达、写通过、直写、写穿透)
        当一个写操作产生时,无论写的是 Cache 还是主存,新值同时写到 Cache 和主存的块中

        • 问题:内存速度太慢,所以会导致 CPU “被拖累”,CPU 性能降低

        • 一种改进措施:在 Cache 和主存之间使用写缓冲 (Write Buffer)

          • 当一个数据等待被写入主存时,先将其存入写缓冲;

          • 在把数据写入 Cache 和写缓冲后,处理器继续执行命令;

          • 当写主存操作结束后,写缓冲里的数据释放

        • 写穿透 / 直达 Cache 可用写分配或写不分配

      • Write Back (写回、一次性写、回写)
        当一个写操作产生时,新值仅被写到 Cache 中,而不被写入主存

        • 特点

          • 大大降低主存带宽需求

          • 提高系统性能

          • 控制可能很复杂

        • 每个 Cache 行都设置一个修改位 (“dirty bit - 脏位”)

          • 如果修改位为 “0”,则说明对应主存块没有被修改过

          • 如果对应 Cache 行中的主存块被修改,就同时置修改位为 “1”

          • 只有当修改位为 “1” 的块从 cache 中替换出去时,才把它写回主存
            (写不命中时 Cache 里没有有效的对应块,即使写分配也不需要考虑 Cache 中的数据。但是写分配写入 Cache 新的行时,被替换掉的行不能忘了考虑写回)

          • 修改位为 “0” 的块从 cache 中替换出去时,不写回主存

        • 写回 Cache 通常用写分配

    • 写不命中 (Write Miss)
      要写的单元不在 Cache 中时

      • Allocate-on-miss (写分配)

        • 更新主存块中相应单元,再将该主存块装入 Cache;

        • 试图利用空间局部性,但增加了从主存读入一个块的开销

      • No-allocate-on-write (写不分配)

        • 直接写主存,不把被写数据放入 Cache,同时将 Cache 中该行 Valid 位置 0;

        • 减少了从主存读一个块的开销,但没有利用空间局部性

    # Cache 和主存之间的映射

    # 直接映射 / 模映射 (Direct Mapped)

    • 原理:

      • 把主存的每一块映射到一个固定的 Cache 行中。
        Cache 总行数为 N,将存储空间总块数为 M 的主存分成 K = M/N 个组群,每个组群有 N 个块;
        即每个主存地址对应于高速缓存中唯一的地址,也称模映射(本质是哈希)
    • 映射关系为:Cache 行号 = 主存块号 mod Cache 行数

    • 物理地址

      • Tag 位,标记位
        长度 = log _2 (主存中 块 的 数量 / Cache 中 块 的 数量) =log_2 (K)

      • Index 位,Cache 槽号
        长度 = log _2 (Cache 中 块 的 数量) = Log_2 (N)

      • Offset 位,块内地址
        长度 = log _2 (块 的 大小) (按编址单位算)

    • 特点

      • 关联度 = 1

      • 易实现 (求低地址就行了) 电路简单,命中时间短

      • 淘汰 / 替换策略简单

      • 不灵活,Cache 存储空间得不到充分利用,命中率低,抖动多

    # 全相联映射(Fully Associative)

    • 原理:
      主存块可装到 Cache 任一行 / 槽中,称为全相联映射
      Cache 有几个 slot 就最多需要查找多少次 slot,查找时需要一个 slot 一个 slot 比较 Tag 位查找(除非并行同时比较所有 Cache 行的 Tag,硬件实现更复杂)

    • 物理地址

      • Tag 位,标记位
        长度 = log _2 (主存中 块 的 数量)

      • Offset 位,块内地址
        长度 = log _2 (块 的 大小) (按编址单位算)

    • 特点:

      • 关联度 = Cache 的行的数量

      • 块冲突概率低:只要有空闲 Cache 块,都不会发生冲突

      • 实现复杂 (比较逻辑的硬件代价大)、速度慢

    # 组相联映射(N-way Set Associative)

    • n - 路组相联将总行数为 N 的 Cache 分成 S = N/n 组,每一组有 n 行;
      将存储空间总块数为 M 的主存分成 K = M/S = M*n/N 个组群,每个组群有 S 个块;
      把主存块映射到 Cache 固定组的任一行中。即:组间模映射、组内全映射

    • 物理地址

      • Tag 位,标记位
        长度 = log _2 (主存中 块 的 数量 / 组的数量) = log _2 (K)

      • 组号位
        长度 = log _2 (组 的 数量) = log _2 (S)

      • Offset 位,块内地址
        长度 = log _2 (块 的 大小) (按编址单位算)

    • 特点

      • 关联度 = n

      • 结合直接映射和全相联映射的优点。当 Cache 的组数为 1 时,则为全相联映射;

      • 每组两个行(2 路组相联)较常用。在较大容量的 L2 Cahce 和 L3 Cahce 中使用 4 路以上

      • 当每组只有一行时,则为直接映射

      • 当只有一组时,则为全相联映射

    \ No newline at end of file diff --git a/cs/principles-of-computer-composition/cpu/index.html b/cs/principles-of-computer-composition/cpu/index.html index 035e5da1..e9f9a22d 100644 --- a/cs/principles-of-computer-composition/cpu/index.html +++ b/cs/principles-of-computer-composition/cpu/index.html @@ -1 +1 @@ -CPU - 计算机组成原理 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 思维导图

    CPU

    CPU

    # 单周期 CPU

    # 单周期控制器

    • 控制信号在整个指令执行过程中保持不变

    • 真值表反映了指令和控制信号的关系

    • 按照真值表就可以实现控制器!

    # 多周期 CPU

    # 多周期控制器功能描述方式

    • 有限状态机
      采用组合逻辑设计方法用硬连线路 (PLA) 实现

    • 微程序
      用 ROM 存放微程序实现

    # MIPS 各指令的周期数:

    划分关键是:

    1. 所有指令均需要 IF,ID,EX 这三个周期
    2. 需要访存的指令额外需要一个周期 MEM(ID 内的访存不算,所以只有 Load/Store)
    3. 需要写回寄存器的指令额外需要一个周期 WB
    • R-Type (寄存器之间操作):4
      包含 IF,ID,EX,WB 共 4 个周期

    • I-Type

      • 运算指令 (一个寄存器、一个立即数) 如:4
        包含 IF,ID,EX,WB 共 4 个周期

      • Load 指令:4
        包含 IF,ID,EX,WB 共 4 个周期

      • Store 指令:5
        包含 IF,ID,EX,MEM,WB 共 5 个周期

      • 条件分支指令:3
        包含 IF,ID,EX 共 3 个周期

    • J-Type:3
      包含 IF,ID,EX 共 3 个周期

    # 竞争 race :在组合电路中,信号经由不同的路径到达某一会合点的时间有先有后的现象

    # ?冒险:竞争而引起电路输出发生瞬间错误。表现为输出端出现原设计中没有的毛刺

    # 如何避免竞争问题:

    确认地址和数据在第 N 周期结束时已稳定

    • 数据:就绪的早,撤的晚
      在 “写使能” 信号无效前地址和数据保持不变

    • 写使能:就绪的晚,撤的早
      使 “写使能” 信号在一个周期后 (即:第 N+1 周期) 有效

    # 竞争冒险:因为信号传输延迟时间不同,而引起输出逻辑错误的现象

    # 流水线 CPU

    # 特点

    • 适合于大量重复的时序过程

    • 时空图是流水线分析的最好表示方法

    • 单个任务执行时,所需总执行时间没有缩短

    • 多个任务执行时,所有工作可以并行处理:单位时间内完成的工作量大大增加。在不改变单个任务执行时间的前提下,流水线提高了整个执行过程的吞吐率

    • 最大加速比 <流水栈数 (深度)

    • 流水线控制器的数据通路与单周期控制器十分类似

    # 按级划分

    • 三级 CPU 划分:

        1. 取指 IF
        1. 译码 ID
        1. 执行 EX
    • 五级 CPU 划分:

        1. 取指 IF
        1. 译码 ID
        1. 执行 EX
        1. 访存 MEM
        1. 写回 WB

    # 流水线越深(栈数越多)不一定提升性能:

    • (1)每两个相邻阶段之间都需要增加寄存器,因此会增大电路的复杂度,占用芯片的电路面积也就越大,容易增加成本、功耗。

    • (2)增加的寄存器也会使数据的传输时间变得更长,增加了执行指令的额外时间。复杂机制的影响(转移猜测、指令相关性等)。

    • 这些负面影响有可能会抵消增加流水线级数带来的正面影响。

    # n 级流水线 CPU 完成 k 条指令需要周期为:

    n + k -1 (最后一天条指令取指令后还需执行 k-1 个周期)

    # 具有什么特征的指令集有利于流水线执行呢?

    • 指令长度尽量一致:有利于简化取指令和指令译码操作,不一致则需要取一次判断一次要不要再取

      • MIPS 指令都是 32 位,每次取 4 字节的指令,下址计算方便 (PC+4)

      • X86 指令从 1 字节到 17 字节不等,使取指部件极其复杂

    • 指令格式少,且源寄存器位置相同:有利于在指令未知时预取操作数

      • MIPS 指令的 Rs 和 Rt 位置一定,指令译码时就可读 Rs 和 Rt 的值

      • 若位置随指令不同而变,需先译码确定指令后才能取寄存器编号

    • 只有 load / Store 指令才能访问存储器,有利于减少操作步骤,规整流水线

      • MIPS 把 lw/sw 指令地址计算和运算指令执行步骤规整在同一个周期

      • X86 运算类指令操作数有内存数据,需计算地址、访存、执行

    • 数据和指令在内存中需要 “对齐” 存放,有利于减少访存次数和流水线的规整

    # 冒险(Hazards)

    指流水线遇到了无法正确执行后续指令或执行了不该执行的指令的现象

    • 结构冒险 (Struture Hazard)
      一个功能部件同时被多条指令使用的现象,或称为资源冲突,解决策略:

      • 在流水线中插入 “Bubble”(气泡)

        • 控制逻辑复杂

        • 使 n 级流水线中的每条指令都有相同的 n 个阶段,实际上没有执行的阶段(例如 5 级流水线中的 MEM 和 WB)为 NOP (空操作)

        • 设置多个部件,以避免冲突

    • 控制冒险 (Control Hazard)
      分支指令 (Branch) 引起的,亦称为分支冒险或转移冒险 (Branch Hazard),解决策略:

      • 硬件层面解决:
        流水线阻塞 (stall),也称为延迟和 “气泡 Bubble”,控制相当复杂,需要改数据通路

        • 编译器(软件)层面解决:

          • 分支预测 (Predict)

            • 简单 (静态) 预测 (statistic predictor):

              • 总是预测条件不满足,即:继续执行分支指令的后续指令

              • 可加启发式规则:在特定情况下总是预测满足 (taken),其他情况总是预测不满足。如:循环顶 (底) 部分支总是预测为不满足 (满足)。预测准确率可达 65%-85%

            • 动态预测 (dynamic predictor) :

              • 根据程序执行的历史情况,进行动态预测调整,能达 90% 的预测准确率

              • 利用分支历史记录表 BHT(或 BTB、BPB)

              • 现在几乎所有的处理器都采用动态预测

              • 预测器总结

                • 简单的 2 位预测器

                  • 根据 2 位饱和计数器的值选择 Taken 和 ¬Taken

                  • 连续错误两次则改变预测结果

                • 相关预测器

                  • 为每个分支设置 2n 个 2 位的预测器

                  • 根据最近全局发生的 n 次分支从 2n 个 2 位的预测器中选出一个

                • 局部预测器

                  • 为每个分支设置 2n 个 2 位的预测器

                  • 根据最近本身发生的 n 次分支从 2n 个 2 位的预测器中选出一个

                • 竞赛预测器(Tournament Predictor)

                  • 在局部预测器和相关预测器之间动态选择

                  • 采用一个饱和计数器(如 2 位计数器),在两者之间选择

                    • 计数器值为 00、01,选择局部预测器

                    • 计数器值为 10、11,选择相关预测器

            • 注:采用分支预测时,流水线控制必须确保错误预测指令的执行结果不能生效,能从正确的分支地址处重新启动流水线工作

          • 指令静态调度:编译优化指令顺序 / 延迟分支 (Delayed branch)

            • 延迟分支:把分支指令前面与分支指令无关的指令调到分支指令后面执行,也称之为延迟转移

            • 区分:分支延迟
              分支延迟是由于分支指令引起的延迟

            • 减少分支延迟的方法

              • 在流水线中尽早判断出分支转移是否成功

              • 尽早计算出分支目标地址

          • 插入三条 “NOP” 指令

    • 数据冒险 (Data Hazard)
      装入指令 (Load) 引起的,亦称为数据相关 (Data Dependency),解决策略:

      • 采用转发 / 旁路 (Forwarding/Bypassing)(技术现代常用):

        • 可以解决所有 R 指令后间隔的 lw 指令或 R 指令 的数据冒险问题
          任何运算指令总是在 ALU 阶段(EX 执行阶段)产生数据(第 3 个时钟)
          对于同一个 ALU 时,ALU 输出端直接输出到自己的输入端
          或把数据从流水段寄存器中直接取到 ALU 的输入端
          寄存器写 / 读口分别在前 / 后半周期
          使写入被直接读出

        • lw 指令在第 4 个时钟才产生数据,所以 lw 指令后紧接着 R 指令或 lw 指令转发无法解决,只能阻塞

        • 硬件层面解决:

        流水线阻塞 (stall),也称为延迟和 “气泡 Bubble”,控制相当复杂,需要改数据通路

        • 编译器(软件)层面解决:

          • 指令静态调度:编译优化指令顺序,拉大具有数据冒险指令的距离,减少流水线可能产生的停顿

          • 插入 NOP 指令
            最糟糕的做法,没有其他无关指令可以插入了

    \ No newline at end of file +CPU - 计算机组成原理 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 思维导图

    CPU

    CPU

    # 单周期 CPU

    # 单周期控制器

    • 控制信号在整个指令执行过程中保持不变

    • 真值表反映了指令和控制信号的关系

    • 按照真值表就可以实现控制器!

    # 多周期 CPU

    # 多周期控制器功能描述方式

    • 有限状态机
      采用组合逻辑设计方法用硬连线路 (PLA) 实现

    • 微程序
      用 ROM 存放微程序实现

    # MIPS 各指令的周期数:

    划分关键是:

    1. 所有指令均需要 IF,ID,EX 这三个周期
    2. 需要访存的指令额外需要一个周期 MEM(ID 内的访存不算,所以只有 Load/Store)
    3. 需要写回寄存器的指令额外需要一个周期 WB
    • R-Type (寄存器之间操作):4
      包含 IF,ID,EX,WB 共 4 个周期

    • I-Type

      • 运算指令 (一个寄存器、一个立即数) 如:4
        包含 IF,ID,EX,WB 共 4 个周期

      • Load 指令:4
        包含 IF,ID,EX,WB 共 4 个周期

      • Store 指令:5
        包含 IF,ID,EX,MEM,WB 共 5 个周期

      • 条件分支指令:3
        包含 IF,ID,EX 共 3 个周期

    • J-Type:3
      包含 IF,ID,EX 共 3 个周期

    # 竞争 race :在组合电路中,信号经由不同的路径到达某一会合点的时间有先有后的现象

    # ?冒险:竞争而引起电路输出发生瞬间错误。表现为输出端出现原设计中没有的毛刺

    # 如何避免竞争问题:

    确认地址和数据在第 N 周期结束时已稳定

    • 数据:就绪的早,撤的晚
      在 “写使能” 信号无效前地址和数据保持不变

    • 写使能:就绪的晚,撤的早
      使 “写使能” 信号在一个周期后 (即:第 N+1 周期) 有效

    # 竞争冒险:因为信号传输延迟时间不同,而引起输出逻辑错误的现象

    # 流水线 CPU

    # 特点

    • 适合于大量重复的时序过程

    • 时空图是流水线分析的最好表示方法

    • 单个任务执行时,所需总执行时间没有缩短

    • 多个任务执行时,所有工作可以并行处理:单位时间内完成的工作量大大增加。在不改变单个任务执行时间的前提下,流水线提高了整个执行过程的吞吐率

    • 最大加速比 <流水栈数 (深度)

    • 流水线控制器的数据通路与单周期控制器十分类似

    # 按级划分

    • 三级 CPU 划分:

        1. 取指 IF
        1. 译码 ID
        1. 执行 EX
    • 五级 CPU 划分:

        1. 取指 IF
        1. 译码 ID
        1. 执行 EX
        1. 访存 MEM
        1. 写回 WB

    # 流水线越深(栈数越多)不一定提升性能:

    • (1)每两个相邻阶段之间都需要增加寄存器,因此会增大电路的复杂度,占用芯片的电路面积也就越大,容易增加成本、功耗。

    • (2)增加的寄存器也会使数据的传输时间变得更长,增加了执行指令的额外时间。复杂机制的影响(转移猜测、指令相关性等)。

    • 这些负面影响有可能会抵消增加流水线级数带来的正面影响。

    # n 级流水线 CPU 完成 k 条指令需要周期为:

    n + k -1 (最后一天条指令取指令后还需执行 k-1 个周期)

    # 具有什么特征的指令集有利于流水线执行呢?

    • 指令长度尽量一致:有利于简化取指令和指令译码操作,不一致则需要取一次判断一次要不要再取

      • MIPS 指令都是 32 位,每次取 4 字节的指令,下址计算方便 (PC+4)

      • X86 指令从 1 字节到 17 字节不等,使取指部件极其复杂

    • 指令格式少,且源寄存器位置相同:有利于在指令未知时预取操作数

      • MIPS 指令的 Rs 和 Rt 位置一定,指令译码时就可读 Rs 和 Rt 的值

      • 若位置随指令不同而变,需先译码确定指令后才能取寄存器编号

    • 只有 load / Store 指令才能访问存储器,有利于减少操作步骤,规整流水线

      • MIPS 把 lw/sw 指令地址计算和运算指令执行步骤规整在同一个周期

      • X86 运算类指令操作数有内存数据,需计算地址、访存、执行

    • 数据和指令在内存中需要 “对齐” 存放,有利于减少访存次数和流水线的规整

    # 冒险(Hazards)

    指流水线遇到了无法正确执行后续指令或执行了不该执行的指令的现象

    • 结构冒险 (Struture Hazard)
      一个功能部件同时被多条指令使用的现象,或称为资源冲突,解决策略:

      • 在流水线中插入 “Bubble”(气泡)

        • 控制逻辑复杂

        • 使 n 级流水线中的每条指令都有相同的 n 个阶段,实际上没有执行的阶段(例如 5 级流水线中的 MEM 和 WB)为 NOP (空操作)

        • 设置多个部件,以避免冲突

    • 控制冒险 (Control Hazard)
      分支指令 (Branch) 引起的,亦称为分支冒险或转移冒险 (Branch Hazard),解决策略:

      • 硬件层面解决:
        流水线阻塞 (stall),也称为延迟和 “气泡 Bubble”,控制相当复杂,需要改数据通路

        • 编译器(软件)层面解决:

          • 分支预测 (Predict)

            • 简单 (静态) 预测 (statistic predictor):

              • 总是预测条件不满足,即:继续执行分支指令的后续指令

              • 可加启发式规则:在特定情况下总是预测满足 (taken),其他情况总是预测不满足。如:循环顶 (底) 部分支总是预测为不满足 (满足)。预测准确率可达 65%-85%

            • 动态预测 (dynamic predictor) :

              • 根据程序执行的历史情况,进行动态预测调整,能达 90% 的预测准确率

              • 利用分支历史记录表 BHT(或 BTB、BPB)

              • 现在几乎所有的处理器都采用动态预测

              • 预测器总结

                • 简单的 2 位预测器

                  • 根据 2 位饱和计数器的值选择 Taken 和 ¬Taken

                  • 连续错误两次则改变预测结果

                • 相关预测器

                  • 为每个分支设置 2n 个 2 位的预测器

                  • 根据最近全局发生的 n 次分支从 2n 个 2 位的预测器中选出一个

                • 局部预测器

                  • 为每个分支设置 2n 个 2 位的预测器

                  • 根据最近本身发生的 n 次分支从 2n 个 2 位的预测器中选出一个

                • 竞赛预测器(Tournament Predictor)

                  • 在局部预测器和相关预测器之间动态选择

                  • 采用一个饱和计数器(如 2 位计数器),在两者之间选择

                    • 计数器值为 00、01,选择局部预测器

                    • 计数器值为 10、11,选择相关预测器

            • 注:采用分支预测时,流水线控制必须确保错误预测指令的执行结果不能生效,能从正确的分支地址处重新启动流水线工作

          • 指令静态调度:编译优化指令顺序 / 延迟分支 (Delayed branch)

            • 延迟分支:把分支指令前面与分支指令无关的指令调到分支指令后面执行,也称之为延迟转移

            • 区分:分支延迟
              分支延迟是由于分支指令引起的延迟

            • 减少分支延迟的方法

              • 在流水线中尽早判断出分支转移是否成功

              • 尽早计算出分支目标地址

          • 插入三条 “NOP” 指令

    • 数据冒险 (Data Hazard)
      装入指令 (Load) 引起的,亦称为数据相关 (Data Dependency),解决策略:

      • 采用转发 / 旁路 (Forwarding/Bypassing)(技术现代常用):

        • 可以解决所有 R 指令后间隔的 lw 指令或 R 指令 的数据冒险问题
          任何运算指令总是在 ALU 阶段(EX 执行阶段)产生数据(第 3 个时钟)
          对于同一个 ALU 时,ALU 输出端直接输出到自己的输入端
          或把数据从流水段寄存器中直接取到 ALU 的输入端
          寄存器写 / 读口分别在前 / 后半周期
          使写入被直接读出

        • lw 指令在第 4 个时钟才产生数据,所以 lw 指令后紧接着 R 指令或 lw 指令转发无法解决,只能阻塞

        • 硬件层面解决:

        流水线阻塞 (stall),也称为延迟和 “气泡 Bubble”,控制相当复杂,需要改数据通路

        • 编译器(软件)层面解决:

          • 指令静态调度:编译优化指令顺序,拉大具有数据冒险指令的距离,减少流水线可能产生的停顿

          • 插入 NOP 指令
            最糟糕的做法,没有其他无关指令可以插入了

    \ No newline at end of file diff --git a/cs/principles-of-computer-composition/data-path/index.html b/cs/principles-of-computer-composition/data-path/index.html index 31b4a90d..76d94eac 100644 --- a/cs/principles-of-computer-composition/data-path/index.html +++ b/cs/principles-of-computer-composition/data-path/index.html @@ -1 +1 @@ -数据通路 - 计算机组成原理 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 思维导图

    数据通路

    # CPU (Processor)

    # Control 控制器

    指令执行过程中,数据所经过的路径 (包括路径中的部件)—— 指令的执行部件

    • 控制器的基本功能

      • 取指令 (控制指令流出 — PC)

      • 分析指令 (控制指令分析 — IR)

      • 执行指令,发出各种操作命令
        (控制指令执行 — REGs 和 ALU)

      • 确定下一条指令的地址
        (控制指令流向 —PC)

      • 执行环境的建立与保护 (控制执行环境的维护 —FLAGs/PSW)

    # Datapath 数据通路

    对指令进行译码,生成指令对应的控制信号,控制数据通路的动作,能对指令的执行部件发出控制信号 —— 指令的控制部件

    • 数据通路的构成
      由 “操作元件” 和 “存储元件” 通过总线或分散方式连接而成

    • 数据通路的功能
      进行数据存储、处理、传送

    # 数据通路由两类部件组成

    # 组合逻辑元件 or 操作元件

    • 组合逻辑元件的特点:

      • 其输出只取决于当前的输入

      • 所有输入到达后,经过一定的逻辑门延时,输出端改变,并保持到下次改变,不需要时钟信号来定时

    • 包括:

      • 译码器 (Decoder)

        • 多路选择器 (MUX)

        • 加法器 (Adder)

        • 算术逻辑部件 (ALU)

    # 存储元件 or 状态元件

    • 状态 (存储) 元件的特点

      • 具有存储功能,在时钟控制下输入状态被写到电路中,直到下一个时钟到达

      • 输入端状态由时钟决定何时写入,输出端状态随时可读出

    • 定时方式:规定信号何时写入状态元件或何时从状态元件读出

      • 边沿触发 (edge-triggered) 方式:状态单元中的值只在时钟边沿改变。每个时钟周期改变一次

        • 上升沿 (rising edge) 触发:在时钟正跳变时进行读 / 写

        • 下降沿 (falling edge) 触发:在时钟负跳变时进行读 / 写

    • 包括

      • 寄存器 (组)

        • 寄存器 (Register)

          • 写使能 (Write Enable-WE) 信号

            • 0: 时钟边沿到时,输出不变

            • 1: 时钟边沿到时,输出开始变为输入

          • 若每个时钟边沿都写入,则不需 WE 信号

        • 寄存器组 (Register File)

          • 两个读口 (组合逻辑操作):busA 和 busB
            不需要 “读使能”

            • 分别由 RA 和 RB 给出地址。地址 RA 或 RB 有效后,经一个 “取数时间 (AccessTime)”,busA 和 busB 有效
        • 一个写口 (时序逻辑操作):写使能

          • 为 1 且时钟边沿到时,busW 传来的值开始被写入 RW 指定的寄存器中
      • 存储器(理想存储器)
        按存储内容分为:指令存储器 和 数据存储器

        • Data Out:32/64 位读出数据

        • Data In: 32/64 位写入数据

        • Address:读写共用一个 32/64 位地址

        • 读 (组合逻辑操作):地址 Address 有效后,经一个 “取数时间 (AccessTime)”,Data Out 上数据有效

        • 写 (时序逻辑操作):写使能为 1 且时钟 Clk 边沿到来,Data In 传来的值开始被写入 Address 指定的存储单元

    # 元件之间的连接方式

    # 总线式连接

    • 单总线数据通路

    • 三总线数据通路

      • 总线 A、B 分别传送两个源操作数,

      • 总线 C 传送结果

      • 采用双口寄存器

    # 分散式连接

    # 早期累加器型指令数据通路

    \ No newline at end of file +数据通路 - 计算机组成原理 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 思维导图

    数据通路

    # CPU (Processor)

    # Control 控制器

    指令执行过程中,数据所经过的路径 (包括路径中的部件)—— 指令的执行部件

    • 控制器的基本功能

      • 取指令 (控制指令流出 — PC)

      • 分析指令 (控制指令分析 — IR)

      • 执行指令,发出各种操作命令
        (控制指令执行 — REGs 和 ALU)

      • 确定下一条指令的地址
        (控制指令流向 —PC)

      • 执行环境的建立与保护 (控制执行环境的维护 —FLAGs/PSW)

    # Datapath 数据通路

    对指令进行译码,生成指令对应的控制信号,控制数据通路的动作,能对指令的执行部件发出控制信号 —— 指令的控制部件

    • 数据通路的构成
      由 “操作元件” 和 “存储元件” 通过总线或分散方式连接而成

    • 数据通路的功能
      进行数据存储、处理、传送

    # 数据通路由两类部件组成

    # 组合逻辑元件 or 操作元件

    • 组合逻辑元件的特点:

      • 其输出只取决于当前的输入

      • 所有输入到达后,经过一定的逻辑门延时,输出端改变,并保持到下次改变,不需要时钟信号来定时

    • 包括:

      • 译码器 (Decoder)

        • 多路选择器 (MUX)

        • 加法器 (Adder)

        • 算术逻辑部件 (ALU)

    # 存储元件 or 状态元件

    • 状态 (存储) 元件的特点

      • 具有存储功能,在时钟控制下输入状态被写到电路中,直到下一个时钟到达

      • 输入端状态由时钟决定何时写入,输出端状态随时可读出

    • 定时方式:规定信号何时写入状态元件或何时从状态元件读出

      • 边沿触发 (edge-triggered) 方式:状态单元中的值只在时钟边沿改变。每个时钟周期改变一次

        • 上升沿 (rising edge) 触发:在时钟正跳变时进行读 / 写

        • 下降沿 (falling edge) 触发:在时钟负跳变时进行读 / 写

    • 包括

      • 寄存器 (组)

        • 寄存器 (Register)

          • 写使能 (Write Enable-WE) 信号

            • 0: 时钟边沿到时,输出不变

            • 1: 时钟边沿到时,输出开始变为输入

          • 若每个时钟边沿都写入,则不需 WE 信号

        • 寄存器组 (Register File)

          • 两个读口 (组合逻辑操作):busA 和 busB
            不需要 “读使能”

            • 分别由 RA 和 RB 给出地址。地址 RA 或 RB 有效后,经一个 “取数时间 (AccessTime)”,busA 和 busB 有效
        • 一个写口 (时序逻辑操作):写使能

          • 为 1 且时钟边沿到时,busW 传来的值开始被写入 RW 指定的寄存器中
      • 存储器(理想存储器)
        按存储内容分为:指令存储器 和 数据存储器

        • Data Out:32/64 位读出数据

        • Data In: 32/64 位写入数据

        • Address:读写共用一个 32/64 位地址

        • 读 (组合逻辑操作):地址 Address 有效后,经一个 “取数时间 (AccessTime)”,Data Out 上数据有效

        • 写 (时序逻辑操作):写使能为 1 且时钟 Clk 边沿到来,Data In 传来的值开始被写入 Address 指定的存储单元

    # 元件之间的连接方式

    # 总线式连接

    • 单总线数据通路

    • 三总线数据通路

      • 总线 A、B 分别传送两个源操作数,

      • 总线 C 传送结果

      • 采用双口寄存器

    # 分散式连接

    # 早期累加器型指令数据通路

    \ No newline at end of file diff --git a/cs/principles-of-computer-composition/dram-refresh/index.html b/cs/principles-of-computer-composition/dram-refresh/index.html index 0c0b200b..19f6feab 100644 --- a/cs/principles-of-computer-composition/dram-refresh/index.html +++ b/cs/principles-of-computer-composition/dram-refresh/index.html @@ -1 +1 @@ -DRAM 刷新 - 计算机组成原理 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 思维导图

    DRAM刷新

    # 为什么要刷新?

    # DRAM 存储位元的特点

    # 靠电容上的电荷存储效应记忆信息,虽然有 MOS 高电阻 (10121015Ω10^{12} \sim 10^{15} \Omega),仍会泄漏电荷

    # 为保证所存信息的正确性,需要用充电的方法及时使所有位元的电容上电荷恢复到泄漏前的状态

    # DRAM 刷新的有关参数

    # 刷新周期 Trc (refresh cycle)

    • 对同一存储位元连续两次刷新,仍能保证鉴别出原存信息的最大允许间隔时间,即在 Trc 内必须对每个单元刷新一遍

    • 一般为 ms 级,亦称为刷新间隔时间

    • 在 Trc 时间内,应刷新存储芯片中的所有存储位

    • 在 DRAM 芯片的主要性能参数中,都给出 Trc。通常为 2ms、4ms、8ms

    • Trc < Tref

    # 刷新操作周期 Troc (refresh operating cycle)

    刷新一行存储位元所需时间,通常和 DRAM 读写周期 tRC /tWC 相同

    # 刷新操作周期数 Nr (Number of refresh operating cycle)

    存储芯片所有位元刷新一遍所需的刷新操作周期个数。它与芯片的内部结构有关。
    即每次可以刷新一行, Nr 决定于存储矩阵的行数

    # 信息保持时间 Tref (不太重要)

    从信息以电荷形式存入电容,到电荷经过一段时间泄漏,读放仍能鉴别出原存信息的时间

    # 按刷新操作周期的分配方式分类

    # 集中式刷新

    也叫 “批刷新”,从刷新周期 Trc 中抽出连续一段长度的访存周期作为刷新操作周期集中进行刷新

    • 缺点:集中式刷新使 98.976% 的时间用于访存,这期间存储器的效能得以充分发挥,但有 1.024% 的时间,即在 8ms 中有 81.92μs 不允许访存,CPU 要处于等待状态,影响了计算机的工作效率

    • 优点:控制逻辑简单,设计容易实现

    # 分散式刷新

    把集中到一起的不允许访存的刷新时间分散开,每个等分的最后一个访存周期用作刷新操作周期,以完成一行存储位元的刷新,其余时间则用于访存

    • 优点:降低了访存的阻塞时间,提高了计算机的工作效率

    • 缺点:控制逻辑复杂,设计不易实现

    # 透明式刷新

    设一个系统的访存周期是存储器实际访存周期的两倍,并令系统访存周期的前半周期用于访存,后半周期用于刷新

    • 优点:控制简单、设计容易,不需增加多少器件

    • 缺点:存储器的效能仅利用 50%,仅用于低速系统

    \ No newline at end of file +DRAM 刷新 - 计算机组成原理 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 思维导图

    DRAM刷新

    # 为什么要刷新?

    # DRAM 存储位元的特点

    # 靠电容上的电荷存储效应记忆信息,虽然有 MOS 高电阻 (10121015Ω10^{12} \sim 10^{15} \Omega),仍会泄漏电荷

    # 为保证所存信息的正确性,需要用充电的方法及时使所有位元的电容上电荷恢复到泄漏前的状态

    # DRAM 刷新的有关参数

    # 刷新周期 Trc (refresh cycle)

    • 对同一存储位元连续两次刷新,仍能保证鉴别出原存信息的最大允许间隔时间,即在 Trc 内必须对每个单元刷新一遍

    • 一般为 ms 级,亦称为刷新间隔时间

    • 在 Trc 时间内,应刷新存储芯片中的所有存储位

    • 在 DRAM 芯片的主要性能参数中,都给出 Trc。通常为 2ms、4ms、8ms

    • Trc < Tref

    # 刷新操作周期 Troc (refresh operating cycle)

    刷新一行存储位元所需时间,通常和 DRAM 读写周期 tRC /tWC 相同

    # 刷新操作周期数 Nr (Number of refresh operating cycle)

    存储芯片所有位元刷新一遍所需的刷新操作周期个数。它与芯片的内部结构有关。
    即每次可以刷新一行, Nr 决定于存储矩阵的行数

    # 信息保持时间 Tref (不太重要)

    从信息以电荷形式存入电容,到电荷经过一段时间泄漏,读放仍能鉴别出原存信息的时间

    # 按刷新操作周期的分配方式分类

    # 集中式刷新

    也叫 “批刷新”,从刷新周期 Trc 中抽出连续一段长度的访存周期作为刷新操作周期集中进行刷新

    • 缺点:集中式刷新使 98.976% 的时间用于访存,这期间存储器的效能得以充分发挥,但有 1.024% 的时间,即在 8ms 中有 81.92μs 不允许访存,CPU 要处于等待状态,影响了计算机的工作效率

    • 优点:控制逻辑简单,设计容易实现

    # 分散式刷新

    把集中到一起的不允许访存的刷新时间分散开,每个等分的最后一个访存周期用作刷新操作周期,以完成一行存储位元的刷新,其余时间则用于访存

    • 优点:降低了访存的阻塞时间,提高了计算机的工作效率

    • 缺点:控制逻辑复杂,设计不易实现

    # 透明式刷新

    设一个系统的访存周期是存储器实际访存周期的两倍,并令系统访存周期的前半周期用于访存,后半周期用于刷新

    • 优点:控制简单、设计容易,不需增加多少器件

    • 缺点:存储器的效能仅利用 50%,仅用于低速系统

    \ No newline at end of file diff --git a/cs/principles-of-computer-composition/exception-and-interruption/index.html b/cs/principles-of-computer-composition/exception-and-interruption/index.html index d12b98a0..d0df0d5d 100644 --- a/cs/principles-of-computer-composition/exception-and-interruption/index.html +++ b/cs/principles-of-computer-composition/exception-and-interruption/index.html @@ -1 +1 @@ -异常和中断 - 计算机组成原理 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 思维导图

    异常和中断

    # 异常和中断的概念

    # 异常:

    异常来自于 cpu 内部的正在执行的指令本身

    # 中断:

    中断来自于 cpu 外部的硬件触发

    # 统称为 “中断”:

    • 内中断 (内部异常)
      CPU 自己产生而不通过中断请求线请求,皆为不可屏蔽中断

    • 外中断 (外部中断)
      通过中断请求线 INTR 和 NMI 来实现

    # 内部 “异常”:CPU 内发生的意外事件或特殊事件

    # 按发生原因分为硬故障中断和程序性中断两类

    • 硬故障中断:如电源掉电、硬件线路故障等

    • 程序性中断:执行某条指令时发生的 “异常 (Exception)”,如

      • 溢出

      • 缺页

      • 越界

      • 越权

      • 非法指令

      • 除数为 0

      • 断点

      • 单步跟踪

      • 系统调用

    # 按处理方式分 故障 (fault)、自陷 (Trap) 和终止 (Abort) 三类

    • 故障:执行指令引起的异常事件,如 溢出 / 缺页 / 访问超时
      属于程序性中断

    • 自陷:预先安排的事件,如 单步跟踪、系统调用等 (自愿中断)
      属于程序性中断

    • 终止:硬故障事件,机器将终止,调出中断程序重启操作系统
      属于硬故障中断

    # 外部 “中断”:

    # INTR(Interrupt Require):

    可屏蔽中断 (外设中断源引起的中断)
    CPU 可以响应、也可以不响应
    响应与否受两个控制:

    • 本身的屏蔽位

    • CPU 标志寄存器中的中断允许位 (IF)

      • 置 IF=1:指令 STI—— 开中断

      • 清 IF=0:指令 CLI—— 关中断

    # NMI(Nonmaskable Interrupt):

    不可屏蔽中断 (重要或紧急的硬件故障),属于类型 2 中断
    CPU 必须无条件响应
    如电源掉电、存储器读写出错、总线奇偶位出错等

    # 处理器中的异常处理机制

    # 关中断:

    使处理器处于 “禁止中断” 状态,以防止新异常 (或中断) 破坏断点和现场

    # 保护断点和程序状态:

    将断点和程序状态保存到堆栈或特殊寄存器中

    • PC→堆栈 或 EPC (专门存放断点的寄存器)

    • PSWR →堆栈 或 EPSWR (专门保存程序状态的寄存器)

      • PSW—Program Status Word:程序状态字,包括条件码、中断码、状态位等

      • PSWR (PSW 寄存器):用于存放程序状态字寄存器。如 X86 的 FLAGS

    # 识别异常事件:软件识别和硬件识别

    • 软件识别 (非向量中断、MIPS 采用)
      设置一个异常状态寄存器 (如 MIPS 中为 Cause 寄存器),用于记录异常原因。操作系统用一个异常处理程序,按优先级顺序查询异常状态寄存器,识别出异常事件,转入相应的中断服务程序执行

      • 数据通路中需增加两个寄存器

        • EPC:32 位,用于存放断点 (异常处理后返回到的指令的地址)

          • 写入 EPC 的断点可能是正在执行的指令 (故障 时):把 PC 的值减 4 后送到 EPC

          • 写入 EPC 的断点可能是下条指令 (自陷 和 中断 时):直接送 PC 到 EPC

        • Cause:32 位,记录异常原因

          • 处理的异常类型:例如未定义指令 (Cause=10)、算术溢出 (Cause=12)
      • 需要加入两个寄存器的 “写使能” 控制信号

        • EPCWr:在保存断点时该信号有效,使断点 PC 写入 EPC

        • CauseWr:在处理器发现异常 (如非法指令等) 时,该信号有效,使异常类型被写到 Cause 寄存器

      • 需要一个控制信号 IntCause:选择正确的值写入到 Cause 中

      • 需要将异常查询程序的入口地址 (MIPS 为 0x8000 0180) 写入 PC:在原来 PCSource 控制的多路复用器中再增加一路,其输入为 0x8000 0180

    • 硬件识别 (向量中断、80x86 采用)
      用专门的硬件查询电路按优先级顺序识别异常,得到 “中断类型号”,到中断向量表中读取对应的中断服务程序的入口地址

      • 中断向量表(中断入口地址表),位于 0000H~03FFH

      • 含 256 个中断向量,每一项就是对应的中断服务程序的入口地址,每个向量长度为四个字节 (CS:IP)
        256 = 41616/4

        • 中断向量地址 = 中断类型号 ×4

        • 若已知一个中断类型码,需要通过两次地址转换:

          • 中断类型码到中断向量表地址

          • 中断向量表地址到中断处理程序地址

    # 流水线方式下的异常和中断处理

    # 异常和中断都会改变程序的执行顺序

    # 某条指令发现异常时,后面多条指令已被取到流水线中执行

    处理时需要首先清除当前指令以及后面的所有已在流水线中的指令

    # 如果是内部异常,n 级流水线中同时有 n 条指令,哪一条发生异常?

    根据异常发生的流水段可确定是哪条指令,因为各类异常发生的流水段不同

    • “溢出” 在 EX 段检出

    • “无效指令” 在 ID 段检出

    • “除数为 0” 在 ID 段检出

    • “无效指令地址” 在 IF 段检出

    • “无效数据地址” 在 Load/Store 指令的 EX 段检出

    # 如果是外部中断,外部中断与特定指令无关,如何确定处理点?

    • 可在 IF 段或 WB 段中进行中断查询,需要保证当前 WB 段的指令能正确完成,并在有中断发生时,确保下个时钟开始执行中断服务程序

    # 检测到异常时,指令已经取出多条,当前 PC 的值已不是断点,怎么办?

    • 指令地址存放在流水段寄存器中,把这个地址送到 EPC 保存,以实现精确中断

    • 精确中断 (precise interrupt) 或精确异常 (precise exception)

      • 对于发生中断或者异常的单条指令,中断被称为 “精确” 当且仅当满足:
        所有该指令之前的指令都已经提交其状态;
        所有后继指令 (包括中断的指令) 没有改变任何机器的状态

      • 在流水线计算机中,异常或中断与其产生的指令始终能精确地关联起来

    \ No newline at end of file +异常和中断 - 计算机组成原理 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 思维导图

    异常和中断

    # 异常和中断的概念

    # 异常:

    异常来自于 cpu 内部的正在执行的指令本身

    # 中断:

    中断来自于 cpu 外部的硬件触发

    # 统称为 “中断”:

    • 内中断 (内部异常)
      CPU 自己产生而不通过中断请求线请求,皆为不可屏蔽中断

    • 外中断 (外部中断)
      通过中断请求线 INTR 和 NMI 来实现

    # 内部 “异常”:CPU 内发生的意外事件或特殊事件

    # 按发生原因分为硬故障中断和程序性中断两类

    • 硬故障中断:如电源掉电、硬件线路故障等

    • 程序性中断:执行某条指令时发生的 “异常 (Exception)”,如

      • 溢出

      • 缺页

      • 越界

      • 越权

      • 非法指令

      • 除数为 0

      • 断点

      • 单步跟踪

      • 系统调用

    # 按处理方式分 故障 (fault)、自陷 (Trap) 和终止 (Abort) 三类

    • 故障:执行指令引起的异常事件,如 溢出 / 缺页 / 访问超时
      属于程序性中断

    • 自陷:预先安排的事件,如 单步跟踪、系统调用等 (自愿中断)
      属于程序性中断

    • 终止:硬故障事件,机器将终止,调出中断程序重启操作系统
      属于硬故障中断

    # 外部 “中断”:

    # INTR(Interrupt Require):

    可屏蔽中断 (外设中断源引起的中断)
    CPU 可以响应、也可以不响应
    响应与否受两个控制:

    • 本身的屏蔽位

    • CPU 标志寄存器中的中断允许位 (IF)

      • 置 IF=1:指令 STI—— 开中断

      • 清 IF=0:指令 CLI—— 关中断

    # NMI(Nonmaskable Interrupt):

    不可屏蔽中断 (重要或紧急的硬件故障),属于类型 2 中断
    CPU 必须无条件响应
    如电源掉电、存储器读写出错、总线奇偶位出错等

    # 处理器中的异常处理机制

    # 关中断:

    使处理器处于 “禁止中断” 状态,以防止新异常 (或中断) 破坏断点和现场

    # 保护断点和程序状态:

    将断点和程序状态保存到堆栈或特殊寄存器中

    • PC→堆栈 或 EPC (专门存放断点的寄存器)

    • PSWR →堆栈 或 EPSWR (专门保存程序状态的寄存器)

      • PSW—Program Status Word:程序状态字,包括条件码、中断码、状态位等

      • PSWR (PSW 寄存器):用于存放程序状态字寄存器。如 X86 的 FLAGS

    # 识别异常事件:软件识别和硬件识别

    • 软件识别 (非向量中断、MIPS 采用)
      设置一个异常状态寄存器 (如 MIPS 中为 Cause 寄存器),用于记录异常原因。操作系统用一个异常处理程序,按优先级顺序查询异常状态寄存器,识别出异常事件,转入相应的中断服务程序执行

      • 数据通路中需增加两个寄存器

        • EPC:32 位,用于存放断点 (异常处理后返回到的指令的地址)

          • 写入 EPC 的断点可能是正在执行的指令 (故障 时):把 PC 的值减 4 后送到 EPC

          • 写入 EPC 的断点可能是下条指令 (自陷 和 中断 时):直接送 PC 到 EPC

        • Cause:32 位,记录异常原因

          • 处理的异常类型:例如未定义指令 (Cause=10)、算术溢出 (Cause=12)
      • 需要加入两个寄存器的 “写使能” 控制信号

        • EPCWr:在保存断点时该信号有效,使断点 PC 写入 EPC

        • CauseWr:在处理器发现异常 (如非法指令等) 时,该信号有效,使异常类型被写到 Cause 寄存器

      • 需要一个控制信号 IntCause:选择正确的值写入到 Cause 中

      • 需要将异常查询程序的入口地址 (MIPS 为 0x8000 0180) 写入 PC:在原来 PCSource 控制的多路复用器中再增加一路,其输入为 0x8000 0180

    • 硬件识别 (向量中断、80x86 采用)
      用专门的硬件查询电路按优先级顺序识别异常,得到 “中断类型号”,到中断向量表中读取对应的中断服务程序的入口地址

      • 中断向量表(中断入口地址表),位于 0000H~03FFH

      • 含 256 个中断向量,每一项就是对应的中断服务程序的入口地址,每个向量长度为四个字节 (CS:IP)
        256 = 41616/4

        • 中断向量地址 = 中断类型号 ×4

        • 若已知一个中断类型码,需要通过两次地址转换:

          • 中断类型码到中断向量表地址

          • 中断向量表地址到中断处理程序地址

    # 流水线方式下的异常和中断处理

    # 异常和中断都会改变程序的执行顺序

    # 某条指令发现异常时,后面多条指令已被取到流水线中执行

    处理时需要首先清除当前指令以及后面的所有已在流水线中的指令

    # 如果是内部异常,n 级流水线中同时有 n 条指令,哪一条发生异常?

    根据异常发生的流水段可确定是哪条指令,因为各类异常发生的流水段不同

    • “溢出” 在 EX 段检出

    • “无效指令” 在 ID 段检出

    • “除数为 0” 在 ID 段检出

    • “无效指令地址” 在 IF 段检出

    • “无效数据地址” 在 Load/Store 指令的 EX 段检出

    # 如果是外部中断,外部中断与特定指令无关,如何确定处理点?

    • 可在 IF 段或 WB 段中进行中断查询,需要保证当前 WB 段的指令能正确完成,并在有中断发生时,确保下个时钟开始执行中断服务程序

    # 检测到异常时,指令已经取出多条,当前 PC 的值已不是断点,怎么办?

    • 指令地址存放在流水段寄存器中,把这个地址送到 EPC 保存,以实现精确中断

    • 精确中断 (precise interrupt) 或精确异常 (precise exception)

      • 对于发生中断或者异常的单条指令,中断被称为 “精确” 当且仅当满足:
        所有该指令之前的指令都已经提交其状态;
        所有后继指令 (包括中断的指令) 没有改变任何机器的状态

      • 在流水线计算机中,异常或中断与其产生的指令始终能精确地关联起来

    \ No newline at end of file diff --git a/cs/principles-of-computer-composition/instruction-system/index.html b/cs/principles-of-computer-composition/instruction-system/index.html index 18c98319..4358d87b 100644 --- a/cs/principles-of-computer-composition/instruction-system/index.html +++ b/cs/principles-of-computer-composition/instruction-system/index.html @@ -1 +1 @@ -指令系统 - 计算机组成原理 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 思维导图

    指令系统

    # 两大类型

    # CISC

    • (1) 指令系统复杂
    • (2) 指令周期长
    • (3) 各种指令都能访问存储器
    • (4) 有专用寄存器(寻址快)
    • (5) 采用微程序控制
    • (6) 难以进行编译优化生成高效目标代码
    • (7) 指令长度不等长,
      变长指令字 (Expanding Opcodes)

    # RISC

    • (1) 简化的指令系统
    • (2) 指令周期短
    • (3) 有专门的访存指令如 load、store,
      以寄存器 - 寄存器方式工作 (三地址指令是 RISC 风格)
    • (4) 采用大量通用寄存器,以减少访存次数
    • (5) 采用组合逻辑电路控制,不用或少用微程序控制
    • (6) 采用优化的编译系统,力求有效地支持高级语言程序
    • (7) 指令长度几乎全部相等,
      定长指令字 (Fixed Length Opcodes)

    # 设计原则

    # 完备性

    # 有效性

    • 每条指令的操作码只能是一个

    # 规整性

    # 兼容性

    • 向上兼容
    • 向下兼容

    # 指令系统的结构

    # 堆栈结构:零地址指令

    # 累加器结构:一地址指令

    # 通用寄存器结构:二、三地址指令

    # 指令 Command

    指令是指挥计算机实现某个基本操作的命令

    # MIPS 三种指令类型

    • R-Type (用于寄存器 Register)
      两个操作数都是寄存器的运算指令。如:sub rd, rs, rt

    • I-Type

      • 运算指令:一个寄存器、一个立即数。如:
        ori rt, rs, imm16
      • Load 和 Store 指令。如:
        lw rt, rs, imm16
      • 条件分支指令。如:
        beq rs, rt, imm16
    • J-Type
      无条件跳转指令。如:j target

    注意:J 指令是绝对寻址 (伪直接寻址),不是相对寻址!

    # 一条指令

    • 操作码

    • 地址码

      • 寻址方式码
      • 操作数

    # 地址码结构

    • 零地址指令

      • (1) 无需操作数。如:空操作/停机等
      • (2) 所需操作数为默认的。如:堆栈等
    • 一地址指令
      其地址既是源操作数地址,也是存放结果地址

      • (1) 单目运算:如:取反/取负等
      • (2) 双目运算:另一操作数为默认的 如:累加器等
    • 二地址指令 (最常用)
      分别存放双目运算中两个源操作数地址,并将其中一个地址作为存放结果地址

    • 三地址指令 (RISC 风格)
      三个地址分别为双目运算中两个源操作数地址和一个结果地址

    • 多地址指令
      用于成批数据处理的指令,如:向量指令 等

    # 一个较完善的指令系统应该包括

    # 数据传送指令

    # 输入输出指令

    # 算术运算指令

    • 单目运算
    • 双面运算

    # 逻辑运算指令

    • 单目运算
    • 双面运算

    # 系统控制指令

    # 程序控制指令

    # 寻址

    # 寻址方式出现的目的

    • 扩大访存范围

    • 提高访问数据的灵活性和有效性

    • 支持软件技术的发展:多道程序设计

    # 指令系统中的两类寻址

    • 寻找操作数
      较复杂

      • 操作数的来源:

        • 寄存器
        • 外设端口
        • 主 (虚) 存
        • 堆栈
      • 操作数的数据结构:

        • 字节
        • 半字
        • 双字
        • 一维表
    • 寻找 (下一条) 指令
      较简单

      • 顺序寻址
        PC 增值

      • 跳跃寻址
        跳转 :操作数就是目标语句块第一条指令的地址
        相对寻址

        • jump
        • branch
        • call
        • return

    # 寻址方式的确定

    • 在操作码中隐含寻址方式
    • 专门的寻址方式位显式给出

    # 基本寻址方式

    # 一、立即数寻址

    “不寻址”,源操作数直接在指令中

    • 1. 指令执行时间很短,无需访存
    • 2. 操作数的大小受地址字段长度的限制
    • 3. 广泛使用

    # 二、存储器直接寻址

    操作数在存储器中,指令地址字段直接给出操作数在存储器中的地址
    无括号包围

    • 1. 处理简单、直接
    • 2. 寻址空间受到指令的地址字段长度限制
    • 3. 较少使用,8 位计算机和一些 16 位计算机

    # 三、寄存器直接寻址

    操作数在寄存器中,指令地址字段直接给出存放操作数的寄存器编号
    汇编语言:一个括号包围一个寄存器 (编号)

    • 1. 只需要很短的地址字段
    • 2. 无需访存,指令执行速度快
    • 3. 地址范围有限,可以编程使用的通用寄存器不多
    • 4. 使用最多,是提高性能的常用手段

    # 四、存储器间接寻址

    操作数和操作数地址都在存储器中指令地址字段直接给出操作数地址在存储器中的地址
    汇编语言:双重括号包围一个主存地址

    • 1. 寻址空间大,灵活,便于编程
    • 2. 至少需要两次访存才能取到操作数
    • 3. 执行速度慢
    • 4. 用的非常少

    # 五、寄存器间接寻址

    汇编语言:双重括号包围一个寄存器 (编号)

    • 操作数在存储器中
    • 操作数地址在寄存器中
    • 指令地址字段给出的寄存器的内容是操作数在存储器中的地址
    • 1. 比存储器间接寻址少访问存储器一次
    • 2. 寻址空间大,使用比较普遍

    # 六、偏移寻址

    EA = (R) + A
    A 由立即数的形式在指令中给出
    EA (Effective Address):操作数实际地址

    • 相对寻址:EA = (PC) + A
      相对于当前指令处位移量为 A 的单元

    • 基址寻址:EA = (B) + A
      相对于基址 (B) 处位移量为 A 的单元

      • 对于一道程序,基址是不变的,程序中的所有地址都是相对于基址变化的
      • 在基址寻址中,偏移量位数较短
      • 基址寻址立足于面向系统,主要是解决程序逻辑空间与存储器物理空间的无关性
      • 例如:实现程序重定位
    • 变址寻址:EA = A + (I)
      相对于形式地址 A 处位移量为 (I) 的单元

      • 对于变址寻址,形式地址给出的是一个存储器地址基准,变址寄存器存放的是相对于该基准地址的偏移量
      • 而在变址寻址中,偏移量位数足以表示整个存储空间
      • 而变址寻址立足于用户,主要是为编写高效访问一片存储空间的程序
      • 例如:实现线性表元素 (数组) 的存取

    # 七、堆栈寻址

    EA = 栈顶

    • 堆栈的结构:

      • 一段连续的内存区域
      • 栈底
      • 栈顶
    • 堆栈指针 (SP):一个特殊寄存器,指向栈顶

    • 堆栈操作

      • PUSH (从寄存器到堆栈)
      • POP (从堆栈到寄存器)

    # 混合寻址方式

    也会出现多个括号包围

    \ No newline at end of file +指令系统 - 计算机组成原理 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 思维导图

    指令系统

    # 两大类型

    # CISC

    • (1) 指令系统复杂
    • (2) 指令周期长
    • (3) 各种指令都能访问存储器
    • (4) 有专用寄存器(寻址快)
    • (5) 采用微程序控制
    • (6) 难以进行编译优化生成高效目标代码
    • (7) 指令长度不等长,
      变长指令字 (Expanding Opcodes)

    # RISC

    • (1) 简化的指令系统
    • (2) 指令周期短
    • (3) 有专门的访存指令如 load、store,
      以寄存器 - 寄存器方式工作 (三地址指令是 RISC 风格)
    • (4) 采用大量通用寄存器,以减少访存次数
    • (5) 采用组合逻辑电路控制,不用或少用微程序控制
    • (6) 采用优化的编译系统,力求有效地支持高级语言程序
    • (7) 指令长度几乎全部相等,
      定长指令字 (Fixed Length Opcodes)

    # 设计原则

    # 完备性

    # 有效性

    • 每条指令的操作码只能是一个

    # 规整性

    # 兼容性

    • 向上兼容
    • 向下兼容

    # 指令系统的结构

    # 堆栈结构:零地址指令

    # 累加器结构:一地址指令

    # 通用寄存器结构:二、三地址指令

    # 指令 Command

    指令是指挥计算机实现某个基本操作的命令

    # MIPS 三种指令类型

    • R-Type (用于寄存器 Register)
      两个操作数都是寄存器的运算指令。如:sub rd, rs, rt

    • I-Type

      • 运算指令:一个寄存器、一个立即数。如:
        ori rt, rs, imm16
      • Load 和 Store 指令。如:
        lw rt, rs, imm16
      • 条件分支指令。如:
        beq rs, rt, imm16
    • J-Type
      无条件跳转指令。如:j target

    注意:J 指令是绝对寻址 (伪直接寻址),不是相对寻址!

    # 一条指令

    • 操作码

    • 地址码

      • 寻址方式码
      • 操作数

    # 地址码结构

    • 零地址指令

      • (1) 无需操作数。如:空操作/停机等
      • (2) 所需操作数为默认的。如:堆栈等
    • 一地址指令
      其地址既是源操作数地址,也是存放结果地址

      • (1) 单目运算:如:取反/取负等
      • (2) 双目运算:另一操作数为默认的 如:累加器等
    • 二地址指令 (最常用)
      分别存放双目运算中两个源操作数地址,并将其中一个地址作为存放结果地址

    • 三地址指令 (RISC 风格)
      三个地址分别为双目运算中两个源操作数地址和一个结果地址

    • 多地址指令
      用于成批数据处理的指令,如:向量指令 等

    # 一个较完善的指令系统应该包括

    # 数据传送指令

    # 输入输出指令

    # 算术运算指令

    • 单目运算
    • 双面运算

    # 逻辑运算指令

    • 单目运算
    • 双面运算

    # 系统控制指令

    # 程序控制指令

    # 寻址

    # 寻址方式出现的目的

    • 扩大访存范围

    • 提高访问数据的灵活性和有效性

    • 支持软件技术的发展:多道程序设计

    # 指令系统中的两类寻址

    • 寻找操作数
      较复杂

      • 操作数的来源:

        • 寄存器
        • 外设端口
        • 主 (虚) 存
        • 堆栈
      • 操作数的数据结构:

        • 字节
        • 半字
        • 双字
        • 一维表
    • 寻找 (下一条) 指令
      较简单

      • 顺序寻址
        PC 增值

      • 跳跃寻址
        跳转 :操作数就是目标语句块第一条指令的地址
        相对寻址

        • jump
        • branch
        • call
        • return

    # 寻址方式的确定

    • 在操作码中隐含寻址方式
    • 专门的寻址方式位显式给出

    # 基本寻址方式

    # 一、立即数寻址

    “不寻址”,源操作数直接在指令中

    • 1. 指令执行时间很短,无需访存
    • 2. 操作数的大小受地址字段长度的限制
    • 3. 广泛使用

    # 二、存储器直接寻址

    操作数在存储器中,指令地址字段直接给出操作数在存储器中的地址
    无括号包围

    • 1. 处理简单、直接
    • 2. 寻址空间受到指令的地址字段长度限制
    • 3. 较少使用,8 位计算机和一些 16 位计算机

    # 三、寄存器直接寻址

    操作数在寄存器中,指令地址字段直接给出存放操作数的寄存器编号
    汇编语言:一个括号包围一个寄存器 (编号)

    • 1. 只需要很短的地址字段
    • 2. 无需访存,指令执行速度快
    • 3. 地址范围有限,可以编程使用的通用寄存器不多
    • 4. 使用最多,是提高性能的常用手段

    # 四、存储器间接寻址

    操作数和操作数地址都在存储器中指令地址字段直接给出操作数地址在存储器中的地址
    汇编语言:双重括号包围一个主存地址

    • 1. 寻址空间大,灵活,便于编程
    • 2. 至少需要两次访存才能取到操作数
    • 3. 执行速度慢
    • 4. 用的非常少

    # 五、寄存器间接寻址

    汇编语言:双重括号包围一个寄存器 (编号)

    • 操作数在存储器中
    • 操作数地址在寄存器中
    • 指令地址字段给出的寄存器的内容是操作数在存储器中的地址
    • 1. 比存储器间接寻址少访问存储器一次
    • 2. 寻址空间大,使用比较普遍

    # 六、偏移寻址

    EA = (R) + A
    A 由立即数的形式在指令中给出
    EA (Effective Address):操作数实际地址

    • 相对寻址:EA = (PC) + A
      相对于当前指令处位移量为 A 的单元

    • 基址寻址:EA = (B) + A
      相对于基址 (B) 处位移量为 A 的单元

      • 对于一道程序,基址是不变的,程序中的所有地址都是相对于基址变化的
      • 在基址寻址中,偏移量位数较短
      • 基址寻址立足于面向系统,主要是解决程序逻辑空间与存储器物理空间的无关性
      • 例如:实现程序重定位
    • 变址寻址:EA = A + (I)
      相对于形式地址 A 处位移量为 (I) 的单元

      • 对于变址寻址,形式地址给出的是一个存储器地址基准,变址寄存器存放的是相对于该基准地址的偏移量
      • 而在变址寻址中,偏移量位数足以表示整个存储空间
      • 而变址寻址立足于用户,主要是为编写高效访问一片存储空间的程序
      • 例如:实现线性表元素 (数组) 的存取

    # 七、堆栈寻址

    EA = 栈顶

    • 堆栈的结构:

      • 一段连续的内存区域
      • 栈底
      • 栈顶
    • 堆栈指针 (SP):一个特殊寄存器,指向栈顶

    • 堆栈操作

      • PUSH (从寄存器到堆栈)
      • POP (从堆栈到寄存器)

    # 混合寻址方式

    也会出现多个括号包围

    \ No newline at end of file diff --git a/cs/principles-of-computer-composition/memory/index.html b/cs/principles-of-computer-composition/memory/index.html index e13b2e1f..878a846a 100644 --- a/cs/principles-of-computer-composition/memory/index.html +++ b/cs/principles-of-computer-composition/memory/index.html @@ -1 +1 @@ -存储器 - 计算机组成原理 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 思维导图

    存储器

    # 基本术语

    # 记忆单元 / 存储位元 / 位元 (Cell)

    具有两种稳态的能够表示二进制数 0 和 1 的物理器件

    # 存储单元 / 编址单位 / 寻址单位 (Addressing Unit)

    存储器中具有相同地址的那些位构成一个存储单元,又称为一个编址单位

    # 存储体 / 存储矩阵 / 存储阵列 (Bank)

    所有存储单元构成一个存储阵列

    # 存储器地址寄存器 (Memory Address Register – MAR)

    用于存放主存单元地址的寄存器

    # 存储器数据寄存器 (Memory Data Register – MDR)

    用于存放主存单元中数据的寄存器

    # 机器字长

    运算器中参加运算的寄存器的位数,即:数据通路的宽度

    # 存储字

    存储芯片中一个读写单位,一般等于存储器的数据线宽度
    32bit 系统是 32 位,64 位系统是 64 位……

    # 编址方式 (Addressing Mode):对存储体中各存储单元进行编号的方法

    • 按字节编址 (现代计算机基本上都按字节编址)

    • 按字编址 (早期有机器按字编址)

    # 传输单位:一次读写存储器的数据单位

    • 对主存而言,指一次从主存读出或写入的数据位数

    • 对辅存而言,数据通常按块传输,传输单位为块

    # 计算机存储层次结构

    #

    # 从上到下一级一级查找,找到后再从下往上一级一级存

    • CPU 运行时,需要的操作数大部分来自寄存器

    • 需从 (向) 存储器中取 (存) 数时,先访问 cache;若在,取自 cache

    • 若不在 cache,则访问 RAM;若在,则取自 RAM (还会保存到 cache 上)

    • 若不在 RAM,则访问磁盘,操作数从磁盘中读出→RAM →cache

    # 按存储器中信息的可保存性分类

    # 断电后是否丢失数据

    • 挥发性存储器 (易失性存储器)
      特点:断电后,信息即丢失
      SRAM、DRAM

    • 非挥发性存储器 (非易失性 / 永久性存储器)
      特点:断电后,信息不丢失
      ROM、磁盘、闪存

    # 读出后是否保持数据

    • 破坏性存储器
      特点:读出时原存信息被破坏,需重写
      DRAM

    • 非破坏性存储器
      特点:读出时,原存信息不被破坏
      SRAM

    # 存储器技术指标

    # 存取时间 TA 与 存储周期 TS

    • 存取时间 TA:启动一次存储器操作到完成该操作所用时间

      • 读:从启动读命令到读出的数据送 MDR 所需的时间

      • 写:从启动到将 MDR 内容写入指定存储单元的时间
        亦称访问时间,是反映存储器速度的指标,决定了 CPU 发出读 / 写命令后必须等待的时间

    • 存储周期 TS:亦称存取周期或访问周期
      两连续两次启动同一存储器进行存取操作所需的最小时间间隔次启动包括:两次读,两次写,一次读或一次写

    • TA<TS :
      TA、TS 的差别依赖于存储信息的器件和电路,存储介质和控制线路需要恢复时间
      破坏性读出须重写;非破坏性读出不须重写

    # 存取宽度 与 存储带宽

    • 存取宽度:一次访存操作可存取的数据位数或字节数 (银河 - I 的存取宽度是 64 位)

    • 存储带宽:每秒传输的最大数据量 (位 / 秒)

    # MTBF(Mean Time Between Failure)

    • MTBF 表示两次故障之间的平均时间间隔

    • 采用纠错编码技术进行容错处理

    # 主存和 CPU 之间的两种通信方式

    # 异步方式过程 (需握手信号)

    • 读操作

    • CPU 送地址到地址线,主存进行地址译码

    • CPU 发读命令,然后等待存储器发回 “完成” 信号

    • 主存收到读命令后开始读数,完成后发 “完成” 信号给 CPU

    • CPU 接收到 “完成” 信号,从数据线取数

    # 同步方式的特点

    • CPU 和主存由统一时钟信号控制,无需应答信号

    • 主存总是在确定的时间内准备好数据

    • CPU 送出地址和读命令后,总是在确定的时间取数据

    • 存储器芯片必须支持同步方式

    # 主存和 CPU 的连接

    #

    # 地址线的连接

    • CPU 地址线数决定了整个主存空间的寻址范围

    • CPU 地址线数 > 存储芯片地址引脚线

    • 通常将 CPU 地址线的低位和存储芯片地址线相连,高位用作字扩展时的片选信号的译码

    # 数据线的连接

    • CPU 数据线数决定了一次可读写的最大数据宽度

    • CPU 数据线数 > 存储芯片数据引脚线

    • 通常将 CPU 数据线连到多个位扩展的芯片中,使扩展后的位数与 CPU 数据线数相等

    # 控制线的连接

    • 若 CPU 读 / 写命令线和存储芯片的读 / 写控制线是一根,且电平信号一致,则可直接相连

    • 若 CPU 读 / 写命令线分开,则需分别进行连接

    • ROM read only 不需要连接写使能 WR

    # 按在计算机中的作用分类

    # 寄存器型存储器

    • 亦称便笺式存储器,由寄存器构成,封装在 CPU 内

    • 用于存放当前正在执行的指令和使用的数据

    • 用触发器实现,速度快,容量小 O (KB)

    # 高速缓冲存储器 (Cache)

    • 位于 CPU 内部或位于主存和 CPU 之间

    • 用来存放当前正在执行的局部程序段和数据

    • 用 SRAM 实现,速度 O (ns)、容量 O (MB)

    # 主存储器

    • 位于 CPU 之外

    • 用来存放已被启动的程序及所用数据

    • 用 DRAM 实现,速度 O (ns)、容量 O (GB)

    # 辅助存储器

    • 不能由 CPU 的指令直接访问

    • 用来存放暂不运行的程序、数据或存档文件

    • 用磁、光等存储器件实现,容量大,但速度慢 O (ms)

    # 其它功能的存储器

    • 控存 CM:存储微程序代码

    • 表格存储器:为加快 CPU 处理而设置 (倒数表、函数表)

    • 字库和数据缓冲存储器:显示和印刷输出设备中

    # 按存储介质分类

    # 半导体存储器 (SCM)

    • 速度快,用作内存

    • 记忆原理:双稳态触发器、电容 (静态、动态)

    # 磁表面存储器 MSM

    • 用陶瓷、非磁性金属或塑料作载磁体,磁化后具有两种不同的剩磁状态记录信息 “1” 和 “0”

    • 非易失,容量大并且每位价格低 。用作外存

    # 光盘存储器 ODM

    • 有机玻璃作载磁体,利用磁化、晶态 / 非晶态表示信息

    • 非易失,可靠性高,保存时间长,容量大且易于更换

    • 存储速度比硬盘低一个数量级

    # 铁电存储器 FeM (FRAM)、相变存储器 PCM、阻变存储器 ReRAM、……

    • 掉电数据不丢失 (non-volatile),速度快

    # 按存储方式分类

    # 随机访问存储器 (RAM)

    • 存储器任意单元可随时访问且访问所需时间基本相同

    • 访问时间与存储单元所处的物理位置无关

    • 速度快 (ns)

    • 主存和 Cache

    # 只读存储器 (ROM)(Read Only Memory)

    • 只读是指正常工作时只读,能随机读出,不能随机写入

    • MROM:只能读,不能修改

    • PROM:只能修改一次,修改后只读

    • 可多次改写 ROM:

      • 光擦除电可编程只读存储器,亦称 UV EPROM
        ——s 级别

      • 电擦除电可编程只读存储器 E2PROM
        ——s 级别

      • 块擦除电可编程的快擦写存储器 Flash Memory
        ——ms 级别(闪存)

    # 顺序存储器 (SAS)(Sequential Access Memory)

    • 存储时以数据块为单位存储,顺序地记录在存储介质上

    • 数据按顺序从存储载体的始端读出或写入,存取时间的长短与数据所在位置有关

    • 速度慢、容量大、成本低,用作后援外存

    # 磁带、电荷耦合器件 CCD、VCD

    # 常用半导体存储器(常考)

    # 随机存取存储器 (RAM)

    • SRAM 静态随机存取存储器 (—— 用作 Cache)

      • 特点

        • 只要加上电源,信息就能一直保持

        • 对电气干扰相对不很敏感

        • 比 DRAM 更快,也更贵

      • 六管静态 MOS 管电路

        • SRAM 数据保存在一对正负反馈门电路中,只要供电,数据就一直保持,不是破环性读出,也不需要重写数据来保持数据不变。即:无需刷新!
    • DRAM 动态随机存取存储器 (—— 用作主存储器)

      • 特点

        • 每隔一段时间必须刷新一次

        • 对电气干扰比较敏感

        • 比 SRAM 慢,但便宜

        • DRAM 读比写稍微快一点,因为读需要等行列地址就绪?

      • 动态单管 MOS 电路

        • 优点

          • 电路元件少

          • 工作功耗小

          • 集成度高,广泛用于大容量主存储器中

        • 缺点

          • 速度慢

          • 破坏性读出 (读后状态改变,需重写)

          • 需要定时刷新

    • 随机存储器的地址译码

      • 线选法 (一维地址译码) 存储矩阵

      • 位片式 (二维双译码) 存储矩阵

      • 地址引脚复用

        • RAS 有效时送行地址

        • CAS 有效时送列地址

    # 只读存储器 (ROM)

    • 不可在线改写内容的 ROM (—— 用作 BIOS)

    • 闪存 (Flash Memory)(——U 盘、SSD 等)

    # 划分为主存空间包 ROM 和 RAM 区

    # ROM 区用来存放系统程序、标准子程序等,选 ROM 芯片构造

    # RAM 区用来存放用户程序,选 RAM 芯片构造

    # 存储容量的扩展

    默认使用同一种芯片扩展

    # 位扩展

    存储芯片 (mk×n 位 / 片) 构成存储器 (mk×N 位)

    • 特点:

      • 字数不变 (存储单元个数不变),位数扩展 (字长加长)
        芯片地址码位数 = 存储器的地址码位数
        芯片存储单元中所含位元数 < 存储器存储单元中所含位元数

      • 给出地址后,该存储单元中所有芯片同时工作

      • 所有芯片地址范围相同

    • 需存储芯片数:ceil (N /n) 片

    # 字扩展

    存储芯片 (mk×n 位 / 片) 构成存储器 (Mk×n 位)

    • 特点:

      • 位数不变 (字长不变)、扩充容量 (存储单元个数增加)
        芯片地址码位数 < 存储器的地址码位数
        芯片存储单元中所含位元数 = 存储器存储单元中所含位元数

      • 给出地址后,选中芯片工作

      • 每个芯片地址范围不同

    • 需存储芯片数:ceil (M /m) 片

    • 片选 依据 增加的高位地址码
      增加的高位地址码 = 存储器地址码位数 - 每片芯片的地址码位数

    # 字位同时扩展

    存储芯片 (mk×n 位) 构成存储器 (Mk×N 位)

    • 特点:

      • 存储单元个数,字长同时增加
        芯片地址码位数 < 存储器的地址码位数
        芯片存储单元中所含位元数 < 存储器存储单元中所含位元数

      • 给出地址后,同行芯片均工作

    • 需存储芯片数: ceil (M /m) * ceil ( N /n ) 片

    \ No newline at end of file +存储器 - 计算机组成原理 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 思维导图

    存储器

    # 基本术语

    # 记忆单元 / 存储位元 / 位元 (Cell)

    具有两种稳态的能够表示二进制数 0 和 1 的物理器件

    # 存储单元 / 编址单位 / 寻址单位 (Addressing Unit)

    存储器中具有相同地址的那些位构成一个存储单元,又称为一个编址单位

    # 存储体 / 存储矩阵 / 存储阵列 (Bank)

    所有存储单元构成一个存储阵列

    # 存储器地址寄存器 (Memory Address Register – MAR)

    用于存放主存单元地址的寄存器

    # 存储器数据寄存器 (Memory Data Register – MDR)

    用于存放主存单元中数据的寄存器

    # 机器字长

    运算器中参加运算的寄存器的位数,即:数据通路的宽度

    # 存储字

    存储芯片中一个读写单位,一般等于存储器的数据线宽度
    32bit 系统是 32 位,64 位系统是 64 位……

    # 编址方式 (Addressing Mode):对存储体中各存储单元进行编号的方法

    • 按字节编址 (现代计算机基本上都按字节编址)

    • 按字编址 (早期有机器按字编址)

    # 传输单位:一次读写存储器的数据单位

    • 对主存而言,指一次从主存读出或写入的数据位数

    • 对辅存而言,数据通常按块传输,传输单位为块

    # 计算机存储层次结构

    #

    # 从上到下一级一级查找,找到后再从下往上一级一级存

    • CPU 运行时,需要的操作数大部分来自寄存器

    • 需从 (向) 存储器中取 (存) 数时,先访问 cache;若在,取自 cache

    • 若不在 cache,则访问 RAM;若在,则取自 RAM (还会保存到 cache 上)

    • 若不在 RAM,则访问磁盘,操作数从磁盘中读出→RAM →cache

    # 按存储器中信息的可保存性分类

    # 断电后是否丢失数据

    • 挥发性存储器 (易失性存储器)
      特点:断电后,信息即丢失
      SRAM、DRAM

    • 非挥发性存储器 (非易失性 / 永久性存储器)
      特点:断电后,信息不丢失
      ROM、磁盘、闪存

    # 读出后是否保持数据

    • 破坏性存储器
      特点:读出时原存信息被破坏,需重写
      DRAM

    • 非破坏性存储器
      特点:读出时,原存信息不被破坏
      SRAM

    # 存储器技术指标

    # 存取时间 TA 与 存储周期 TS

    • 存取时间 TA:启动一次存储器操作到完成该操作所用时间

      • 读:从启动读命令到读出的数据送 MDR 所需的时间

      • 写:从启动到将 MDR 内容写入指定存储单元的时间
        亦称访问时间,是反映存储器速度的指标,决定了 CPU 发出读 / 写命令后必须等待的时间

    • 存储周期 TS:亦称存取周期或访问周期
      两连续两次启动同一存储器进行存取操作所需的最小时间间隔次启动包括:两次读,两次写,一次读或一次写

    • TA<TS :
      TA、TS 的差别依赖于存储信息的器件和电路,存储介质和控制线路需要恢复时间
      破坏性读出须重写;非破坏性读出不须重写

    # 存取宽度 与 存储带宽

    • 存取宽度:一次访存操作可存取的数据位数或字节数 (银河 - I 的存取宽度是 64 位)

    • 存储带宽:每秒传输的最大数据量 (位 / 秒)

    # MTBF(Mean Time Between Failure)

    • MTBF 表示两次故障之间的平均时间间隔

    • 采用纠错编码技术进行容错处理

    # 主存和 CPU 之间的两种通信方式

    # 异步方式过程 (需握手信号)

    • 读操作

    • CPU 送地址到地址线,主存进行地址译码

    • CPU 发读命令,然后等待存储器发回 “完成” 信号

    • 主存收到读命令后开始读数,完成后发 “完成” 信号给 CPU

    • CPU 接收到 “完成” 信号,从数据线取数

    # 同步方式的特点

    • CPU 和主存由统一时钟信号控制,无需应答信号

    • 主存总是在确定的时间内准备好数据

    • CPU 送出地址和读命令后,总是在确定的时间取数据

    • 存储器芯片必须支持同步方式

    # 主存和 CPU 的连接

    #

    # 地址线的连接

    • CPU 地址线数决定了整个主存空间的寻址范围

    • CPU 地址线数 > 存储芯片地址引脚线

    • 通常将 CPU 地址线的低位和存储芯片地址线相连,高位用作字扩展时的片选信号的译码

    # 数据线的连接

    • CPU 数据线数决定了一次可读写的最大数据宽度

    • CPU 数据线数 > 存储芯片数据引脚线

    • 通常将 CPU 数据线连到多个位扩展的芯片中,使扩展后的位数与 CPU 数据线数相等

    # 控制线的连接

    • 若 CPU 读 / 写命令线和存储芯片的读 / 写控制线是一根,且电平信号一致,则可直接相连

    • 若 CPU 读 / 写命令线分开,则需分别进行连接

    • ROM read only 不需要连接写使能 WR

    # 按在计算机中的作用分类

    # 寄存器型存储器

    • 亦称便笺式存储器,由寄存器构成,封装在 CPU 内

    • 用于存放当前正在执行的指令和使用的数据

    • 用触发器实现,速度快,容量小 O (KB)

    # 高速缓冲存储器 (Cache)

    • 位于 CPU 内部或位于主存和 CPU 之间

    • 用来存放当前正在执行的局部程序段和数据

    • 用 SRAM 实现,速度 O (ns)、容量 O (MB)

    # 主存储器

    • 位于 CPU 之外

    • 用来存放已被启动的程序及所用数据

    • 用 DRAM 实现,速度 O (ns)、容量 O (GB)

    # 辅助存储器

    • 不能由 CPU 的指令直接访问

    • 用来存放暂不运行的程序、数据或存档文件

    • 用磁、光等存储器件实现,容量大,但速度慢 O (ms)

    # 其它功能的存储器

    • 控存 CM:存储微程序代码

    • 表格存储器:为加快 CPU 处理而设置 (倒数表、函数表)

    • 字库和数据缓冲存储器:显示和印刷输出设备中

    # 按存储介质分类

    # 半导体存储器 (SCM)

    • 速度快,用作内存

    • 记忆原理:双稳态触发器、电容 (静态、动态)

    # 磁表面存储器 MSM

    • 用陶瓷、非磁性金属或塑料作载磁体,磁化后具有两种不同的剩磁状态记录信息 “1” 和 “0”

    • 非易失,容量大并且每位价格低 。用作外存

    # 光盘存储器 ODM

    • 有机玻璃作载磁体,利用磁化、晶态 / 非晶态表示信息

    • 非易失,可靠性高,保存时间长,容量大且易于更换

    • 存储速度比硬盘低一个数量级

    # 铁电存储器 FeM (FRAM)、相变存储器 PCM、阻变存储器 ReRAM、……

    • 掉电数据不丢失 (non-volatile),速度快

    # 按存储方式分类

    # 随机访问存储器 (RAM)

    • 存储器任意单元可随时访问且访问所需时间基本相同

    • 访问时间与存储单元所处的物理位置无关

    • 速度快 (ns)

    • 主存和 Cache

    # 只读存储器 (ROM)(Read Only Memory)

    • 只读是指正常工作时只读,能随机读出,不能随机写入

    • MROM:只能读,不能修改

    • PROM:只能修改一次,修改后只读

    • 可多次改写 ROM:

      • 光擦除电可编程只读存储器,亦称 UV EPROM
        ——s 级别

      • 电擦除电可编程只读存储器 E2PROM
        ——s 级别

      • 块擦除电可编程的快擦写存储器 Flash Memory
        ——ms 级别(闪存)

    # 顺序存储器 (SAS)(Sequential Access Memory)

    • 存储时以数据块为单位存储,顺序地记录在存储介质上

    • 数据按顺序从存储载体的始端读出或写入,存取时间的长短与数据所在位置有关

    • 速度慢、容量大、成本低,用作后援外存

    # 磁带、电荷耦合器件 CCD、VCD

    # 常用半导体存储器(常考)

    # 随机存取存储器 (RAM)

    • SRAM 静态随机存取存储器 (—— 用作 Cache)

      • 特点

        • 只要加上电源,信息就能一直保持

        • 对电气干扰相对不很敏感

        • 比 DRAM 更快,也更贵

      • 六管静态 MOS 管电路

        • SRAM 数据保存在一对正负反馈门电路中,只要供电,数据就一直保持,不是破环性读出,也不需要重写数据来保持数据不变。即:无需刷新!
    • DRAM 动态随机存取存储器 (—— 用作主存储器)

      • 特点

        • 每隔一段时间必须刷新一次

        • 对电气干扰比较敏感

        • 比 SRAM 慢,但便宜

        • DRAM 读比写稍微快一点,因为读需要等行列地址就绪?

      • 动态单管 MOS 电路

        • 优点

          • 电路元件少

          • 工作功耗小

          • 集成度高,广泛用于大容量主存储器中

        • 缺点

          • 速度慢

          • 破坏性读出 (读后状态改变,需重写)

          • 需要定时刷新

    • 随机存储器的地址译码

      • 线选法 (一维地址译码) 存储矩阵

      • 位片式 (二维双译码) 存储矩阵

      • 地址引脚复用

        • RAS 有效时送行地址

        • CAS 有效时送列地址

    # 只读存储器 (ROM)

    • 不可在线改写内容的 ROM (—— 用作 BIOS)

    • 闪存 (Flash Memory)(——U 盘、SSD 等)

    # 划分为主存空间包 ROM 和 RAM 区

    # ROM 区用来存放系统程序、标准子程序等,选 ROM 芯片构造

    # RAM 区用来存放用户程序,选 RAM 芯片构造

    # 存储容量的扩展

    默认使用同一种芯片扩展

    # 位扩展

    存储芯片 (mk×n 位 / 片) 构成存储器 (mk×N 位)

    • 特点:

      • 字数不变 (存储单元个数不变),位数扩展 (字长加长)
        芯片地址码位数 = 存储器的地址码位数
        芯片存储单元中所含位元数 < 存储器存储单元中所含位元数

      • 给出地址后,该存储单元中所有芯片同时工作

      • 所有芯片地址范围相同

    • 需存储芯片数:ceil (N /n) 片

    # 字扩展

    存储芯片 (mk×n 位 / 片) 构成存储器 (Mk×n 位)

    • 特点:

      • 位数不变 (字长不变)、扩充容量 (存储单元个数增加)
        芯片地址码位数 < 存储器的地址码位数
        芯片存储单元中所含位元数 = 存储器存储单元中所含位元数

      • 给出地址后,选中芯片工作

      • 每个芯片地址范围不同

    • 需存储芯片数:ceil (M /m) 片

    • 片选 依据 增加的高位地址码
      增加的高位地址码 = 存储器地址码位数 - 每片芯片的地址码位数

    # 字位同时扩展

    存储芯片 (mk×n 位) 构成存储器 (Mk×N 位)

    • 特点:

      • 存储单元个数,字长同时增加
        芯片地址码位数 < 存储器的地址码位数
        芯片存储单元中所含位元数 < 存储器存储单元中所含位元数

      • 给出地址后,同行芯片均工作

    • 需存储芯片数: ceil (M /m) * ceil ( N /n ) 片

    \ No newline at end of file diff --git a/cs/principles-of-computer-composition/operation/index.html b/cs/principles-of-computer-composition/operation/index.html index 0f7bb5f1..8aa4025d 100644 --- a/cs/principles-of-computer-composition/operation/index.html +++ b/cs/principles-of-computer-composition/operation/index.html @@ -1 +1 @@ -运算 - 计算机组成原理 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 思维导图

    运算

    # 补码

    # 符号部分同原码

    • 数的最高位为符号位,0 表示正数,1 表示负数

    # 数字部分与它的符号位有关

    • 对于正数,补码数值部分与原码数值部分相同

    • 对于负数,补码数值部分是将原码数值部分按位取反再加 1 ,即在反码数值部分基础上加 1

    # 阶码

    # 补码和移码的关系:符号位相反、数值位相同

    • 阶码用移码表示,阶码加减运算后得到补码,再将符号位取反即可得到运算后的阶码

    • 移码的和、差 = 真值的和、差的补码

    # 补码 E_1 的 (mod 2^n) n 位移码:

    [E_1] 移 = 2^(n-1) + E_1

    # 补码 E_1 的 (mod 2^n) n 位补码:

    [E_1] 补 = 2^n + E_1

    # 浮点数(IEEE754 标准)

    # 单精度浮点数

    • 31:1 位符号位

    • 30~23:8 位阶码

    • 22~0:23 位尾数

    # 双精度浮点数

    • 63:1 位符号位

    • 62~52:11 位阶码

    • 51~0:52 位尾数

    # 阶码全部为 0:

    • 尾数不为 0:表示非规格化数

    • 尾数为 0:表示数字 0

    # 阶码全部为 1:

    • 尾数不为 0:NaN(非数)

    • 尾数为 0:

      • 符号位为 0:表示正无穷

      • 符号位为 1:表示负无穷

    # 状态标志位

    # CF—— 进位标志位

    • 加法

    • 减法

    # OF—— 溢出标志位

    # ZF

    # 计算机运算部件 (ALU) 的硬件组织

    # 1 位全加器 / 普通进位加法器 CPA

    # 存储进位加法器 CSA

    • n 位串行加法器
      分 n 步实现,每步只求一位和

      • 进位按串行方式传递,速度慢

      • 进位延时较长,关键路径延时 O (n)

      • 实现复杂度 O (n)

    • n 位并行加法器

      • 同时产生进位

      • 加法延时缩短,关键路径延时 O (log n)

      • 实现相对复杂,复杂度 O (n2)

      • 完全大规模并行进位实现困难,因为高位的进位形成逻辑涉及输入变量过多,将受到器件扇入系数的限制

    • 分组并行进位加法器

      • 组内并行,组间串行

    # 快速乘法器

    • 柱形乘法器

    • 阵列乘法器

    # 快速除法器

    • 阵列除法器

    # 定点整数阶码运算器和定点小数尾数运算器组成浮点运算器

    # 与其他寄存器组构成完整的运算部件 (ALU)

    # 移位

    # 算术左移

    # 逻辑左移

    # 算术右移

    # 逻辑右移

    # 注意易错

    • 对于整数,左移指令相当于将原数与 2 的幂次方相乘;右移指令则相当于将原来的整数除以 2 的幂次方。正数没啥问题,负数不行!

    # 位扩展

    短数转为长数

    # 0 扩展:无符号数,即:前面补 “0”,

    # 符号扩展:有符号整数,即:前面补 “数符

    # 应用

    • 强制类型转换

    • 无符号数加入寄存器时用这种方式填补寄存器左侧多余位置

    # 位截断

    长数转为短数

    # 强行将一个长数的高位丢弃

    # 可能会发生数据 “溢出” 或者数据不正确

    # 数据语义上的高位截断,与大小端存储方式无关

    # 应用

    • 强制类型转换

    • 有符号数加入寄存器时用这种方式填补寄存器左侧多余位置

    # 加减法

    # 有符号原码加减法

    符号位和数值部分分别处理
    减法转化为加法处理

    • 同号:

      • 数值位直接相加

      • 和的符号取被加数的符号

      • 若最高位产生进位,则结果溢出

    • 异号:负数取补码,与正数相加,分二种情况讨论:

      • 最高数值位产生进位,符号位为 0,表明加法结果为正,所得数值位正确。

      • 最高数值位没有产生进位,符号位为 1,表明加法结果为负,得到的是数值位的补码形式,需对结果求补,得到原码结果

    # 无符号原码加减法

    # 补码加减法

    补码可以化简加法运算
    还可以将减法运算转化为加法运算

    • 溢出
      同符号数相加或异符号数相减才可能溢出

      • 补码中采用两位符号位 (变形补码),符号位仍然参与运算,结果中的两位符号位标示溢出状态

        • 11、00:正常

        • 01:正溢出

        • 10:负溢出

      • C 语言不考虑溢出的异常处理

        • 对于 unsigned 整型溢出:做模运算

        • 对于 signed 整型的溢出,C 的规范定义是 “undefined behavior”

      • MIPS:

        • MIPS 上的 C 编译器会选用无符号的算术运算指令 (如 addu、addiu、subu)

        • MIPS 上的 Fortran 编译器则会根据操作数的不同类型选用合适的算术运算指令 (如 add、addi、sub)

        • MIPS 检测到有符号数运算溢出时,产生一个异常(exception/interrupt)

        • 造成溢出的那条指令的地址被存放在一个特定寄存器 —— 异常程序计数器 (EPC)

        • mfc0 指令将 EPC 中的地址复制到某个寄存器 (k0/k0/k1),用 jr 指令返回到溢出而中断的代码处

    # 移码加减法

    • ① 加法:直接将 [E1] 移 和 [E2] 移进行模 2n 相加,对结果符号取反得到结果的阶码

    • ② 减法:先将减数 [E2] 移 求补 (各位取反,末位加 1),再与被减数 [E1] 移进行模 2n 相加,对结果的符号取反

    • ③ 溢出判断:进行模 2n 相加时,如果两个加数的符号相同,且与和数的符号也相同,则发生溢出

    # 乘法

    “加” + “右移”

    # 原码一位乘法

    • 符号位和数值位分开运算,数值部分用无符号数乘法实现

    • 用于浮点数尾数乘法运算

    # 补码乘法

    • 用于定点整数乘法法运算

    • 现在主流乘法器是 3 位乘法,甚至 4 位,都是基于两位判断位

    • Booth 一位乘法

      • 被乘数 A 和部分积 P 均取两位符号位即变形码,乘数取一位符号位,符号位参与运算

      • 乘数末尾增设附加位 Bn+1 ,其初始值为 0

      • Bn 和 Bn+1 构成各步运算的乘数判断位

      • 按补码移位规则:部分积为正 (第 1 符号位为 0),右移时有效位最高位补 0;部分积为负,右移时有效位最高位补 1

      • 按 Booth 乘法表算法进行到第 n+1 步,但第 n+1 步的部分积不再移位

    • Booth 二位乘法

      • 两位两位看,从右往左看,但每次 [A] 补的 “数量级” 不一样
    • (-1) X (-1) 是定点小数补码乘法唯一溢出情况

    # 使用伪加器构成柱形乘法器

    • 三个数的求和由两次普通加法变成了:一次伪加,一次普通加

    • 四个数的求和,通过两次伪加,一次普通加

    • 求和加的数越多,普通加仍是一次,伪加增多,伪加器相比全加器效率更高

    # 利用实现多位乘的专用芯片构成阵列乘法器

    # 除法

    “加 / 减”+“左移”

    # 原码除法

    • 符号位和数值位分开,数值部分用无符号数除法实现

    • 用于浮点数尾数除法运算

    • 有恢复余数和加减交替法两种

    # 补码除法

    • 符号位和数值位一起运算

    • 用于定点整数除法运算

    # 快速除法器:阵列除法器

    # 浮点数加减法

    # 求阶差

    # 对阶

    # 尾数相加减

    # 规格化并判断溢出

    • 尾数规格化范围:1 ≤ 尾数真值 < 2

    • 阶码不溢出的范围:127 ≥ 阶码真值 ≥ -126

    • 阶码上溢:结果发生溢出,一般将其认为是+∞和-∞

    • 阶码下溢:结果为 0

    • 尾数上溢:

      • 不一定浮点数溢出,即不一定会发生 “异常”

      • 右归:将尾数右移,阶码增 1 来重新对齐

    • 尾数下溢:

      • 进行舍入处理

      • 在运算过程中,添加保护位

    # 舍入

    • 就近舍入,舍入为最近可表示的数

    • 附加位为:

      • 11:入

      • 01:舍

      • 10(强制结果为偶数)

      • 00:保持结果不变

    # 置 0

    • 尾数为 0 说明结果也为 0,根据 IEEE754,阶码和尾数全为 0,需要将阶码置零

    # 注意易错

    • 浮点数加减法不满足结合律

    # 浮点数乘除法

    # 尾数用定点原码乘 / 除运算实现

    # 阶码用定点数加 / 减运算实现

    # 规格化

    • 乘法

      • 1≤尾数 1 × 尾数 2<4

      • 不需左规、最多右规 (➗2) 一次

    • 除法

      • 0.5 < 尾数 1 ÷ 尾数 2<2

      • 不需右规、最多左规 (✖️2) 一次

    # 符号确定

    • 同号:结果为正

    • 异号:结果为负

    \ No newline at end of file +运算 - 计算机组成原理 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 思维导图

    运算

    # 补码

    # 符号部分同原码

    • 数的最高位为符号位,0 表示正数,1 表示负数

    # 数字部分与它的符号位有关

    • 对于正数,补码数值部分与原码数值部分相同

    • 对于负数,补码数值部分是将原码数值部分按位取反再加 1 ,即在反码数值部分基础上加 1

    # 阶码

    # 补码和移码的关系:符号位相反、数值位相同

    • 阶码用移码表示,阶码加减运算后得到补码,再将符号位取反即可得到运算后的阶码

    • 移码的和、差 = 真值的和、差的补码

    # 补码 E_1 的 (mod 2^n) n 位移码:

    [E_1] 移 = 2^(n-1) + E_1

    # 补码 E_1 的 (mod 2^n) n 位补码:

    [E_1] 补 = 2^n + E_1

    # 浮点数(IEEE754 标准)

    # 单精度浮点数

    • 31:1 位符号位

    • 30~23:8 位阶码

    • 22~0:23 位尾数

    # 双精度浮点数

    • 63:1 位符号位

    • 62~52:11 位阶码

    • 51~0:52 位尾数

    # 阶码全部为 0:

    • 尾数不为 0:表示非规格化数

    • 尾数为 0:表示数字 0

    # 阶码全部为 1:

    • 尾数不为 0:NaN(非数)

    • 尾数为 0:

      • 符号位为 0:表示正无穷

      • 符号位为 1:表示负无穷

    # 状态标志位

    # CF—— 进位标志位

    • 加法

    • 减法

    # OF—— 溢出标志位

    # ZF

    # 计算机运算部件 (ALU) 的硬件组织

    # 1 位全加器 / 普通进位加法器 CPA

    # 存储进位加法器 CSA

    • n 位串行加法器
      分 n 步实现,每步只求一位和

      • 进位按串行方式传递,速度慢

      • 进位延时较长,关键路径延时 O (n)

      • 实现复杂度 O (n)

    • n 位并行加法器

      • 同时产生进位

      • 加法延时缩短,关键路径延时 O (log n)

      • 实现相对复杂,复杂度 O (n2)

      • 完全大规模并行进位实现困难,因为高位的进位形成逻辑涉及输入变量过多,将受到器件扇入系数的限制

    • 分组并行进位加法器

      • 组内并行,组间串行

    # 快速乘法器

    • 柱形乘法器

    • 阵列乘法器

    # 快速除法器

    • 阵列除法器

    # 定点整数阶码运算器和定点小数尾数运算器组成浮点运算器

    # 与其他寄存器组构成完整的运算部件 (ALU)

    # 移位

    # 算术左移

    # 逻辑左移

    # 算术右移

    # 逻辑右移

    # 注意易错

    • 对于整数,左移指令相当于将原数与 2 的幂次方相乘;右移指令则相当于将原来的整数除以 2 的幂次方。正数没啥问题,负数不行!

    # 位扩展

    短数转为长数

    # 0 扩展:无符号数,即:前面补 “0”,

    # 符号扩展:有符号整数,即:前面补 “数符

    # 应用

    • 强制类型转换

    • 无符号数加入寄存器时用这种方式填补寄存器左侧多余位置

    # 位截断

    长数转为短数

    # 强行将一个长数的高位丢弃

    # 可能会发生数据 “溢出” 或者数据不正确

    # 数据语义上的高位截断,与大小端存储方式无关

    # 应用

    • 强制类型转换

    • 有符号数加入寄存器时用这种方式填补寄存器左侧多余位置

    # 加减法

    # 有符号原码加减法

    符号位和数值部分分别处理
    减法转化为加法处理

    • 同号:

      • 数值位直接相加

      • 和的符号取被加数的符号

      • 若最高位产生进位,则结果溢出

    • 异号:负数取补码,与正数相加,分二种情况讨论:

      • 最高数值位产生进位,符号位为 0,表明加法结果为正,所得数值位正确。

      • 最高数值位没有产生进位,符号位为 1,表明加法结果为负,得到的是数值位的补码形式,需对结果求补,得到原码结果

    # 无符号原码加减法

    # 补码加减法

    补码可以化简加法运算
    还可以将减法运算转化为加法运算

    • 溢出
      同符号数相加或异符号数相减才可能溢出

      • 补码中采用两位符号位 (变形补码),符号位仍然参与运算,结果中的两位符号位标示溢出状态

        • 11、00:正常

        • 01:正溢出

        • 10:负溢出

      • C 语言不考虑溢出的异常处理

        • 对于 unsigned 整型溢出:做模运算

        • 对于 signed 整型的溢出,C 的规范定义是 “undefined behavior”

      • MIPS:

        • MIPS 上的 C 编译器会选用无符号的算术运算指令 (如 addu、addiu、subu)

        • MIPS 上的 Fortran 编译器则会根据操作数的不同类型选用合适的算术运算指令 (如 add、addi、sub)

        • MIPS 检测到有符号数运算溢出时,产生一个异常(exception/interrupt)

        • 造成溢出的那条指令的地址被存放在一个特定寄存器 —— 异常程序计数器 (EPC)

        • mfc0 指令将 EPC 中的地址复制到某个寄存器 (k0/k0/k1),用 jr 指令返回到溢出而中断的代码处

    # 移码加减法

    • ① 加法:直接将 [E1] 移 和 [E2] 移进行模 2n 相加,对结果符号取反得到结果的阶码

    • ② 减法:先将减数 [E2] 移 求补 (各位取反,末位加 1),再与被减数 [E1] 移进行模 2n 相加,对结果的符号取反

    • ③ 溢出判断:进行模 2n 相加时,如果两个加数的符号相同,且与和数的符号也相同,则发生溢出

    # 乘法

    “加” + “右移”

    # 原码一位乘法

    • 符号位和数值位分开运算,数值部分用无符号数乘法实现

    • 用于浮点数尾数乘法运算

    # 补码乘法

    • 用于定点整数乘法法运算

    • 现在主流乘法器是 3 位乘法,甚至 4 位,都是基于两位判断位

    • Booth 一位乘法

      • 被乘数 A 和部分积 P 均取两位符号位即变形码,乘数取一位符号位,符号位参与运算

      • 乘数末尾增设附加位 Bn+1 ,其初始值为 0

      • Bn 和 Bn+1 构成各步运算的乘数判断位

      • 按补码移位规则:部分积为正 (第 1 符号位为 0),右移时有效位最高位补 0;部分积为负,右移时有效位最高位补 1

      • 按 Booth 乘法表算法进行到第 n+1 步,但第 n+1 步的部分积不再移位

    • Booth 二位乘法

      • 两位两位看,从右往左看,但每次 [A] 补的 “数量级” 不一样
    • (-1) X (-1) 是定点小数补码乘法唯一溢出情况

    # 使用伪加器构成柱形乘法器

    • 三个数的求和由两次普通加法变成了:一次伪加,一次普通加

    • 四个数的求和,通过两次伪加,一次普通加

    • 求和加的数越多,普通加仍是一次,伪加增多,伪加器相比全加器效率更高

    # 利用实现多位乘的专用芯片构成阵列乘法器

    # 除法

    “加 / 减”+“左移”

    # 原码除法

    • 符号位和数值位分开,数值部分用无符号数除法实现

    • 用于浮点数尾数除法运算

    • 有恢复余数和加减交替法两种

    # 补码除法

    • 符号位和数值位一起运算

    • 用于定点整数除法运算

    # 快速除法器:阵列除法器

    # 浮点数加减法

    # 求阶差

    # 对阶

    # 尾数相加减

    # 规格化并判断溢出

    • 尾数规格化范围:1 ≤ 尾数真值 < 2

    • 阶码不溢出的范围:127 ≥ 阶码真值 ≥ -126

    • 阶码上溢:结果发生溢出,一般将其认为是+∞和-∞

    • 阶码下溢:结果为 0

    • 尾数上溢:

      • 不一定浮点数溢出,即不一定会发生 “异常”

      • 右归:将尾数右移,阶码增 1 来重新对齐

    • 尾数下溢:

      • 进行舍入处理

      • 在运算过程中,添加保护位

    # 舍入

    • 就近舍入,舍入为最近可表示的数

    • 附加位为:

      • 11:入

      • 01:舍

      • 10(强制结果为偶数)

      • 00:保持结果不变

    # 置 0

    • 尾数为 0 说明结果也为 0,根据 IEEE754,阶码和尾数全为 0,需要将阶码置零

    # 注意易错

    • 浮点数加减法不满足结合律

    # 浮点数乘除法

    # 尾数用定点原码乘 / 除运算实现

    # 阶码用定点数加 / 减运算实现

    # 规格化

    • 乘法

      • 1≤尾数 1 × 尾数 2<4

      • 不需左规、最多右规 (➗2) 一次

    • 除法

      • 0.5 < 尾数 1 ÷ 尾数 2<2

      • 不需右规、最多左规 (✖️2) 一次

    # 符号确定

    • 同号:结果为正

    • 异号:结果为负

    \ No newline at end of file diff --git a/cs/principles-of-computer-composition/virtual-memory/index.html b/cs/principles-of-computer-composition/virtual-memory/index.html index 5691de88..3be230d4 100644 --- a/cs/principles-of-computer-composition/virtual-memory/index.html +++ b/cs/principles-of-computer-composition/virtual-memory/index.html @@ -1 +1 @@ -虚拟存储器 - 计算机组成原理 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 思维导图

    虚拟存储器

    # 分区方案

    # 简单分区:

    • 原理:使用长度不等的固定长分区,当一个进程调入主存时,分配给它一个能容纳的最小分区

    • 缺点:固定长度的分区,可能会浪费 主存空间。多数情况下,进程对分区大小的需求不可能和提供的分区大小一样

    # 可变长分区(Partitioning)

    • 原理:分配的分区大小与进程所需大小一样

    • 特点:开始较好 ,但到最后存储器中会有许多碎片。时间越长,存储器的碎片会越来越多,存储器的利用率下降

    • 程序员自己手动管理内存,可以改善内存碎片化问题(例如一次性 malloc 或 new、stl 新建容器好几 kb 大小……

    # 分页(Paging)

    • 基本思想

      • 内存分成固定长且较小的存储块 (页框 /page frame)

      • 每个进程也划分成固定长的程序块 (页 /page)

      • 程序块 (页 /page) 可装到内存可用的存储块 (页框 /page frame) 中

      • 无需用连续页框来存放一个进程

      • 操作系统为每个进程生成一个页表:页和页框一一对应的关系表

        • 通过页表 (page table) 实现逻辑地址向物理地址转换 (Address Mapping)

    # 虚存技术的实质

    # 程序员在比实际主存空间大得多的逻辑地址空间编写程序

    # 程序执行时,把当前需要的程序段和相应的数据块调入主存,其他暂不用的部分存放在磁盘上

    # 指令执行时,通过硬件将逻辑地址 (亦称虚拟地址或虚地址) 转化为物理地址 (亦称主存地址或实地址)// 分页查表

    # 在发生指令或数据访问失效时,由操作系统进行主存和磁盘之间的信息交换

    # 每个进程都有专属于自己的从 0 到 FFFFFFFFH 的地址空间

    # 快表 / 转换后备缓冲器 / TLB (Translation Lookaside Buffer)

    # 原理:

    • 使用 Cache 来存储页表项,称为 TLB,它包含最近使用的那些页表项,加速虚实地址转换

    • 多用全相联:命中率高,小、成本不高

    • 采用随机替换策略:降低替换算法开销

    • 采用回写策略:减少访问内存的次数

    # TLB 页表项

    • Tag 位

    • Index 位,组相联时才有

    • Physical Adress,对应物理页框号

    • 其他标志位

      • Valid 位,1 位

      • Dirty 位,1 位

      • Ref

      • Access

    # TLB 缺失

    # 基本概念

    # 主存分配

    • 操作系统:固定(ROM)

    • 用户区:分区(RAM)

    # 三种地址空间:

    • 虚拟地址空间

    • 主存地址空间

    • 辅存地址空间

    # 地址映射:把虚拟地址空间映射到主存地址空间

    # 地址变换:程序运行时,把虚拟地址变换成主存物理地址

    # 为什么需要虚存

    # 同时运行更多进程,内存需求增大

    # 允许多个程序有效而安全地共享存储器,消除内存因小而有限的存储器容量给程序设计造成的障碍

    • 可以更有效的共享处理器和主存

    • 虚拟存储器实现程序地址空间到物理空间的转换,加强了各个程序地址空间之间的保护

    # 允许单用户程序大小超过主存储器的容量

    • 虚拟存储器自动管理主存和辅存组成的两级层次结构

    # 虚存组织方式

    # 页式

    • 原理:

      • 虚存地址空间和主存地址空间按统一大小分成若干页,虚存称为虚页,主存称为实页

      • 程序按页从磁盘调入主存 (某一虚页调入某一实页)

      • 指令给出虚拟地址 / 逻辑地址

      • 每个页表项记录对应的虚页情况

      • Valid 为 0 说明 “miss”(称为 page fault / 缺页)

      • CPU 执行指令时,先将逻辑地址转换为物理地址

      • 地址转换由 MMU 实现

      • 页大小比 Cache 中 Block 大得多

      • 采用全相联映射

      • 通过软件来处理 “缺页”

      • 采用 Write Back 写策略

    • 地址格式

      • 虚存的地址格式 (逻辑地址格式):

        • 直接映射虚拟地址

          • 高位,虚拟地址比物理地址长的位数,算在 Tag 位里

          • Tag 位,标记位
            长度 = log _2 (主存中 页 的 数量 / TLB 中 页表项 的 数量)

          • Index 位,Cache 槽号
            长度 = log _2 (TLB 中 页表项 的 数量)

          • Offset 位,页内地址
            长度 = log _2 (页 的 大小) (按编址单位算)

        • 全相联虚拟地址

          • 高位,虚拟地址比物理地址长的位数,算在 Tag 位里

          • Tag 位,标记位
            长度 = log _2 (主存中 页 的 数量)

          • Offset 位,页内地址
            长度 = log _2 (页 的 大小) (按编址单位算)

        • n 路组相联虚拟地址

          • 高位,虚拟地址比物理地址长的位数,算在 Tag 位里

          • Tag 位,标记位
            长度 = log _2 (主存中 页 的 数量 / 组的数量)

          • 组号位
            长度 = log _2 (组 的 数量)

          • Offset 位,页内地址
            长度 = log _2 (页 的 大小) (按编址单位算)

      • 主存的地址格式 (物理地址格式):

        • 实页号 / 物理页号

        • 页内地址

    • 页表 (Page Table)
      建立一个映射关系表来存放虚拟地址与物理地址之间的映射关系,该映射关系表被称为页表,用来实现虚实地址的转换

      • 建立在主存中,操作系统为每道程序建立一个页表

      • 设置页表基址寄存器:保存页表在主存中的起始地址

      • 页表项

        • 装入位

        • 修改 (Dirty) 位

        • 替换控制位

        • 访问权限位

        • 禁止缓存位

        • 实页号

      • 页表的项数 = 页表的容量 = 最大页表项 = 虚存大小 / 页大小

      • 页表大小 = 页表的项数 * 页表项大小

    • 异常情况

      • 缺页 (page fault)

        • 产生条件:当 Valid (有效位 / 装入位) 为 0 时

        • 相应处理:从磁盘中读信息到内存,若内存没有空间,则还要从内存选择一页替换到磁盘上,替换算法类似于 Cache,采用回写法,页面淘汰时,根据 “dirty” 位确定是否要写磁盘

        • 异常处理结束后:缺页发生时,当前指令的执行被阻塞,当前进程挂起;缺页处理结束后,回到原指令继续执行

      • 保护违例 (protection_violation_fault)

        • 产生条件: 当 Access Rights (存取权限) 与所指定的具体操作不相符时

        • 相应处理:在屏幕上显示 “内存保护错” 信息

        • 异常处理结束后:当前指令执行被阻塞,当前进程终止

        • Access Rights (存取权限) 可能的取值有哪些?

          • R = Read-only

          • R/W = read/write

          • X = execute only

    • 特点

      • 优点:实现简单,开销少。因为只有进程的最后一个零头(内部碎片)不能利用,故浪费很小

      • 缺点:由于页不是逻辑上独立的实体,可能会出现如 “一条指令跨页” 等情况(CISC),使处理、管理、保护和共享都不方便

    # 段式

    # 段页式

    \ No newline at end of file +虚拟存储器 - 计算机组成原理 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 思维导图

    虚拟存储器

    # 分区方案

    # 简单分区:

    • 原理:使用长度不等的固定长分区,当一个进程调入主存时,分配给它一个能容纳的最小分区

    • 缺点:固定长度的分区,可能会浪费 主存空间。多数情况下,进程对分区大小的需求不可能和提供的分区大小一样

    # 可变长分区(Partitioning)

    • 原理:分配的分区大小与进程所需大小一样

    • 特点:开始较好 ,但到最后存储器中会有许多碎片。时间越长,存储器的碎片会越来越多,存储器的利用率下降

    • 程序员自己手动管理内存,可以改善内存碎片化问题(例如一次性 malloc 或 new、stl 新建容器好几 kb 大小……

    # 分页(Paging)

    • 基本思想

      • 内存分成固定长且较小的存储块 (页框 /page frame)

      • 每个进程也划分成固定长的程序块 (页 /page)

      • 程序块 (页 /page) 可装到内存可用的存储块 (页框 /page frame) 中

      • 无需用连续页框来存放一个进程

      • 操作系统为每个进程生成一个页表:页和页框一一对应的关系表

        • 通过页表 (page table) 实现逻辑地址向物理地址转换 (Address Mapping)

    # 虚存技术的实质

    # 程序员在比实际主存空间大得多的逻辑地址空间编写程序

    # 程序执行时,把当前需要的程序段和相应的数据块调入主存,其他暂不用的部分存放在磁盘上

    # 指令执行时,通过硬件将逻辑地址 (亦称虚拟地址或虚地址) 转化为物理地址 (亦称主存地址或实地址)// 分页查表

    # 在发生指令或数据访问失效时,由操作系统进行主存和磁盘之间的信息交换

    # 每个进程都有专属于自己的从 0 到 FFFFFFFFH 的地址空间

    # 快表 / 转换后备缓冲器 / TLB (Translation Lookaside Buffer)

    # 原理:

    • 使用 Cache 来存储页表项,称为 TLB,它包含最近使用的那些页表项,加速虚实地址转换

    • 多用全相联:命中率高,小、成本不高

    • 采用随机替换策略:降低替换算法开销

    • 采用回写策略:减少访问内存的次数

    # TLB 页表项

    • Tag 位

    • Index 位,组相联时才有

    • Physical Adress,对应物理页框号

    • 其他标志位

      • Valid 位,1 位

      • Dirty 位,1 位

      • Ref

      • Access

    # TLB 缺失

    # 基本概念

    # 主存分配

    • 操作系统:固定(ROM)

    • 用户区:分区(RAM)

    # 三种地址空间:

    • 虚拟地址空间

    • 主存地址空间

    • 辅存地址空间

    # 地址映射:把虚拟地址空间映射到主存地址空间

    # 地址变换:程序运行时,把虚拟地址变换成主存物理地址

    # 为什么需要虚存

    # 同时运行更多进程,内存需求增大

    # 允许多个程序有效而安全地共享存储器,消除内存因小而有限的存储器容量给程序设计造成的障碍

    • 可以更有效的共享处理器和主存

    • 虚拟存储器实现程序地址空间到物理空间的转换,加强了各个程序地址空间之间的保护

    # 允许单用户程序大小超过主存储器的容量

    • 虚拟存储器自动管理主存和辅存组成的两级层次结构

    # 虚存组织方式

    # 页式

    • 原理:

      • 虚存地址空间和主存地址空间按统一大小分成若干页,虚存称为虚页,主存称为实页

      • 程序按页从磁盘调入主存 (某一虚页调入某一实页)

      • 指令给出虚拟地址 / 逻辑地址

      • 每个页表项记录对应的虚页情况

      • Valid 为 0 说明 “miss”(称为 page fault / 缺页)

      • CPU 执行指令时,先将逻辑地址转换为物理地址

      • 地址转换由 MMU 实现

      • 页大小比 Cache 中 Block 大得多

      • 采用全相联映射

      • 通过软件来处理 “缺页”

      • 采用 Write Back 写策略

    • 地址格式

      • 虚存的地址格式 (逻辑地址格式):

        • 直接映射虚拟地址

          • 高位,虚拟地址比物理地址长的位数,算在 Tag 位里

          • Tag 位,标记位
            长度 = log _2 (主存中 页 的 数量 / TLB 中 页表项 的 数量)

          • Index 位,Cache 槽号
            长度 = log _2 (TLB 中 页表项 的 数量)

          • Offset 位,页内地址
            长度 = log _2 (页 的 大小) (按编址单位算)

        • 全相联虚拟地址

          • 高位,虚拟地址比物理地址长的位数,算在 Tag 位里

          • Tag 位,标记位
            长度 = log _2 (主存中 页 的 数量)

          • Offset 位,页内地址
            长度 = log _2 (页 的 大小) (按编址单位算)

        • n 路组相联虚拟地址

          • 高位,虚拟地址比物理地址长的位数,算在 Tag 位里

          • Tag 位,标记位
            长度 = log _2 (主存中 页 的 数量 / 组的数量)

          • 组号位
            长度 = log _2 (组 的 数量)

          • Offset 位,页内地址
            长度 = log _2 (页 的 大小) (按编址单位算)

      • 主存的地址格式 (物理地址格式):

        • 实页号 / 物理页号

        • 页内地址

    • 页表 (Page Table)
      建立一个映射关系表来存放虚拟地址与物理地址之间的映射关系,该映射关系表被称为页表,用来实现虚实地址的转换

      • 建立在主存中,操作系统为每道程序建立一个页表

      • 设置页表基址寄存器:保存页表在主存中的起始地址

      • 页表项

        • 装入位

        • 修改 (Dirty) 位

        • 替换控制位

        • 访问权限位

        • 禁止缓存位

        • 实页号

      • 页表的项数 = 页表的容量 = 最大页表项 = 虚存大小 / 页大小

      • 页表大小 = 页表的项数 * 页表项大小

    • 异常情况

      • 缺页 (page fault)

        • 产生条件:当 Valid (有效位 / 装入位) 为 0 时

        • 相应处理:从磁盘中读信息到内存,若内存没有空间,则还要从内存选择一页替换到磁盘上,替换算法类似于 Cache,采用回写法,页面淘汰时,根据 “dirty” 位确定是否要写磁盘

        • 异常处理结束后:缺页发生时,当前指令的执行被阻塞,当前进程挂起;缺页处理结束后,回到原指令继续执行

      • 保护违例 (protection_violation_fault)

        • 产生条件: 当 Access Rights (存取权限) 与所指定的具体操作不相符时

        • 相应处理:在屏幕上显示 “内存保护错” 信息

        • 异常处理结束后:当前指令执行被阻塞,当前进程终止

        • Access Rights (存取权限) 可能的取值有哪些?

          • R = Read-only

          • R/W = read/write

          • X = execute only

    • 特点

      • 优点:实现简单,开销少。因为只有进程的最后一个零头(内部碎片)不能利用,故浪费很小

      • 缺点:由于页不是逻辑上独立的实体,可能会出现如 “一条指令跨页” 等情况(CISC),使处理、管理、保护和共享都不方便

    # 段式

    # 段页式

    \ No newline at end of file diff --git a/cs/principles-of-computer-composition/von-neumann-architecture/index.html b/cs/principles-of-computer-composition/von-neumann-architecture/index.html index 3846c49e..5949fc55 100644 --- a/cs/principles-of-computer-composition/von-neumann-architecture/index.html +++ b/cs/principles-of-computer-composition/von-neumann-architecture/index.html @@ -1 +1 @@ -冯・诺伊曼体系结构 - 计算机组成原理 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party
    \ No newline at end of file +冯・诺伊曼体系结构 - 计算机组成原理 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party
    \ No newline at end of file diff --git a/cs/principles-of-computer-composition/yatcpu/chisel/index.html b/cs/principles-of-computer-composition/yatcpu/chisel/index.html index 12016358..090b684a 100644 --- a/cs/principles-of-computer-composition/yatcpu/chisel/index.html +++ b/cs/principles-of-computer-composition/yatcpu/chisel/index.html @@ -1,4 +1,4 @@ -Chisel - YatCPU - 计算机组成原理 - 编程语言 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # Scala

    # import

    在 Scala 中,import 语句用于引入其他类、对象或特质,以便在当前作用域中可以直接访问它们的成员。下面是一些关于 Scala 中 import 的常见用法:

    1. 基本用法
    a
    // 导入单个类
    import scala.collection.mutable.ListBuffer
    // 导入整个包
    import scala.collection.mutable._
    // 重命名导入的类 / 对象 / 特质
    import java.util.{ArrayList => JArrayList, HashMap => JHashMap}
    // 隐藏某个成员
    import java.util.{Random => _, _} // 隐藏 java.util.Random
    1. 导入对象的成员
    a
    import java.awt.Color._
    val c = RED // 直接访问 Color 对象的成员 RED
    1. 静态导入
    a
    import java.lang.Math._
    val maxNum = max(5, 10) // 直接使用 Math 类的静态方法 max

    # 测试

    # ScalaTest

    要使用 ScalaTest 进行测试可以按照以下基本步骤进行操作:

    导入必要的包和类:

    a
    import org.scalatest.flatspec.AnyFlatSpec
    import org.scalatest.matchers.should.Matchers

    创建测试类并扩展相应的测试风格特质:

    a
    class MyTest extends AnyFlatSpec with Matchers {
      // 在这里编写测试代码
    }

    在这个例子中,我们创建了一个名为 "MyTest" 的测试类,并将其扩展为 AnyFlatSpec 和 Matchers。AnyFlatSpec 是 ScalaTest 提供的一种测试规范风格,而 Matchers 允许你使用自然语言风格编写断言。

    编写测试代码:
    在测试类中,你可以编写各种测试方法来验证你的代码行为是否符合预期,例如:

    a
    it should "return the correct result" in {
      // 编写测试代码来验证某个函数或者方法的行为
      val result = myFunction()
      result should equal (expectedResult)
    }

    在这个例子中,我们使用 "it should" 来描述测试的行为,然后使用 "in" 关键字定义测试代码块。在测试代码块中,我们调用了一个名为 "myFunction" 的函数,并使用 "should equal" 断言来验证其返回值是否等于 "expectedResult"。

    运行测试:
    使用构建工具(如 sbt)来运行测试。在 sbt 中,你可以执行以下命令来运行测试:

    h
    sbt test

    这将会执行所有的测试,并输出测试结果。

    # Chisel

    在 Scala 中,Chisel 是一个常用于硬件设计的领域特定语言(DSL),它提供了许多有用的库和工具,一些常用的库包括:

    1. chisel3:这是 Chisel 的核心库,它定义了 Chisel 的语法和 API。
    2. firrtl:这是 Chisel 的前端,用于将 Chisel 代码转换为可执行的 RTL 代码。
    3. chisel-iotesters:这是一个用于测试 Chisel 模块的库,它提供了一些常见的测试工具和模拟器。
    4. treadle:这是一个开源的 Verilog 仿真器,可以与 Chisel 代码一起使用进行仿真和调试。
    5. chisel-testers2:这是一个测试框架,用于编写和运行 Chisel 测试。
    6. chisel-formal:这是一个用于形式验证的库,可用于对 Chisel 模块进行形式化验证。
    7. chisel-fluent:这是一个用于创建可重用组件库的库,使得在不同的项目中可以轻松地重用 Chisel 模块。

    在 chisel3 库中,常用的一些子库包括:

    1. chisel3:Chisel3 核心库,定义了 Chisel 的语法和 API。
    2. chisel3.util:提供了一些常见的工具和实用函数,如位宽转换、数据选择器、优先级编码器等。
    3. chisel3.experimental:包含一些实验性质的功能和模块,可能还处于开发或测试阶段。
    4. chisel3.testers:提供了对 Chisel 模块进行测试的相关工具和框架,用于验证设计的正确性。
    5. chisel3.tester:一个用于编写和运行 Chisel 测试的测试框架,支持生成测试向量、执行仿真等操作。
    6. chisel3.stage:用于将 Chisel 生成的硬件描述转换为 Verilog 或 FIRRTL 格式的工具。
    7. chisel3.formal:用于形式验证的库,可以对 Chisel 模块进行形式化验证。

    有的时候不知道为什么 import chisel._ 了还是有些函数或运算符编译反馈 not found 或者 warnning,解决办法是 “更具体地 import”,比如:
    遇到编译报错: not found value Cat
    可以在文件开头加上 import chisel.util.Cat 来解决

    # 数据类型

    # Seq

    Seq (Sequence) 是一个表示序列的数据类型。它用于定义一系列连续的元素,并提供了一些方法来操作和访问这些元素。

    Seq 可以包含任意类型的元素,例如整数、布尔值或自定义的 Chisel 数据类型。通过使用 Seq.apply 方法或使用逗号分隔的元素列表,可以创建一个 Seq 实例

    以下是一个使用 Seq 的示例:

    import chisel3._
    +Chisel - YatCPU - 计算机组成原理 - 编程语言 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # Scala

    # import

    在 Scala 中,import 语句用于引入其他类、对象或特质,以便在当前作用域中可以直接访问它们的成员。下面是一些关于 Scala 中 import 的常见用法:

    1. 基本用法
    a
    // 导入单个类
    import scala.collection.mutable.ListBuffer
    // 导入整个包
    import scala.collection.mutable._
    // 重命名导入的类 / 对象 / 特质
    import java.util.{ArrayList => JArrayList, HashMap => JHashMap}
    // 隐藏某个成员
    import java.util.{Random => _, _} // 隐藏 java.util.Random
    1. 导入对象的成员
    a
    import java.awt.Color._
    val c = RED // 直接访问 Color 对象的成员 RED
    1. 静态导入
    a
    import java.lang.Math._
    val maxNum = max(5, 10) // 直接使用 Math 类的静态方法 max

    # 测试

    # ScalaTest

    要使用 ScalaTest 进行测试可以按照以下基本步骤进行操作:

    导入必要的包和类:

    a
    import org.scalatest.flatspec.AnyFlatSpec
    import org.scalatest.matchers.should.Matchers

    创建测试类并扩展相应的测试风格特质:

    a
    class MyTest extends AnyFlatSpec with Matchers {
      // 在这里编写测试代码
    }

    在这个例子中,我们创建了一个名为 "MyTest" 的测试类,并将其扩展为 AnyFlatSpec 和 Matchers。AnyFlatSpec 是 ScalaTest 提供的一种测试规范风格,而 Matchers 允许你使用自然语言风格编写断言。

    编写测试代码:
    在测试类中,你可以编写各种测试方法来验证你的代码行为是否符合预期,例如:

    a
    it should "return the correct result" in {
      // 编写测试代码来验证某个函数或者方法的行为
      val result = myFunction()
      result should equal (expectedResult)
    }

    在这个例子中,我们使用 "it should" 来描述测试的行为,然后使用 "in" 关键字定义测试代码块。在测试代码块中,我们调用了一个名为 "myFunction" 的函数,并使用 "should equal" 断言来验证其返回值是否等于 "expectedResult"。

    运行测试:
    使用构建工具(如 sbt)来运行测试。在 sbt 中,你可以执行以下命令来运行测试:

    h
    sbt test

    这将会执行所有的测试,并输出测试结果。

    # Chisel

    在 Scala 中,Chisel 是一个常用于硬件设计的领域特定语言(DSL),它提供了许多有用的库和工具,一些常用的库包括:

    1. chisel3:这是 Chisel 的核心库,它定义了 Chisel 的语法和 API。
    2. firrtl:这是 Chisel 的前端,用于将 Chisel 代码转换为可执行的 RTL 代码。
    3. chisel-iotesters:这是一个用于测试 Chisel 模块的库,它提供了一些常见的测试工具和模拟器。
    4. treadle:这是一个开源的 Verilog 仿真器,可以与 Chisel 代码一起使用进行仿真和调试。
    5. chisel-testers2:这是一个测试框架,用于编写和运行 Chisel 测试。
    6. chisel-formal:这是一个用于形式验证的库,可用于对 Chisel 模块进行形式化验证。
    7. chisel-fluent:这是一个用于创建可重用组件库的库,使得在不同的项目中可以轻松地重用 Chisel 模块。

    在 chisel3 库中,常用的一些子库包括:

    1. chisel3:Chisel3 核心库,定义了 Chisel 的语法和 API。
    2. chisel3.util:提供了一些常见的工具和实用函数,如位宽转换、数据选择器、优先级编码器等。
    3. chisel3.experimental:包含一些实验性质的功能和模块,可能还处于开发或测试阶段。
    4. chisel3.testers:提供了对 Chisel 模块进行测试的相关工具和框架,用于验证设计的正确性。
    5. chisel3.tester:一个用于编写和运行 Chisel 测试的测试框架,支持生成测试向量、执行仿真等操作。
    6. chisel3.stage:用于将 Chisel 生成的硬件描述转换为 Verilog 或 FIRRTL 格式的工具。
    7. chisel3.formal:用于形式验证的库,可以对 Chisel 模块进行形式化验证。

    有的时候不知道为什么 import chisel._ 了还是有些函数或运算符编译反馈 not found 或者 warnning,解决办法是 “更具体地 import”,比如:
    遇到编译报错: not found value Cat
    可以在文件开头加上 import chisel.util.Cat 来解决

    # 数据类型

    # Seq

    Seq (Sequence) 是一个表示序列的数据类型。它用于定义一系列连续的元素,并提供了一些方法来操作和访问这些元素。

    Seq 可以包含任意类型的元素,例如整数、布尔值或自定义的 Chisel 数据类型。通过使用 Seq.apply 方法或使用逗号分隔的元素列表,可以创建一个 Seq 实例

    以下是一个使用 Seq 的示例:

    import chisel3._
     
     class MyModule extends Module {
       val io = IO(new Bundle {
    @@ -217,4 +217,4 @@
         true
       }
     }
    -

    在这个例子中,我们定义了一个名为 performSomeOperation 的独立函数,并在测试用例中调用它。这种方式可以使测试代码更加清晰,同时也方便测试用例的复用。

    \ No newline at end of file +

    在这个例子中,我们定义了一个名为 performSomeOperation 的独立函数,并在测试用例中调用它。这种方式可以使测试代码更加清晰,同时也方便测试用例的复用。

    \ No newline at end of file diff --git a/cs/principles-of-computer-composition/yatcpu/lab-reports/lab1/index.html b/cs/principles-of-computer-composition/yatcpu/lab-reports/lab1/index.html index 752fb90b..1628b1ee 100644 --- a/cs/principles-of-computer-composition/yatcpu/lab-reports/lab1/index.html +++ b/cs/principles-of-computer-composition/yatcpu/lab-reports/lab1/index.html @@ -1 +1 @@ -lab1 - YatCPU - 计算机组成原理 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    YatCPU - - - lab1 - - - lab2 - - - lab3 - - - lab4

    # 测试用例及其波形图分析

    # SayGoodbyeTest

    src/test/scala/riscv/BoardTest.scala

    avatar

    其中 Top 模块来自:

    avatar

    # 测试用例的功能

    用于烧板验证,但由于 CPU 还未能实现中断,该程序只能通过 LED 灯闪烁判断 CPU 能否正常工作

    在 lab1 中该测试用例可以用于检查 CPU 及其内部各个模块、模块内的端口信号是否有完整地实现、初始化,即 "Fully initialized"。若有信号定义后没有完整实现,则该项会测试后报错: Reference xxx is not fully initialized. 则表明可能该处有代码空缺没有填补完整。

    # 从什么层面测试 CPU

    从完整性的层面测试 CPU,CPU 能否正常工作

    但测试程序并没有测试 CPU 的正确性,只是进行了 pokestep ,没有对任何信号进行 expect 检查。

    # 加载测试程序的方法

    利用定义在 ssrc/main/scala/board/z710/Top.scala 中的模块 Top ,用 RISC-V 程序 src/main/resources/say_goodbye.asmbin 初始化进行仿真

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    该测试只向波形文件写入了 Top 模块的信号

    # 执行不同指令时候对应的部件的关键信号的变化情况

    由 clock 信号变化可以看出时钟周期为 2 ps

    测试中循环执行了 50000 次 step(5) ,即 250000 个时钟周期,输出的波形图总时长为 500001 ps ,除去半个用于 reset 的时钟周期之外恰好 250000 个时钟周期。

    可以看到波形图中除了 clock 信号有规律的周期性变化之外, io_rx 也在周期性地在 0 和 1 之间变化。从波形图也可以看出每 10 ps 即 5 个时钟周期 io_rx 变化一次,符合 step(5) 一次 c.io.rx.poke((i % 2).U) 一次的变化规律。

    # ExecuteTest

    src/test/scala/riscv/singlecycle/ExecuteTest.scala

    avatar

    # 测试用例的功能

    该测试用例用于检查 Execute 模块的正确性,包括:

    # 执行 ALU 加法运算

    测试程序循环执行了 100 次两个随机生成的 int 整型操作数 op1 和 op2 的求和运算,检查 R 型指令 add 的正确性。

    同时检查跳转信号是否始终保持 0 (fasle),确保 CPU 在非跳转指令下不跳转

    # 处理 beq 指令是否跳转

    测试程序按顺序测试了两个情况:

    跳转,即 op1 等于 op2 ,此时跳转信号应该为 1 (true),原指令地址 (PC):2 应该与立即数 2 相加得到新的指令地址 (PC):4

    不跳转,即 op1 不等于 op2 ,此时跳转信号应该为 0 (false),指令地址 (PC) 保持 4 不变

    # 从什么层面测试 CPU

    从 CPU 运算处理的层面测试 CPU ,即能否正确控制 ALU 的输入输出,以及输入输出结果对指令跳转的控制

    # 加载测试程序的方法

    利用定义在 src/main/scala/riscv/core/Execute.scala 中的模块 Execute ,不使用任何 RISC-V 程序初始化进行仿真,直接仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    # ALU 加法运算

    总共执行了 102 个时钟周期,测试了 101 次随机样例 (循环从 i=0 到 i=100 执行了 101 次)。

    avatar

    # io_funct3[2:0]

    该信号始终保持 000B 不变,与 io_opcode 一起指示 ALU 执行 R 类型指令 add 的加法运算

    # io_opcode[6:0]

    该信号始终保持 0110011B 不变,与 io_funct3 一起指示 ALU 执行 R 类型指令 add 的加法运算

    avatar

    # io_alu_funct[3:0]

    按照 src/main/scala/riscv/core/ALU.scala 定义的 ChiselEnum 枚举类型如下:

    avatar

    由于总共有 11 种取值,默认从 0 开始计数直到 10 ,需要二进制 4 位数才能表示,因此对应的 io_alu_funct 为 4 位二进制数

    在加法运算时该信号始终保持 0001B 不变,对应 ChiselEnu 类型定义的 ALUFunctions 中从 0 开始计数的第二个 add,指示 ALU 执行加法运算

    # io_op1[31:0]

    该信号来自 c.io.reg1_data.poke(op1.U) ,在 add 指令中 ALU 的运算操作数 op1 来自寄存器,因此 Execute 模块的 io_reg1_data 直接传递给 ALU 模块的 io_op1

    每隔一个时钟周期 (2 ps) 变化一次,每次均为一个新的随机 int 型整数

    # io_op2[31:0]

    该信号来自 c.io.reg2_data.poke(op1.U) ,在 add 指令中 ALU 的运算操作数 op2 来自寄存器,因此 Execute 模块的 io_reg2_data 直接传递给 ALU 模块的 io_op2

    每隔一个时钟周期 (2 ps) 变化一次,每次均为一个新的随机 int 型整数

    # io_result[31:0]

    每隔一个时钟周期 (2 ps) 变化一次,每次均为当前 io_op1 与 io_op2 的算术和,同时该值直接传递给 Execute 模块的 mem_alu_result

    # 指令

    总共执行 3 个时钟周期,先后测试了成功跳转和不成功跳转两种情况。

    avatar

    # io_funct3[2:0]

    该信号始终保持 000B 不变,与 io_opcode 一起指示 ALU 执行 B 类型指令 beq 的加法运算计算跳转目标地址,以及控制跳转信号 io_if_jump 的产生

    # io_opcode[6:0]

    该信号始终保持 1100011B 不变,与 io_funct3 一起指示 ALU 执行 B 类型指令 beq 的加法运算计算跳转目标地址,以及控制跳转信号 io_if_jump 的产生

    avatar

    # io_reg1_data[31:0]

    该信号表示所读取的来自寄存器的数据,在 beq 指令中直接将两个寄存器的值进行比较得到跳转信号 io_if_jump

    在测试的后 2 个时钟周期内其值均为 9 ,来自测试程序中的 c.io.reg1_data.poke(9.U)

    # io_reg2_data[31:0]

    该信号表示所读取的来自寄存器的数据,在 beq 指令中直接将两个寄存器的值进行比较得到跳转信号 io_if_jump

    在第 2 个时钟周期内其值为 9 ,来自测试程序中的 c.io.reg2_data.poke(9.U)

    在第 3 个时钟周期内其值为 19 ,来自测试程序中的 c.io.reg2_data.poke(19.U)

    # io_if_jump

    该信号表示 PC 下一条指令是否要跳转,是则 io_if_jump = 1 ,否则 io_if_jump = 0

    在第 2 个时钟周期其值为 1 ,因为此时 io_reg1_data = 9 = io_reg2_data

    在第 3 个时钟周期其值为 0 ,因为此时 io_reg1_data = 9 , io_reg2_data = 19

    # io_alu_funct[3:0]

    在 beq 指令中,ALU 执行加法运算,io_alu_funct = 1

    # alu_io_op1[31:0]

    由于 c.io.aluop1_source.poke(1.U) ,aluop1_source = 1 指示该信号 alu_io_op1 来自 instruction_address ,同时 c.io.instruction_address.poke(2.U) ,因此 alu_io_op1 = 2

    # alu_io_op2[31:0]

    由于 c.io.aluop2_source.poke(1.U) ,aluop2_source = 1 指示该信号 alu_io_op2 来自立即数 io_immediate ,同时 c.io.immediate.poke(2.U) ,因此 alu_io_op2 = 2

    # io_if_jump_address[31:0]

    表示跳转成功时 PC 跳转的目标指令地址,在 beq 指令中等于 ALU 计算出来的结果 alu_io_result

    在 2 个时钟周期内均为 4 ,即 alu_io_op1 与 alu_io_op2 的和

    # InstructionDecoderTest

    src/test/scala/riscv/singlecycle/InstructionDecoderTest.scala

    avatar

    # 测试用例的功能

    测试了 InstructionDecode 模块的正确性,包括:

    # S 型指令

    测试了对 S 型指令 sw 翻译的正确性,包括读取正确的寄存器并且将正确的操作数传递给 ALU。

    在 S 型指令中,存储的数据来自寄存器,存储目标的内存地址由另一寄存器中的数据和指令中的立即数作为偏移量共同决定。

    # lui 指令

    测试了对高位加载立即数指令 lui 翻译的正确性,包括读取正确的寄存器并且将正确的操作数传递给 ALU

    在 lui 指令中,符号位扩展的 20 位立即数左移 12 位后,将低 12 位置零,传入 ALU,同时 ALU 另一个操作数设置为 0 号寄存器 (内容始终为 0) 的值,即另一个操作数为 0,ALU 做加法运算,相当于将位扩展后的结果直接作为写入的值,写入目标寄存器中

    # add 指令

    测试了对 R 型指令 add 翻译的正确性,主要测试了是否控制选择寄存器作为 ALU 操作数

    # 从什么层面测试 CPU

    从指令的翻译、分析层面,指导 CPU 如何根据当前指令进行工作,包括立即数的提取、各个控制信号的取值等

    # 加载测试程序指令的方法

    利用定义在 src/main/scala/riscv/core/InstructionDecode.scala 中的模块 InstructionDecode ,不使用任何 RISC-V 程序初始化进行仿真,直接仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    # S 型指令

    总共执行了一个时钟周期 (2 ~ 4 ps),测试了 S 型指令 sw 在 ID 阶段的信号输出

    avatar

    # io_instruction[31:0]

    该信号直接来自 c.io.instruction.poke(00a02223L.U) ,对应二进制码为:0000000 0

    avatar

    # io_ex_immediate[31:0]

    该信号由 io_instruction 的 [11:5] 部分 0000000B 和 [11:7] 部分 00100B 组成 004H ,再位扩展后得到 00000004H = 4,作为立即数传入 ALU

    # io_regs_reg1_read_address[4:0]

    该信号来自 io_instruction 的 [19:15] 部分,为 00000B,对应 0 号寄存器

    # io_regs_reg2_read_address[4:0]

    该信号来自 io_instruction 的 [24:20] 部分,为 01010B,对应 10 号寄存器

    # io_ex_aluop1_source

    在 S 型指令中, io_ex_aluop1_source = 0 ,指示读取寄存器的数据传给 ALU 作为其中一个操作数

    # io_ex_aluop2_source

    在 S 型指令中, io_ex_aluop2_source = 1 ,指示将立即数 io_ex_immediate 传给 ALU 作为其中一个操作数

    # io_memory_read_enable

    S 型指令没有读取内存,所以 io_memory_read_enable = 0

    # io_memory_write_enable

    S 型指令需要写内存,所以 io_memory_write_enable = 1

    # lui 指令

    总共执行了一个时钟周期 (4 ~ 6 ps),测试了 lui 指令在 ID 阶段的信号输出

    avatar

    # io_instruction[31:0]

    该信号直接来自 c.io.instruction.poke(000022b7L.U) ,对应二进制码为:00000000000000000010 00101 01

    avatar

    # io_ex_immediate[31:0]

    该信号由 io_instruction 的 [31:12] 部分 00002H 位扩展后得到 00002000H = 2,作为立即数传入 ALU

    # io_reg_write_address[4:0]

    该信号来自 io_instruction 的 [11:7] 部分,其值为 00101 ,对应 5 号寄存器,即结果要写入 5 号寄存器

    # io_ex_aluop1_source

    在 lui 指令中, io_ex_aluop1_source = 0 ,指示读取寄存器 (0 号寄存器) 的数据 ( 0 ) 传给 ALU 作为其中一个操作数

    # io_ex_aluop2_source

    在 lui 指令中, io_ex_aluop1_source = 1 ,指示将立即数 io_ex_immediate 传给 ALU 作为其中一个操作数

    # io_regs_reg1_read_address[4:0]

    在 lui 指令中, 该值被设置为 00000 ,对应 0 号寄存器

    # io_reg_write_enable

    在 lui 指令中,需要写入寄存器,所以 io_reg_write_enable = 1

    # io_wb_reg_write_source[1:0]

    在 lui 指令中,写入寄存器的值为 ALU 计算结果,所以 io_wb_reg_write_source 的值为 00 ,指示选择 ALU 输出结果作为写入寄存器的数据源

    # io_memory_write_enable

    lui 指令没有写内存,所以 io_memory_write_enable = 0 ,避免误写内存

    # 指令

    总共执行了一个时钟周期 (4 ~ 6 ps),测试了 lui 指令在 ID 阶段的信号输出

    avatar

    # io_instruction[31:0]

    该信号直接来自 c.io.instruction.poke(002081b3L.U) ,对应二进制码为:= 0000000 00010 00001 000 00011 0110011 B ,其中 [6:0] 部分、 [14:12] 部分和 [31:25] 部分一起决定了当前指令为 add 指令;[19:15] 部分和 [24:20] 部分分别为操作数寄存器地址,对应信号 io_regs_reg1_read_address 和 io_regs_reg2_read_address ;[11:7] 部分作为目标寄存器的地址,对应 io_reg_write_address

    avatar

    # io_reg_write_address[4:0]

    该信号来自 io_instruction 的 [11:7] 部分,其值为 00011 ,对应 3 号寄存器,即结果要写入 3 号寄存器

    # io_regs_reg1_read_address[4:0]

    该信号来自 io_instruction 的 [19:15] 部分,为 00001B,对应 1 号寄存器

    # io_regs_reg2_read_address[4:0]

    该信号来自 io_instruction 的 [24:20] 部分,为 00010B,对应 2 号寄存器

    # io_ex_aluop1_source

    在 add 指令中, io_ex_aluop1_source = 0 ,指示读取寄存器的数据传给 ALU 作为其中一个操作数

    # io_ex_aluop2_source

    在 add 指令中, io_ex_aluop1_source = 0 ,指示读取寄存器的数据传给 ALU 作为其中一个操作数

    # io_reg_write_enable

    在 add 指令中,需要写入寄存器,所以 io_reg_write_enable = 1

    # io_wb_reg_write_source[1:0]

    在 add 指令中,写入寄存器的值为 ALU 计算结果,所以 io_wb_reg_write_source 的值为 00 ,指示选择 ALU 输出结果作为写入寄存器的数据源

    # io_memory_write_enable

    R 型指令 add 没有写内存,所以 io_memory_write_enable = 0 ,避免误写内存

    # InstructionFetchTest

    src/test/scala/riscv/singlecycle/InstructionFetchTest.scala

    avatar

    # 测试用例的功能

    模拟了 CPU 执行连续的 100 条随机指令时 InstructionFetch 模块能否正确顺序执行指令或跳转

    # 从什么层面测试 CPU

    从指令取指层面测试 CPU 的正确性

    # 加载测试程序指令的方法

    利用定义在 src/main/scala/riscv/core/InstructionFetch.scala 中的模块 InstructionFetch ,不使用任何 RISC-V 程序初始化进行仿真,直接仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    # 不跳转

    avatar

    此时 io_jump_flag = 0,指示不跳转,pc 的值每隔一个时钟周期 (2 ps) 就递增 4 ,指令顺序执行

    # 跳转

    avatar

    io_jump_flag = 1 时指示 pc 跳转,pc 的值在 io_jump_flag = 1 出现后的第二个时钟周期被置为跳转目标地址 io_jump_address_id = 00001000H,实现指令跳转

    # RegisterFileTest 1 : read the writing content

    src/test/scala/riscv/singlecycle/RegisterFileTest.scala

    avatar

    # 测试用例的功能

    测试了写入寄存器操作的时钟周期内目标寄存器内容的变化,即测试了写寄存器功能的正确性

    # 从什么层面测试 CPU

    从寄存器的读写层面测试了 CPU ,测试了寄存器模块

    # 加载测试程序指令的方法

    利用定义在 src/main/scala/riscv/core/RegisterFile.scala 中的模块 RegisterFile ,不使用任何 RISC-V 程序初始化进行仿真,直接仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    测试总共执行了 2 个时钟周期,第 1 个时钟周期内检测了 2 号寄存器内的值是否为 0 ,即是否为初始状态,同时测试程序模拟了一次写寄存器操作,向 2 号寄存器写入数据 DEADBEEFLH 。在第 2 个时钟周期再次检测 2 号寄存器的值,看其内容是否已经变为写入的数据。在这个过程中始终检测读取 2 号寄存器,所以信号 io_read_address 的内容始终为 2 ;第 1 个时钟周期模拟写入 2 号寄存器的过程,第 2 个时钟周期没有再写入过,所以信号 io_write_address 的内容始终为 2 且 io_write_data 的内容始终为 DEADBEEFLH ;在写寄存器操作后的下一个时钟周期上升沿到来时 register_2 [31:0] 的内容变为 DEADBEEFLH ,表明写寄存器操作正确执行;io_read_data_1 [31:0] 的值在第 2 个时钟周期上升沿到来前为 0 ,在第 2 个时钟周期上升沿到来时变为 DEADBEEFLH

    # RegisterFileTest 2 : read the written content

    src/test/scala/riscv/singlecycle/RegisterFileTest.scala

    avatar

    # 测试用例的功能

    测试了写入寄存器操作后的下一个时钟周期能否在目标寄存器读出写入的数据,即测试了写寄存器功能的正确性

    # 从什么层面测试 CPU

    从寄存器的读写层面测试了 CPU ,测试了寄存器模块

    # 加载测试程序指令的方法

    利用定义在 src/main/scala/riscv/core/RegisterFile.scala 中的模块 RegisterFile ,不使用任何 RISC-V 程序初始化进行仿真,直接仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    测试执行了一个时钟周期,测试程序在一个 timescope 内模拟了一次写入寄存器的操作,将数据 DEADBEEFLH 写入 1 号寄存器,这个过程将 io_write_enable 打开;将 io_write_address 赋值为 1 ,即写入 1 号寄存器;将 io_write_data 赋值为 DEADBEEFLH 。然后在下一个时钟周期模拟测试了能否从 1 号寄存器读出正确的数据,这个测试过程由于没有 clock.step() 所以没有显示在波形图中。但在波形图中可以看到第一个时钟周期后在下一个时钟上升沿到来之前,1 号寄存器的内容 register_1 [31:0] 以及变为了 DEADBEEFLH ,表明写入寄存器操作正确执行。

    # RegisterFileTest 3 : x0 always be zero

    src/test/scala/riscv/singlecycle/RegisterFileTest.scala

    avatar

    # 测试用例的功能

    测试了写入 0 号寄存器后 0 号寄存器的内容是否保持 0 不变

    # 从什么层面测试 CPU

    从寄存器的读写层面测试了 CPU ,测试了寄存器模块

    # 加载测试程序指令的方法

    利用定义在 src/main/scala/riscv/core/RegisterFile.scala 中的模块 RegisterFile ,不使用任何 RISC-V 程序初始化进行仿真,直接仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    测试总共执行了一个时钟周期,模拟了将写入目标寄存器设置为 0 号寄存器 x0 时的写寄存器操作。io_write_address = 0 ,表示写入 0 号寄存器;io_write_data = DEADBEEFH ,输入为要写入的数据;io_write_enable = 1 ,打开寄存器写使能;register_0 = 0 始终不变,表明测试成功,0 号寄存器的内容没有被改变始终为 0

    # FibonacciTest

    src/test/scala/riscv/singlecycle/CPUTest.scala

    avatar

    csrc/fibonacci.c

    avatar

    # 测试用例的功能

    测试了利用递归方法计算斐波那契数列第 10 项的程序能否在 CPU 上正常执行,并计算出结果存储到内存目标地址上

    # 从什么层面测试 CPU

    从最终完整执行一个程序的层面测试 CPU ,包括除了中断处理、与外设通信以外的所有 CPU 功能

    # 加载测试程序指令的方法

    利用定义在 src/test/scala/riscv/singlecycle/CPUTest.scala 中的模块 TestTopModule ,模拟一个简单的计算机,连接了 CPU 、Memory、InstructionRom 、ROMLoader几个模块。用 RISC-V 程序 src/main/resources/fibonacci.asmbin``` 初始化 instruction_rom 模块作为运行程序进行仿真

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    # 程序指令的执行

    为了保证程序完整执行完毕,测试程序循环执行了 50 次 c.clock.step(1000) ,总共执行了 50000 个时钟周期。其中大约前 20 ns 的时间内 CPU 各个信号不断变化,表明程序在执行。此后所有信号稳定,表明程序执行已经结束。

    avatar

    avatar

    从第 279 ps 开始 CPU 的信号开始发生变化,计算斐波那契数列的程序开始执行,到 22299 ps 为止 CPU 的所以信号稳定,表明 CPU 已经执行完毕。

    在执行过程中可以看到 io_instruction 信号以 8 ps 为周期变化一次,即 CPU 周期为 8 ps ,相当于 4 个时钟周期。

    avatar

    寄存器信号中可以发现,在程序运行过程中只有 1、2、8、10、11 号寄存器不断发生变化,也即程序只使用了除 0 号寄存器以外 5 个寄存器。

    # 程序结果的检测

    在 csrc/fibonacci.c 程序中可以看到,最终计算结果为斐波那契数列第 10 项,结果存储在地址为 4 的内存空间里。程序结果的检测执行了一个时钟周期

    avatar

    # io_mem_debug_read_address[31:0]

    该信号直接来自 c.io.mem_debug_read_address.poke(4.U) ,表示检测读取的内存地址为 4

    # io_mem_debug_read_data[31:0]

    一个时钟周期后读取内存地址为 4 的内容得到该信号的值为 00000037H = 55 ,表明程序结果正确

    # QuicksortTest

    src/test/scala/riscv/singlecycle/CPUTest.scala

    avatar

    csrc/quicksort.c

    avatar

    # 测试用例的功能

    测试程序给出了一个由 10 个阿拉伯数字组成的数组: {6,2,4,5,3,1,0,9,7,8} ,利用递归实现的 quicksort 函数对该数组进行排序,检测是否得到一个正确的有序数组

    # 从什么层面测试 CPU

    从最终完整执行一个程序的层面测试 CPU ,包括除了中断处理、与外设通信以外的所有 CPU 功能

    # 加载测试程序指令的方法

    利用定义在 src/test/scala/riscv/singlecycle/CPUTest.scala 中的模块 TestTopModule ,模拟一个简单的计算机,连接了 CPU 、Memory、InstructionRom 、ROMLoader 几个模块。用 RISC-V 程序 src/main/resources/quicksort.asmbin 初始化进行仿真

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    # 程序指令的执行

    为了保证程序完整执行完毕,测试程序循环执行了 50 次 c.clock.step(1000) ,总共执行了 50000 个时钟周期。其中大约前 15 ns 的时间内 CPU 各个信号不断变化,表明程序在执行。此后所有信号稳定,表明程序执行已经结束。

    avatar

    avatar

    从第 525 ps 开始 CPU 的信号开始发生变化,快排程序开始执行,到 13931 ps 为止 CPU 的所以信号稳定,表明 CPU 已经执行完毕。

    在执行过程中可以看到 io_instruction 信号以 8 ps 为周期变化一次,即 CPU 周期为 8 ps ,相当于 4 个时钟周期。

    avatar

    寄存器信号中可以发现,在程序运行过程中只有 1、2、8、10、11 、12 号寄存器不断发生变化,也即程序只使用了除 0 号寄存器以外 6 个寄存器。其中 1、2、8、12 号寄存器的变化呈现较同步的规律,说明这 4 个寄存器中的数据与递归有关;而 10、11 号寄存器的内容总是在频繁变化,说明这两个寄存器主要用来存储 ALU 运算的操作数

    # 程序结果的检测

    在 csrc/quicksort.c 程序中可以看到,排列得到的结果,结果存储在起始地址为 4 ,长度为 10 个 int 类型大小的内存空间里。程序结果的检测执行了 10 个时钟周期,利用循环程序对检测的内存地址递增,逐个访问目标内存,看所存储的内容是否上是从 0 到 9 的升序排列数组

    avatar

    # io_mem_debug_read_address[31:0]

    该信号直接来自 c.io.mem_debug_read_address.poke((4*i).U)i 从 1 到 10 递增, 4*i 从 4 到 40 变化,即检测读取的内存地址从 00000004H 到 00000028H 变化

    # io_mem_debug_read_data[31:0]

    内存地址从 00000004H 到 00000028H 变化的同时 io_mem_debug_read_data 读取内存的内容也从 0 到 9 变化,表明程序结果正确

    # ByteAccessTest

    src/test/scala/riscv/singlecycle/CPUTest.scala

    avatar

    csrc/sb.S

    avatar

    # 测试用例的功能

    在 RISCV 手册第 30 页可以查到寄存器编号即助记名如下:

    avatar

    对测试使用的 RISCV 汇编代码具体解释如下:

    RISCV汇编
    .globl _start 			;声明_start为全局符号
    	_start:         	;程序从这里开始执行
    	li a0, 0x4      	;将立即数 0x4 加载到寄存器 a0 (x10)
    	li t0, 0xDEADBEEF ;将立即数 0xDEADBEEF 加载到寄存器 t0(x5)
    	sb t0, 0(a0)    	;将寄存器 t0(x5) 的值存储到地址为 a0(x10) 的内存位置中
    	lw t1, 0(a0)    	;将地址为 a0(x10) 的内存位置中的值加载到寄存器 t1(x6)
    	li s2, 0x15    		;将立即数 0x15 加载到寄存器 s2(x18)
    	sb s2, 1(a0)   		;将寄存器 s2(x18) 的值存储到地址为 (a0+1) 的内存位置中
    	lw ra, 0(a0)    	;将地址为 a0(x10) 的内存位置中的值加载到寄存器 ra(x1)
    loop:           		;循环的标签
    	j loop          	;无条件地跳转到标签 loop,实现无限循环

    # 从什么层面测试 CPU

    从指令的层面测试 CPU ,测试了 CPU 对若干连续的 S 型和 L 型指令能否正确执行。主要检测了执行指令读写寄存器功能是否正确

    # 加载测试程序指令的方法

    利用定义在 src/test/scala/riscv/singlecycle/CPUTest.scala 中的模块 TestTopModule ,模拟一个简单的计算机,连接了 CPU 、Memory、InstructionRom 、ROMLoader 几个模块。用 RISC-V 程序 src/main/resources/sb.asmbin 初始化进行仿真

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    上图中可以看到有 5 个寄存器的值发生了变化:

    register_10 [31:0] 即 10 号寄存器 x10 或 a0 ,由 li a0, 0x4 赋值后信号稳定为 4

    register_5 [31:0] 即 5 号寄存器 x5 或 t0 ,由 li t0, 0xDEADBEEF 赋值时,由于寄存器写使能信号 io_reg_write_enable 此时仍保持打开的,因为上一条指令 li a0, 0x4 执行时也打开了寄存器写使能,所以此时写入目标寄存器 5 号寄存器的内容出现了毛刺 DEADC000H ,但在下一个时钟周期恢复正确值 DEADBEEFH 并稳定

    register_6 [31:0] 即 6 号寄存器 x6 或 t1 ,由 lw t1, 0(a0) 赋值后信号稳定。内存地址 DEADBEEFH 的内容由 sb t0, 0(a0) 赋值,而 sb 指令存储时只存低位字节,且 RISCV 为小端存储模式,所以存入内存的只有 EFH ,读取内存时 EFH 即为最低位字节,存到 6 号寄存器的内容为 000000EFH

    register_18 [31:0] 即 18 号寄存器 x18 或 s2 ,由 li s2, 0x15 赋值后信号稳定为 15H

    register_1 [31:0] 即 1 号寄存器 x1 或 ra,由 lw ra, 0(a0) 赋值后信号稳定。内存地址 DEADBEEFH 的内容原本只有只有 EFH ,sb s2, 1 (a0) 再次对该地址赋值,而 sb 指令存储时只存低位字节,按照 RISCV 的小端存储模式,此时存入内存的只有 15H ,且存到 DEADBEEFH +1 的地址,即存到 EFH 的下一个地址。读取内存时字节从低到高为 EFH、15H ,存到 1 号寄存器的内容为 000015EFH

    测试程序检测了 ,与波形图对应寄存器内容的信号最后稳定的值相同,表明指令正确执行

    # 填空涉及到的信号分析

    (src/main/scala/riscv/)

    # core/InstructionFetch.scala

    # io.jump_flag_id

    该控制信号告诉 CPU 取指阶段下一条指令是否需要跳转,如果是则将 PC 设置为跳转目标指令地址,否则 PC+4,下一个时钟周期顺序读取下一条指令

    # core/InstructionDecode.scala

    # io.ex_aluop2_source

    改控制信号控制 ALU 输入端的第二个输入源,分别有以下两种:

    avatar

    ALUOp2Source.Immediate:
    将 io.ex_aluop2_source 设置为该值后 ALU 输入端端第二个输入源是来自指令的立即数

    ALUOp2Source.Register:
    将 io.ex_aluop2_source 设置为该值后 ALU 输入端端第二个输入源是读取寄存器得到的数据

    avatar

    根据以上 RISC-V 手册第 25 页指令译码过程可以知道,需要使用到立即数的指令有:

    • U 型指令
      • lui
      • auipc
    • J 型指令
      • jal
      • jalr
    • I 型指令
    • L 型指令
    • B 型指令
    • 其他
      • fence 同步内存和 I/O
      • CSR 型指令
      • nop 伪指令

    因此当执行上述指令时,将 io.ex_aluop2_source 设置为 ALUOp2Source.Immediate

    # io.memory_read_enable

    RISCV 中只有 L 型指令需要读取内存,所以执行 L 型指令时将 io.memory_read_enable 设置为 1,否则均设置为 0

    # io.memory_write_enable

    RISCV 中只有 S 型指令需要写内存,所以执行 S 型指令时将 io.memory_write_enable 设置为 1,否则均设置为 0

    # io.wb_reg_write_source

    查阅手册各个指令的功能可知,需要写回寄存器的指令及其对应写入数据源如下:

    • R 型指令:ALU 计算结果
    • U 型指令
      • lui:ALU 计算结果
      • auipc:ALU 计算结果
    • I 型指令:ALU 计算结果
    • L 型指令:内存
    • 其他
      • nop 伪指令:ALU 计算结果

    不考虑中断的 CSR 作为写入数据源,其余指令均可设置 io.wb_reg_write_source 为 NextInstructionAddress (作为默认值)

    # core/Execute.scala

    # alu.io.func

    该信号传给 ALU ,决定 ALU 执行何种运算,实际上该信号的取值为 src/main/scala/riscv/core/ALU.scala 定义的 ChiselEnum 枚举类型,在 “ alu.io.func

    # alu.io.op1

    该信号传给 ALU,决定 ALU 的第一个操作数的数据源为 PC 还是寄存器。其控制已在 InstructionDecode 模块实现,其传入 Execute 模块的信号为 io.aluop1_source ,根据该信号即可确定 alu.io.op1

    # alu.io.op2

    该信号传给 ALU,决定 ALU 的第一个操作数的数据源为立即数还是寄存器。其控制已在 InstructionDecode 模块实现,其传入 Execute 模块的信号为 io.aluop2_source ,根据该信号即可确定 alu.io.op2

    # core/CPU.scala

    所填代码主要涉及传递给 Execute 模块的信号,各个赋值语句对应了连线图的控制信号连线、模块之间的端口的连接

    # ex.io.instruction := inst_fetch.io.instruction

    取指阶段得到的指令传递给 Execute 模块

    # ex.io.instruction_address := inst_fetch.io.instruction_address

    取指阶段得到的指令地址传递给 Execute 模块

    # ex.io.reg1_data := regs.io.read_data1

    寄存器组将读取的第 1 个数据传递给 Execute 模块

    # ex.io.reg2_data := regs.io.read_data2

    寄存器组将读取的第 2 个数据传递给 Execute 模块

    # ex.io.immediate := id.io.ex_immediate

    译码阶段得到的立即数传递给 Execute 模块

    # ex.io.aluop1_source := id.io.ex_aluop1_source

    译码阶段得到的 ALU 第 1 个操作数的数据源的控制信号传递给 Execute 模块

    # ex.io.aluop2_source := id.io.ex_aluop2_source

    译码阶段得到的 ALU 第 2 个操作数的数据源的控制信号传递给 Execute 模块

    # 在完成实验的过程中,遇到的关于实验指导不明确或者其他问题,或者改进的建议。

    使用实验板上的数码管、视频输出等外设,输出一个完整程序的运行结果,或者参考硬件调试一节的内容,用硬件波形的方法捕获程序运行结果。给出你的 CPU 可以正确在实验板上运行程序的照片或者硬件调试器波形截图。

    \ No newline at end of file +lab1 - YatCPU - 计算机组成原理 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    YatCPU - - - lab1 - - - lab2 - - - lab3 - - - lab4

    # 测试用例及其波形图分析

    # SayGoodbyeTest

    src/test/scala/riscv/BoardTest.scala

    avatar

    其中 Top 模块来自:

    avatar

    # 测试用例的功能

    用于烧板验证,但由于 CPU 还未能实现中断,该程序只能通过 LED 灯闪烁判断 CPU 能否正常工作

    在 lab1 中该测试用例可以用于检查 CPU 及其内部各个模块、模块内的端口信号是否有完整地实现、初始化,即 "Fully initialized"。若有信号定义后没有完整实现,则该项会测试后报错: Reference xxx is not fully initialized. 则表明可能该处有代码空缺没有填补完整。

    # 从什么层面测试 CPU

    从完整性的层面测试 CPU,CPU 能否正常工作

    但测试程序并没有测试 CPU 的正确性,只是进行了 pokestep ,没有对任何信号进行 expect 检查。

    # 加载测试程序的方法

    利用定义在 ssrc/main/scala/board/z710/Top.scala 中的模块 Top ,用 RISC-V 程序 src/main/resources/say_goodbye.asmbin 初始化进行仿真

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    该测试只向波形文件写入了 Top 模块的信号

    # 执行不同指令时候对应的部件的关键信号的变化情况

    由 clock 信号变化可以看出时钟周期为 2 ps

    测试中循环执行了 50000 次 step(5) ,即 250000 个时钟周期,输出的波形图总时长为 500001 ps ,除去半个用于 reset 的时钟周期之外恰好 250000 个时钟周期。

    可以看到波形图中除了 clock 信号有规律的周期性变化之外, io_rx 也在周期性地在 0 和 1 之间变化。从波形图也可以看出每 10 ps 即 5 个时钟周期 io_rx 变化一次,符合 step(5) 一次 c.io.rx.poke((i % 2).U) 一次的变化规律。

    # ExecuteTest

    src/test/scala/riscv/singlecycle/ExecuteTest.scala

    avatar

    # 测试用例的功能

    该测试用例用于检查 Execute 模块的正确性,包括:

    # 执行 ALU 加法运算

    测试程序循环执行了 100 次两个随机生成的 int 整型操作数 op1 和 op2 的求和运算,检查 R 型指令 add 的正确性。

    同时检查跳转信号是否始终保持 0 (fasle),确保 CPU 在非跳转指令下不跳转

    # 处理 beq 指令是否跳转

    测试程序按顺序测试了两个情况:

    跳转,即 op1 等于 op2 ,此时跳转信号应该为 1 (true),原指令地址 (PC):2 应该与立即数 2 相加得到新的指令地址 (PC):4

    不跳转,即 op1 不等于 op2 ,此时跳转信号应该为 0 (false),指令地址 (PC) 保持 4 不变

    # 从什么层面测试 CPU

    从 CPU 运算处理的层面测试 CPU ,即能否正确控制 ALU 的输入输出,以及输入输出结果对指令跳转的控制

    # 加载测试程序的方法

    利用定义在 src/main/scala/riscv/core/Execute.scala 中的模块 Execute ,不使用任何 RISC-V 程序初始化进行仿真,直接仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    # ALU 加法运算

    总共执行了 102 个时钟周期,测试了 101 次随机样例 (循环从 i=0 到 i=100 执行了 101 次)。

    avatar

    # io_funct3[2:0]

    该信号始终保持 000B 不变,与 io_opcode 一起指示 ALU 执行 R 类型指令 add 的加法运算

    # io_opcode[6:0]

    该信号始终保持 0110011B 不变,与 io_funct3 一起指示 ALU 执行 R 类型指令 add 的加法运算

    avatar

    # io_alu_funct[3:0]

    按照 src/main/scala/riscv/core/ALU.scala 定义的 ChiselEnum 枚举类型如下:

    avatar

    由于总共有 11 种取值,默认从 0 开始计数直到 10 ,需要二进制 4 位数才能表示,因此对应的 io_alu_funct 为 4 位二进制数

    在加法运算时该信号始终保持 0001B 不变,对应 ChiselEnu 类型定义的 ALUFunctions 中从 0 开始计数的第二个 add,指示 ALU 执行加法运算

    # io_op1[31:0]

    该信号来自 c.io.reg1_data.poke(op1.U) ,在 add 指令中 ALU 的运算操作数 op1 来自寄存器,因此 Execute 模块的 io_reg1_data 直接传递给 ALU 模块的 io_op1

    每隔一个时钟周期 (2 ps) 变化一次,每次均为一个新的随机 int 型整数

    # io_op2[31:0]

    该信号来自 c.io.reg2_data.poke(op1.U) ,在 add 指令中 ALU 的运算操作数 op2 来自寄存器,因此 Execute 模块的 io_reg2_data 直接传递给 ALU 模块的 io_op2

    每隔一个时钟周期 (2 ps) 变化一次,每次均为一个新的随机 int 型整数

    # io_result[31:0]

    每隔一个时钟周期 (2 ps) 变化一次,每次均为当前 io_op1 与 io_op2 的算术和,同时该值直接传递给 Execute 模块的 mem_alu_result

    # 指令

    总共执行 3 个时钟周期,先后测试了成功跳转和不成功跳转两种情况。

    avatar

    # io_funct3[2:0]

    该信号始终保持 000B 不变,与 io_opcode 一起指示 ALU 执行 B 类型指令 beq 的加法运算计算跳转目标地址,以及控制跳转信号 io_if_jump 的产生

    # io_opcode[6:0]

    该信号始终保持 1100011B 不变,与 io_funct3 一起指示 ALU 执行 B 类型指令 beq 的加法运算计算跳转目标地址,以及控制跳转信号 io_if_jump 的产生

    avatar

    # io_reg1_data[31:0]

    该信号表示所读取的来自寄存器的数据,在 beq 指令中直接将两个寄存器的值进行比较得到跳转信号 io_if_jump

    在测试的后 2 个时钟周期内其值均为 9 ,来自测试程序中的 c.io.reg1_data.poke(9.U)

    # io_reg2_data[31:0]

    该信号表示所读取的来自寄存器的数据,在 beq 指令中直接将两个寄存器的值进行比较得到跳转信号 io_if_jump

    在第 2 个时钟周期内其值为 9 ,来自测试程序中的 c.io.reg2_data.poke(9.U)

    在第 3 个时钟周期内其值为 19 ,来自测试程序中的 c.io.reg2_data.poke(19.U)

    # io_if_jump

    该信号表示 PC 下一条指令是否要跳转,是则 io_if_jump = 1 ,否则 io_if_jump = 0

    在第 2 个时钟周期其值为 1 ,因为此时 io_reg1_data = 9 = io_reg2_data

    在第 3 个时钟周期其值为 0 ,因为此时 io_reg1_data = 9 , io_reg2_data = 19

    # io_alu_funct[3:0]

    在 beq 指令中,ALU 执行加法运算,io_alu_funct = 1

    # alu_io_op1[31:0]

    由于 c.io.aluop1_source.poke(1.U) ,aluop1_source = 1 指示该信号 alu_io_op1 来自 instruction_address ,同时 c.io.instruction_address.poke(2.U) ,因此 alu_io_op1 = 2

    # alu_io_op2[31:0]

    由于 c.io.aluop2_source.poke(1.U) ,aluop2_source = 1 指示该信号 alu_io_op2 来自立即数 io_immediate ,同时 c.io.immediate.poke(2.U) ,因此 alu_io_op2 = 2

    # io_if_jump_address[31:0]

    表示跳转成功时 PC 跳转的目标指令地址,在 beq 指令中等于 ALU 计算出来的结果 alu_io_result

    在 2 个时钟周期内均为 4 ,即 alu_io_op1 与 alu_io_op2 的和

    # InstructionDecoderTest

    src/test/scala/riscv/singlecycle/InstructionDecoderTest.scala

    avatar

    # 测试用例的功能

    测试了 InstructionDecode 模块的正确性,包括:

    # S 型指令

    测试了对 S 型指令 sw 翻译的正确性,包括读取正确的寄存器并且将正确的操作数传递给 ALU。

    在 S 型指令中,存储的数据来自寄存器,存储目标的内存地址由另一寄存器中的数据和指令中的立即数作为偏移量共同决定。

    # lui 指令

    测试了对高位加载立即数指令 lui 翻译的正确性,包括读取正确的寄存器并且将正确的操作数传递给 ALU

    在 lui 指令中,符号位扩展的 20 位立即数左移 12 位后,将低 12 位置零,传入 ALU,同时 ALU 另一个操作数设置为 0 号寄存器 (内容始终为 0) 的值,即另一个操作数为 0,ALU 做加法运算,相当于将位扩展后的结果直接作为写入的值,写入目标寄存器中

    # add 指令

    测试了对 R 型指令 add 翻译的正确性,主要测试了是否控制选择寄存器作为 ALU 操作数

    # 从什么层面测试 CPU

    从指令的翻译、分析层面,指导 CPU 如何根据当前指令进行工作,包括立即数的提取、各个控制信号的取值等

    # 加载测试程序指令的方法

    利用定义在 src/main/scala/riscv/core/InstructionDecode.scala 中的模块 InstructionDecode ,不使用任何 RISC-V 程序初始化进行仿真,直接仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    # S 型指令

    总共执行了一个时钟周期 (2 ~ 4 ps),测试了 S 型指令 sw 在 ID 阶段的信号输出

    avatar

    # io_instruction[31:0]

    该信号直接来自 c.io.instruction.poke(00a02223L.U) ,对应二进制码为:0000000 0

    avatar

    # io_ex_immediate[31:0]

    该信号由 io_instruction 的 [11:5] 部分 0000000B 和 [11:7] 部分 00100B 组成 004H ,再位扩展后得到 00000004H = 4,作为立即数传入 ALU

    # io_regs_reg1_read_address[4:0]

    该信号来自 io_instruction 的 [19:15] 部分,为 00000B,对应 0 号寄存器

    # io_regs_reg2_read_address[4:0]

    该信号来自 io_instruction 的 [24:20] 部分,为 01010B,对应 10 号寄存器

    # io_ex_aluop1_source

    在 S 型指令中, io_ex_aluop1_source = 0 ,指示读取寄存器的数据传给 ALU 作为其中一个操作数

    # io_ex_aluop2_source

    在 S 型指令中, io_ex_aluop2_source = 1 ,指示将立即数 io_ex_immediate 传给 ALU 作为其中一个操作数

    # io_memory_read_enable

    S 型指令没有读取内存,所以 io_memory_read_enable = 0

    # io_memory_write_enable

    S 型指令需要写内存,所以 io_memory_write_enable = 1

    # lui 指令

    总共执行了一个时钟周期 (4 ~ 6 ps),测试了 lui 指令在 ID 阶段的信号输出

    avatar

    # io_instruction[31:0]

    该信号直接来自 c.io.instruction.poke(000022b7L.U) ,对应二进制码为:00000000000000000010 00101 01

    avatar

    # io_ex_immediate[31:0]

    该信号由 io_instruction 的 [31:12] 部分 00002H 位扩展后得到 00002000H = 2,作为立即数传入 ALU

    # io_reg_write_address[4:0]

    该信号来自 io_instruction 的 [11:7] 部分,其值为 00101 ,对应 5 号寄存器,即结果要写入 5 号寄存器

    # io_ex_aluop1_source

    在 lui 指令中, io_ex_aluop1_source = 0 ,指示读取寄存器 (0 号寄存器) 的数据 ( 0 ) 传给 ALU 作为其中一个操作数

    # io_ex_aluop2_source

    在 lui 指令中, io_ex_aluop1_source = 1 ,指示将立即数 io_ex_immediate 传给 ALU 作为其中一个操作数

    # io_regs_reg1_read_address[4:0]

    在 lui 指令中, 该值被设置为 00000 ,对应 0 号寄存器

    # io_reg_write_enable

    在 lui 指令中,需要写入寄存器,所以 io_reg_write_enable = 1

    # io_wb_reg_write_source[1:0]

    在 lui 指令中,写入寄存器的值为 ALU 计算结果,所以 io_wb_reg_write_source 的值为 00 ,指示选择 ALU 输出结果作为写入寄存器的数据源

    # io_memory_write_enable

    lui 指令没有写内存,所以 io_memory_write_enable = 0 ,避免误写内存

    # 指令

    总共执行了一个时钟周期 (4 ~ 6 ps),测试了 lui 指令在 ID 阶段的信号输出

    avatar

    # io_instruction[31:0]

    该信号直接来自 c.io.instruction.poke(002081b3L.U) ,对应二进制码为:= 0000000 00010 00001 000 00011 0110011 B ,其中 [6:0] 部分、 [14:12] 部分和 [31:25] 部分一起决定了当前指令为 add 指令;[19:15] 部分和 [24:20] 部分分别为操作数寄存器地址,对应信号 io_regs_reg1_read_address 和 io_regs_reg2_read_address ;[11:7] 部分作为目标寄存器的地址,对应 io_reg_write_address

    avatar

    # io_reg_write_address[4:0]

    该信号来自 io_instruction 的 [11:7] 部分,其值为 00011 ,对应 3 号寄存器,即结果要写入 3 号寄存器

    # io_regs_reg1_read_address[4:0]

    该信号来自 io_instruction 的 [19:15] 部分,为 00001B,对应 1 号寄存器

    # io_regs_reg2_read_address[4:0]

    该信号来自 io_instruction 的 [24:20] 部分,为 00010B,对应 2 号寄存器

    # io_ex_aluop1_source

    在 add 指令中, io_ex_aluop1_source = 0 ,指示读取寄存器的数据传给 ALU 作为其中一个操作数

    # io_ex_aluop2_source

    在 add 指令中, io_ex_aluop1_source = 0 ,指示读取寄存器的数据传给 ALU 作为其中一个操作数

    # io_reg_write_enable

    在 add 指令中,需要写入寄存器,所以 io_reg_write_enable = 1

    # io_wb_reg_write_source[1:0]

    在 add 指令中,写入寄存器的值为 ALU 计算结果,所以 io_wb_reg_write_source 的值为 00 ,指示选择 ALU 输出结果作为写入寄存器的数据源

    # io_memory_write_enable

    R 型指令 add 没有写内存,所以 io_memory_write_enable = 0 ,避免误写内存

    # InstructionFetchTest

    src/test/scala/riscv/singlecycle/InstructionFetchTest.scala

    avatar

    # 测试用例的功能

    模拟了 CPU 执行连续的 100 条随机指令时 InstructionFetch 模块能否正确顺序执行指令或跳转

    # 从什么层面测试 CPU

    从指令取指层面测试 CPU 的正确性

    # 加载测试程序指令的方法

    利用定义在 src/main/scala/riscv/core/InstructionFetch.scala 中的模块 InstructionFetch ,不使用任何 RISC-V 程序初始化进行仿真,直接仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    # 不跳转

    avatar

    此时 io_jump_flag = 0,指示不跳转,pc 的值每隔一个时钟周期 (2 ps) 就递增 4 ,指令顺序执行

    # 跳转

    avatar

    io_jump_flag = 1 时指示 pc 跳转,pc 的值在 io_jump_flag = 1 出现后的第二个时钟周期被置为跳转目标地址 io_jump_address_id = 00001000H,实现指令跳转

    # RegisterFileTest 1 : read the writing content

    src/test/scala/riscv/singlecycle/RegisterFileTest.scala

    avatar

    # 测试用例的功能

    测试了写入寄存器操作的时钟周期内目标寄存器内容的变化,即测试了写寄存器功能的正确性

    # 从什么层面测试 CPU

    从寄存器的读写层面测试了 CPU ,测试了寄存器模块

    # 加载测试程序指令的方法

    利用定义在 src/main/scala/riscv/core/RegisterFile.scala 中的模块 RegisterFile ,不使用任何 RISC-V 程序初始化进行仿真,直接仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    测试总共执行了 2 个时钟周期,第 1 个时钟周期内检测了 2 号寄存器内的值是否为 0 ,即是否为初始状态,同时测试程序模拟了一次写寄存器操作,向 2 号寄存器写入数据 DEADBEEFLH 。在第 2 个时钟周期再次检测 2 号寄存器的值,看其内容是否已经变为写入的数据。在这个过程中始终检测读取 2 号寄存器,所以信号 io_read_address 的内容始终为 2 ;第 1 个时钟周期模拟写入 2 号寄存器的过程,第 2 个时钟周期没有再写入过,所以信号 io_write_address 的内容始终为 2 且 io_write_data 的内容始终为 DEADBEEFLH ;在写寄存器操作后的下一个时钟周期上升沿到来时 register_2 [31:0] 的内容变为 DEADBEEFLH ,表明写寄存器操作正确执行;io_read_data_1 [31:0] 的值在第 2 个时钟周期上升沿到来前为 0 ,在第 2 个时钟周期上升沿到来时变为 DEADBEEFLH

    # RegisterFileTest 2 : read the written content

    src/test/scala/riscv/singlecycle/RegisterFileTest.scala

    avatar

    # 测试用例的功能

    测试了写入寄存器操作后的下一个时钟周期能否在目标寄存器读出写入的数据,即测试了写寄存器功能的正确性

    # 从什么层面测试 CPU

    从寄存器的读写层面测试了 CPU ,测试了寄存器模块

    # 加载测试程序指令的方法

    利用定义在 src/main/scala/riscv/core/RegisterFile.scala 中的模块 RegisterFile ,不使用任何 RISC-V 程序初始化进行仿真,直接仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    测试执行了一个时钟周期,测试程序在一个 timescope 内模拟了一次写入寄存器的操作,将数据 DEADBEEFLH 写入 1 号寄存器,这个过程将 io_write_enable 打开;将 io_write_address 赋值为 1 ,即写入 1 号寄存器;将 io_write_data 赋值为 DEADBEEFLH 。然后在下一个时钟周期模拟测试了能否从 1 号寄存器读出正确的数据,这个测试过程由于没有 clock.step() 所以没有显示在波形图中。但在波形图中可以看到第一个时钟周期后在下一个时钟上升沿到来之前,1 号寄存器的内容 register_1 [31:0] 以及变为了 DEADBEEFLH ,表明写入寄存器操作正确执行。

    # RegisterFileTest 3 : x0 always be zero

    src/test/scala/riscv/singlecycle/RegisterFileTest.scala

    avatar

    # 测试用例的功能

    测试了写入 0 号寄存器后 0 号寄存器的内容是否保持 0 不变

    # 从什么层面测试 CPU

    从寄存器的读写层面测试了 CPU ,测试了寄存器模块

    # 加载测试程序指令的方法

    利用定义在 src/main/scala/riscv/core/RegisterFile.scala 中的模块 RegisterFile ,不使用任何 RISC-V 程序初始化进行仿真,直接仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    测试总共执行了一个时钟周期,模拟了将写入目标寄存器设置为 0 号寄存器 x0 时的写寄存器操作。io_write_address = 0 ,表示写入 0 号寄存器;io_write_data = DEADBEEFH ,输入为要写入的数据;io_write_enable = 1 ,打开寄存器写使能;register_0 = 0 始终不变,表明测试成功,0 号寄存器的内容没有被改变始终为 0

    # FibonacciTest

    src/test/scala/riscv/singlecycle/CPUTest.scala

    avatar

    csrc/fibonacci.c

    avatar

    # 测试用例的功能

    测试了利用递归方法计算斐波那契数列第 10 项的程序能否在 CPU 上正常执行,并计算出结果存储到内存目标地址上

    # 从什么层面测试 CPU

    从最终完整执行一个程序的层面测试 CPU ,包括除了中断处理、与外设通信以外的所有 CPU 功能

    # 加载测试程序指令的方法

    利用定义在 src/test/scala/riscv/singlecycle/CPUTest.scala 中的模块 TestTopModule ,模拟一个简单的计算机,连接了 CPU 、Memory、InstructionRom 、ROMLoader几个模块。用 RISC-V 程序 src/main/resources/fibonacci.asmbin``` 初始化 instruction_rom 模块作为运行程序进行仿真

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    # 程序指令的执行

    为了保证程序完整执行完毕,测试程序循环执行了 50 次 c.clock.step(1000) ,总共执行了 50000 个时钟周期。其中大约前 20 ns 的时间内 CPU 各个信号不断变化,表明程序在执行。此后所有信号稳定,表明程序执行已经结束。

    avatar

    avatar

    从第 279 ps 开始 CPU 的信号开始发生变化,计算斐波那契数列的程序开始执行,到 22299 ps 为止 CPU 的所以信号稳定,表明 CPU 已经执行完毕。

    在执行过程中可以看到 io_instruction 信号以 8 ps 为周期变化一次,即 CPU 周期为 8 ps ,相当于 4 个时钟周期。

    avatar

    寄存器信号中可以发现,在程序运行过程中只有 1、2、8、10、11 号寄存器不断发生变化,也即程序只使用了除 0 号寄存器以外 5 个寄存器。

    # 程序结果的检测

    在 csrc/fibonacci.c 程序中可以看到,最终计算结果为斐波那契数列第 10 项,结果存储在地址为 4 的内存空间里。程序结果的检测执行了一个时钟周期

    avatar

    # io_mem_debug_read_address[31:0]

    该信号直接来自 c.io.mem_debug_read_address.poke(4.U) ,表示检测读取的内存地址为 4

    # io_mem_debug_read_data[31:0]

    一个时钟周期后读取内存地址为 4 的内容得到该信号的值为 00000037H = 55 ,表明程序结果正确

    # QuicksortTest

    src/test/scala/riscv/singlecycle/CPUTest.scala

    avatar

    csrc/quicksort.c

    avatar

    # 测试用例的功能

    测试程序给出了一个由 10 个阿拉伯数字组成的数组: {6,2,4,5,3,1,0,9,7,8} ,利用递归实现的 quicksort 函数对该数组进行排序,检测是否得到一个正确的有序数组

    # 从什么层面测试 CPU

    从最终完整执行一个程序的层面测试 CPU ,包括除了中断处理、与外设通信以外的所有 CPU 功能

    # 加载测试程序指令的方法

    利用定义在 src/test/scala/riscv/singlecycle/CPUTest.scala 中的模块 TestTopModule ,模拟一个简单的计算机,连接了 CPU 、Memory、InstructionRom 、ROMLoader 几个模块。用 RISC-V 程序 src/main/resources/quicksort.asmbin 初始化进行仿真

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    # 程序指令的执行

    为了保证程序完整执行完毕,测试程序循环执行了 50 次 c.clock.step(1000) ,总共执行了 50000 个时钟周期。其中大约前 15 ns 的时间内 CPU 各个信号不断变化,表明程序在执行。此后所有信号稳定,表明程序执行已经结束。

    avatar

    avatar

    从第 525 ps 开始 CPU 的信号开始发生变化,快排程序开始执行,到 13931 ps 为止 CPU 的所以信号稳定,表明 CPU 已经执行完毕。

    在执行过程中可以看到 io_instruction 信号以 8 ps 为周期变化一次,即 CPU 周期为 8 ps ,相当于 4 个时钟周期。

    avatar

    寄存器信号中可以发现,在程序运行过程中只有 1、2、8、10、11 、12 号寄存器不断发生变化,也即程序只使用了除 0 号寄存器以外 6 个寄存器。其中 1、2、8、12 号寄存器的变化呈现较同步的规律,说明这 4 个寄存器中的数据与递归有关;而 10、11 号寄存器的内容总是在频繁变化,说明这两个寄存器主要用来存储 ALU 运算的操作数

    # 程序结果的检测

    在 csrc/quicksort.c 程序中可以看到,排列得到的结果,结果存储在起始地址为 4 ,长度为 10 个 int 类型大小的内存空间里。程序结果的检测执行了 10 个时钟周期,利用循环程序对检测的内存地址递增,逐个访问目标内存,看所存储的内容是否上是从 0 到 9 的升序排列数组

    avatar

    # io_mem_debug_read_address[31:0]

    该信号直接来自 c.io.mem_debug_read_address.poke((4*i).U)i 从 1 到 10 递增, 4*i 从 4 到 40 变化,即检测读取的内存地址从 00000004H 到 00000028H 变化

    # io_mem_debug_read_data[31:0]

    内存地址从 00000004H 到 00000028H 变化的同时 io_mem_debug_read_data 读取内存的内容也从 0 到 9 变化,表明程序结果正确

    # ByteAccessTest

    src/test/scala/riscv/singlecycle/CPUTest.scala

    avatar

    csrc/sb.S

    avatar

    # 测试用例的功能

    在 RISCV 手册第 30 页可以查到寄存器编号即助记名如下:

    avatar

    对测试使用的 RISCV 汇编代码具体解释如下:

    RISCV汇编
    .globl _start 			;声明_start为全局符号
    	_start:         	;程序从这里开始执行
    	li a0, 0x4      	;将立即数 0x4 加载到寄存器 a0 (x10)
    	li t0, 0xDEADBEEF ;将立即数 0xDEADBEEF 加载到寄存器 t0(x5)
    	sb t0, 0(a0)    	;将寄存器 t0(x5) 的值存储到地址为 a0(x10) 的内存位置中
    	lw t1, 0(a0)    	;将地址为 a0(x10) 的内存位置中的值加载到寄存器 t1(x6)
    	li s2, 0x15    		;将立即数 0x15 加载到寄存器 s2(x18)
    	sb s2, 1(a0)   		;将寄存器 s2(x18) 的值存储到地址为 (a0+1) 的内存位置中
    	lw ra, 0(a0)    	;将地址为 a0(x10) 的内存位置中的值加载到寄存器 ra(x1)
    loop:           		;循环的标签
    	j loop          	;无条件地跳转到标签 loop,实现无限循环

    # 从什么层面测试 CPU

    从指令的层面测试 CPU ,测试了 CPU 对若干连续的 S 型和 L 型指令能否正确执行。主要检测了执行指令读写寄存器功能是否正确

    # 加载测试程序指令的方法

    利用定义在 src/test/scala/riscv/singlecycle/CPUTest.scala 中的模块 TestTopModule ,模拟一个简单的计算机,连接了 CPU 、Memory、InstructionRom 、ROMLoader 几个模块。用 RISC-V 程序 src/main/resources/sb.asmbin 初始化进行仿真

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    上图中可以看到有 5 个寄存器的值发生了变化:

    register_10 [31:0] 即 10 号寄存器 x10 或 a0 ,由 li a0, 0x4 赋值后信号稳定为 4

    register_5 [31:0] 即 5 号寄存器 x5 或 t0 ,由 li t0, 0xDEADBEEF 赋值时,由于寄存器写使能信号 io_reg_write_enable 此时仍保持打开的,因为上一条指令 li a0, 0x4 执行时也打开了寄存器写使能,所以此时写入目标寄存器 5 号寄存器的内容出现了毛刺 DEADC000H ,但在下一个时钟周期恢复正确值 DEADBEEFH 并稳定

    register_6 [31:0] 即 6 号寄存器 x6 或 t1 ,由 lw t1, 0(a0) 赋值后信号稳定。内存地址 DEADBEEFH 的内容由 sb t0, 0(a0) 赋值,而 sb 指令存储时只存低位字节,且 RISCV 为小端存储模式,所以存入内存的只有 EFH ,读取内存时 EFH 即为最低位字节,存到 6 号寄存器的内容为 000000EFH

    register_18 [31:0] 即 18 号寄存器 x18 或 s2 ,由 li s2, 0x15 赋值后信号稳定为 15H

    register_1 [31:0] 即 1 号寄存器 x1 或 ra,由 lw ra, 0(a0) 赋值后信号稳定。内存地址 DEADBEEFH 的内容原本只有只有 EFH ,sb s2, 1 (a0) 再次对该地址赋值,而 sb 指令存储时只存低位字节,按照 RISCV 的小端存储模式,此时存入内存的只有 15H ,且存到 DEADBEEFH +1 的地址,即存到 EFH 的下一个地址。读取内存时字节从低到高为 EFH、15H ,存到 1 号寄存器的内容为 000015EFH

    测试程序检测了 ,与波形图对应寄存器内容的信号最后稳定的值相同,表明指令正确执行

    # 填空涉及到的信号分析

    (src/main/scala/riscv/)

    # core/InstructionFetch.scala

    # io.jump_flag_id

    该控制信号告诉 CPU 取指阶段下一条指令是否需要跳转,如果是则将 PC 设置为跳转目标指令地址,否则 PC+4,下一个时钟周期顺序读取下一条指令

    # core/InstructionDecode.scala

    # io.ex_aluop2_source

    改控制信号控制 ALU 输入端的第二个输入源,分别有以下两种:

    avatar

    ALUOp2Source.Immediate:
    将 io.ex_aluop2_source 设置为该值后 ALU 输入端端第二个输入源是来自指令的立即数

    ALUOp2Source.Register:
    将 io.ex_aluop2_source 设置为该值后 ALU 输入端端第二个输入源是读取寄存器得到的数据

    avatar

    根据以上 RISC-V 手册第 25 页指令译码过程可以知道,需要使用到立即数的指令有:

    • U 型指令
      • lui
      • auipc
    • J 型指令
      • jal
      • jalr
    • I 型指令
    • L 型指令
    • B 型指令
    • 其他
      • fence 同步内存和 I/O
      • CSR 型指令
      • nop 伪指令

    因此当执行上述指令时,将 io.ex_aluop2_source 设置为 ALUOp2Source.Immediate

    # io.memory_read_enable

    RISCV 中只有 L 型指令需要读取内存,所以执行 L 型指令时将 io.memory_read_enable 设置为 1,否则均设置为 0

    # io.memory_write_enable

    RISCV 中只有 S 型指令需要写内存,所以执行 S 型指令时将 io.memory_write_enable 设置为 1,否则均设置为 0

    # io.wb_reg_write_source

    查阅手册各个指令的功能可知,需要写回寄存器的指令及其对应写入数据源如下:

    • R 型指令:ALU 计算结果
    • U 型指令
      • lui:ALU 计算结果
      • auipc:ALU 计算结果
    • I 型指令:ALU 计算结果
    • L 型指令:内存
    • 其他
      • nop 伪指令:ALU 计算结果

    不考虑中断的 CSR 作为写入数据源,其余指令均可设置 io.wb_reg_write_source 为 NextInstructionAddress (作为默认值)

    # core/Execute.scala

    # alu.io.func

    该信号传给 ALU ,决定 ALU 执行何种运算,实际上该信号的取值为 src/main/scala/riscv/core/ALU.scala 定义的 ChiselEnum 枚举类型,在 “ alu.io.func

    # alu.io.op1

    该信号传给 ALU,决定 ALU 的第一个操作数的数据源为 PC 还是寄存器。其控制已在 InstructionDecode 模块实现,其传入 Execute 模块的信号为 io.aluop1_source ,根据该信号即可确定 alu.io.op1

    # alu.io.op2

    该信号传给 ALU,决定 ALU 的第一个操作数的数据源为立即数还是寄存器。其控制已在 InstructionDecode 模块实现,其传入 Execute 模块的信号为 io.aluop2_source ,根据该信号即可确定 alu.io.op2

    # core/CPU.scala

    所填代码主要涉及传递给 Execute 模块的信号,各个赋值语句对应了连线图的控制信号连线、模块之间的端口的连接

    # ex.io.instruction := inst_fetch.io.instruction

    取指阶段得到的指令传递给 Execute 模块

    # ex.io.instruction_address := inst_fetch.io.instruction_address

    取指阶段得到的指令地址传递给 Execute 模块

    # ex.io.reg1_data := regs.io.read_data1

    寄存器组将读取的第 1 个数据传递给 Execute 模块

    # ex.io.reg2_data := regs.io.read_data2

    寄存器组将读取的第 2 个数据传递给 Execute 模块

    # ex.io.immediate := id.io.ex_immediate

    译码阶段得到的立即数传递给 Execute 模块

    # ex.io.aluop1_source := id.io.ex_aluop1_source

    译码阶段得到的 ALU 第 1 个操作数的数据源的控制信号传递给 Execute 模块

    # ex.io.aluop2_source := id.io.ex_aluop2_source

    译码阶段得到的 ALU 第 2 个操作数的数据源的控制信号传递给 Execute 模块

    # 在完成实验的过程中,遇到的关于实验指导不明确或者其他问题,或者改进的建议。

    使用实验板上的数码管、视频输出等外设,输出一个完整程序的运行结果,或者参考硬件调试一节的内容,用硬件波形的方法捕获程序运行结果。给出你的 CPU 可以正确在实验板上运行程序的照片或者硬件调试器波形截图。

    \ No newline at end of file diff --git a/cs/principles-of-computer-composition/yatcpu/lab-reports/lab2/index.html b/cs/principles-of-computer-composition/yatcpu/lab-reports/lab2/index.html index c1085844..c6d72505 100644 --- a/cs/principles-of-computer-composition/yatcpu/lab-reports/lab2/index.html +++ b/cs/principles-of-computer-composition/yatcpu/lab-reports/lab2/index.html @@ -1 +1 @@ -lab2 - YatCPU - 计算机组成原理 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    YatCPU - - - lab1 - - - lab2 - - - lab3 - - - lab4

    # 测试用例及其波形图分析

    # BoardSayGoodbyeTest

    src/test/scala/riscv/BoardTest.scala

    avatar

    其中 Top 模块来自:

    avatar

    测试执行程序来自 csrc/say_goodbye.c :

    avatar

    # 测试用例的功能

    测试能否正常执行打印 "Never gonna give you up~ Never gonna let you down~\nNever gonna run around and~ desert you~" ,用于烧板后测试能否实现打印,即测试 CPU 包括 lab1 的功能以及能否实现中断处理

    # 从什么层面测试 CPU

    从完整性的层面测试 CPU ,若 CPU 可以打印,说明 CPU 具备了中断处理的能力

    但并没有测试 CPU 的正确性,只是进行了 pokestep ,但没有对任何信号进行 expect 检查。实际验证需要烧板手动测试能否打印。

    # 加载测试程序的方法

    利用定义在 src/main/scala/board/z710/Top.scala 中的模块 Top ,用 RISC-V 程序 src/main/resources/say_goodbye.asmbin 初始化进行仿真

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    由 clock 信号变化可以看出时钟周期为 2 ps

    测试中循环执行了 200 次 step(1000) ,即 200000 个时钟周期,输出的波形图总时长为 400001 ps ,除去半个用于 reset 的时钟周期之外恰好 200000 个时钟周期。

    波形图中 CPU 内信号一直在不断变化,且按照一定序列无限循环周期变化,说明程序进入了死循环

    Timer 模块中的 count 不断自增,但到仿真结束为止页没达到 limit 信号的值 05F5E100H ,所以一直没有进入中断

    # ExecuteTest

    src/test/scala/riscv/singlecycle/ExecuteTest.scala

    avatar

    # 测试用例的功能

    该测试用例用于检查 Execute 模块的正确性,包括:

    # 执行 csrrci 指令

    测试指令 csrci ,即立即数读后清除控制状态寄存器 (csr) 指令,测试其是否将控制状态寄存器 csr 的值与按位取反后的 5 位的零扩展立即数 zimm 按位与的结果写入 csr

    # 执行 csrrsi 指令

    测试指令 csrrsi ,即立即数读后设置控制状态寄存器 (csr) 指令,测试其是否将控制状态寄存器 csr 的值与 5 位的零扩展立即数 zimm 按位或的结果写入 csr

    # 执行 csrrw 指令

    测试指令 csrrw ,即读后写控制状态寄存器 (csr) 指令,测试其是否将对应寄存器中的值写入 csr

    # 执行 csrrs 指令

    测试指令 csrrs ,即读后置位控制状态寄存器 (csr) 指令,测试其是否将控制状态寄存器 csr 的值与对应寄存器中的值按位或的结果写入 csr

    # 从什么层面测试 CPU

    从 CPU 执行处理层面测试了 CPU,主要测试这个阶段对中断相关的 csr 指令的执行处理

    # 加载测试程序的方法

    利用定义在 src/main/scala/riscv/core/Execute.scala 中的模块 Execute ,不使用任何 RISC-V 程序初始化进行仿真,直接仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    测试主要使用了下图中的指令:

    avatar

    # 指令

    测试执行了 1 个时钟周期

    # io_funct3[2:0]

    该信号为 111B ,与 io_opcode 一起指示 ALU 执行 CPU 执行 csr 类型指令 csrrci

    # io_opcode[6:0]

    该信号为 1110011B ,与 io_funct3 一起指示 CPU 执行 csr 类型指令 csrrci

    # zimm[31:0]

    零扩展的 5 位立即数,来自指令,值为 8

    # io_reg1_data[31:0]

    从寄存器读取的值,直接来自 c.io.reg1_data.poke(0x1880L.U) ,值为 1880H

    # io_csr_reg_read_data[31:0]

    从 csr 寄存器读取的值,直接来自 c.io.csr_reg_data.poke(0x1888L.U) ,值为 1888H

    # io_csr_reg_write_data[31:0]

    写入 csr 寄存器的值,在 csrrci 指令中该值等于 io_csr_reg_read_data & ~zimm ,结果为 1880H ,与波形图结果一致,说明执行正确

    # 指令

    测试执行了 1 个时钟周期

    # io_funct3[2:0]

    该信号为 110B ,与 io_opcode 一起指示 ALU 执行 CPU 执行 csr 类型指令 csrrsi

    # io_opcode[6:0]

    该信号为 1110011B ,与 io_funct3 一起指示 CPU 执行 csr 类型指令 csrrsi

    # zimm[31:0]

    零扩展的 5 位立即数,来自指令,值为 8

    # io_reg1_data[31:0]

    从寄存器读取的值,直接来自 c.io.reg1_data.poke(0x1880L.U) ,值为 1880H

    # io_csr_reg_read_data[31:0]

    从 csr 寄存器读取的值,直接来自 c.io.csr_reg_data.poke(0x1880L.U) ,值为 1880H

    # io_csr_reg_write_data[31:0]

    写入 csr 寄存器的值,在 csrrsi 指令中该值等于 io_csr_reg_read_data | zimm ,结果为 1880H ,与波形图结果一致,说明执行正确

    # 指令

    测试执行了 1 个时钟周期

    # io_funct3[2:0]

    该信号为 001B ,与 io_opcode 一起指示 ALU 执行 CPU 执行 csr 类型指令 csrrsw

    # io_opcode[6:0]

    该信号为 1110011B ,与 io_funct3 一起指示 CPU 执行 csr 类型指令 csrrsw

    # zimm[31:0]

    零扩展的 5 位立即数,来自指令,值为 AH

    # io_reg1_data[31:0]

    从寄存器读取的值,直接来自 c.io.reg1_data.poke(0x1888L.U) ,值为 1888H

    # io_csr_reg_read_data[31:0]

    从 csr 寄存器读取的值,直接来自 c.io.csr_reg_data.poke(0.U) ,值为 0

    # io_csr_reg_write_data[31:0]

    写入 csr 寄存器的值,在 csrrsi 指令中该值等于 io_reg1_data ,结果为 1888H ,与波形图结果一致,说明执行正确

    # 指令

    测试执行了 1 个时钟周期

    # io_funct3[2:0]

    该信号为 010B ,与 io_opcode 一起指示 ALU 执行 CPU 执行 csr 类型指令 csrrs

    # io_opcode[6:0]

    该信号为 1110011B ,与 io_funct3 一起指示 CPU 执行 csr 类型指令 csrrs

    # zimm[31:0]

    零扩展的 5 位立即数,来自指令,值为 0

    # io_reg1_data[31:0]

    从寄存器读取的值,直接来自 c.io.reg1_data.poke(0.U) ,值为 0

    # io_csr_reg_read_data[31:0]

    从 csr 寄存器读取的值,直接来自 c.io.csr_reg_data.poke(0x1888L.U) ,值为 1888H

    # io_csr_reg_write_data[31:0]

    写入 csr 寄存器的值,在 csrrs 指令中该值等于 io_csr_reg_read_data | io_reg1_data ,结果为 1888H ,与波形图结果一致,说明执行正确

    # CLINTCSRTest

    src/test/scala/riscv/singlecycle/CLINTCSRTest.scala

    avatar

    avatar

    # 测试用例的功能

    测试了 CLINT 模块和 CSR 模块的正确性,测试了以下几种情况:

    # 无跳转的情况下处理中断请求

    此时应该进入中断:

    • 将 PC 设为中断处理程序入口
    • 写入 CSR 寄存器
      • 将 MIE 置 0
      • 将 PC 的值存入 MEPC ,实际上为中断发生的指令的下一条指令的地址
      • 写入恰当的中断原因存入 MCAUSE

    # 跳转的情况下处理中断请求

    此时应该进入中断,处理与上一种情况相同,但是存入 MEPC 的值不是 PC 而是跳转目标地址

    # 退出中断

    • 将 PC 设为 MEPC
    • 写入 CSR 寄存器
      • 将 MIE 置为 MPIE

    # 不响应中断

    当 MPIE 位为 0 时,禁用中断

    # 从什么层面测试 CPU

    从中断处理层面测试,主要测试对不同情况 CPU 能否正确决定是否响应中断、响应中断时是否正确处理中断、能否正确退出中断

    # 加载测试程序指令的方法

    利用定义在 src/main/scala/riscv/core/CLINT.scala 中的模块 CLINT 和定义在 src/main/scala/riscv/core/CSR.scala 中的模块 CSR ,不使用任何 RISC-V 程序初始化进行仿真,直接仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    # 无跳转的情况下处理中断请求

    avatar

    # io_interrupt_assert

    该信号决定 PC 是否跳转,在输入的 interrupt_flag 信号拉高的同时也拉高,表示进入中断,跳转到中断处理程序

    # io_interrupt_handler_address[31:0]

    该信号在输入的 interrupt_flag 信号拉高的同时变为 00001144H ,表示中断处理程序入口地址,从下一条指令开始进入中断

    # mepc[31:0]

    该信号在进入中断后的下一时钟周期变为 1904H ,等于原 PC 值 1900H + 4 ,表示原本 PC 所在指令的下一条指令地址

    # mcause[31:0]

    该信号主要由引起中断的 io.interrupt_flag 信号决定,在进入中断后的下一时钟周期变为 80000007H ,表示 timer 引起中断

    # mstatus[31:0]

    该信号在进入中断后的下一时钟周期由 00001888H 变为 00001880H ,表示 MPIE 被置 0 ,禁用中断

    # 退出中断

    avatar

    # io_interrupt_assert

    该信号决定 PC 是否跳转,在输入的 io_instruction 变为 30200073H 的同时拉高,即遇到 mret 信号拉高,表示退出中断,跳转到中断前原程序

    # io_interrupt_handler_address[31:0]

    该信号在退出中断的同时保持 00001904H 不变,为 PC 跳转提供目标地址

    # mstatus[31:0]

    该信号在退出中断后的下一时钟周期由 00001880H 变为 00001888H ,表示 MPIE 被置 1 ,启用中断

    # 跳转的情况下处理中断请求

    avatar

    # io_interrupt_assert

    在输入的 interrupt_flag 信号拉高的同时也高,表示进入中断,跳转到中断处理程序

    由于上一个时钟周期控制器在退出中断,所以上一个时钟周期 PC 也发生跳转,该信号也为 1 ,所以波形图中表现为连续高电位

    # io_interrupt_handler_address[31:0]

    该信号在输入的 interrupt_flag 信号拉高的同时变为 00001144H ,表示中断处理程序入口地址,从下一条指令开始进入中断

    # mepc[31:0]

    该信号在进入中断后的下一时钟周期变为 1990H ,等于原跳转指令对应的跳转目标地址,表示原本 PC 所在指令所要跳转的指令地址

    # mcause[31:0]

    该信号主要由引起中断的 io.interrupt_flag 信号决定,在进入中断后的下一时钟周期变为 8000000BH ,表示外部引起中断

    # mstatus[31:0]

    该信号在进入中断后的下一时钟周期由 00001888H 变为 00001880H ,表示 MPIE 被置 0 ,禁用中断

    # 退出中断

    再次退出中断,与 情况一致

    # 不响应中断

    原本测试程序无法显示该处波形,需要加一条 c.clock.step() 才能显示出波形

    avatar

    # io_clint_access_bundle_mstatus[31:0]

    该信号由 c.io.csr_regs_write_data.poke(0x1880L.U) 输入,表示写入 mstatus 的值。其中低位第三位为 0 ,即 MPIE = 0 ,禁用中断

    # interrupt_enable

    在输入的 io_clint_access_bundle_mstatus 信号变为 1880H 的同时拉低,表示禁用中断

    # io_interrupt_flag[31:0]

    由输入 c.io.interrupt_flag.poke(1.U) 拉高该信号,表示中断请求

    # io_interrupt_assert

    io_interrupt_flag 拉高的同时,该信号为低电位,表示拒绝中断请求

    # TimerTest

    src/test/scala/riscv/singlecycle/TimerTest.scala

    avatar

    # 测试用例的功能

    测试了 Timer 模块能否正确处理信号

    # 写 limit

    检测 Timer 模块能否实现正确的写入到 limit 变量操作

    # 写 enabled

    检测 Timer 模块能否实现正确的写入到 enabled 变量操作,以及根据变量输出正确的中断信号

    # 从什么层面测试 CPU

    从外围设备层面测试 CPU 的正确性,准确来说测试 peripheral 设备 Timer 的正确性

    # 加载测试程序指令的方法

    利用定义在 src/main/scala/riscv/core/peripheral/Timer.scala 中的模块 Timer,不使用任何 RISC-V 程序初始化进行仿真,直接仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    # limit

    io_bundle_write_data 变为 00990315H 时,io_bundkle_address 值为 4,表示写入 limit 的地址,写入数据为 00990315 , limit 的值也在写入操作的下一个时钟周期由初始值变为 00990315H ,可见写操作正确执行

    # enabled

    io_bundle_write_data 变为 0 时,io_bundkle_address 值为 8,表示写入 enable 的地址,写入数据为 0 ,enabled 信号关闭不允许启用中断,enabled 的值也在写入操作的下一个时钟周期由 1 变为 0 ,可见写操作正确执行

    # SimpleTrapTest

    src/test/scala/riscv/singlecycle/SimpleTrapTest.scala

    avatar

    csrc/simpletest.c

    avatar

    # 测试用例的功能

    测试了 CPU 能否完整的运行一个中断处理程序,并退出中断

    # 从什么层面测试 CPU

    从 CPU 整体功能层面测试了 CPU ,测试了 CPU 处理自陷的功能

    # 加载测试程序指令的方法

    利用定义在 src/test/scala/riscv/singlecycle/CPUTest.scala 中的模块 TestTopModule ,模拟一个简单的计算机,连接了 CPU 、Memory、InstructionRom 、ROMLoader 几个模块。用 RISC-V 程序 src/main/resources/simpletest.asmbin 初始化进行仿真

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    测试总共执行了 2007 个时钟周期,在前 1000 多个时钟周期测试了 simpletest.asmbin 的程序,即将数据 DEADBEEFH 存到内存地址为 4 的内存空间。后 1000 多个时钟周期产生了一个中断请求信号,完整执行了一遍 simpletest.asmbin 自定义的中断处理程序。中断处理程序将数据 2022 重新写入内存地址为 4 的内存空间覆盖原来的数据,因此测试程序最后从该地址读取的数据为 2022 ,表明中断处理程序正确执行

    波形图表现为前后两段信号不断变化的区间,分别代表 simpletest.asmbin 的 main 函数程序和中断处理程序 trap_handler 的执行。由波形图可见中断处理程序结束后顺利退出了中断

    # 填空涉及到的信号分析

    (src/main/scala/riscv/)

    主要在填空代码处以注释的形式进行了分析

    # 在完成实验的过程中,遇到的关于实验指导不明确或者其他问题,或者改进的建议。

    使用实验板上的数码管、视频输出等外设,输出一个完整程序的运行结果,或者参考硬件调试一节的内容,用硬件波形的方法捕获程序运行结果。给出你的 CPU 可以正确在实验板上运行程序的照片或者硬件调试器波形截图。

    \ No newline at end of file +lab2 - YatCPU - 计算机组成原理 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    YatCPU - - - lab1 - - - lab2 - - - lab3 - - - lab4

    # 测试用例及其波形图分析

    # BoardSayGoodbyeTest

    src/test/scala/riscv/BoardTest.scala

    avatar

    其中 Top 模块来自:

    avatar

    测试执行程序来自 csrc/say_goodbye.c :

    avatar

    # 测试用例的功能

    测试能否正常执行打印 "Never gonna give you up~ Never gonna let you down~\nNever gonna run around and~ desert you~" ,用于烧板后测试能否实现打印,即测试 CPU 包括 lab1 的功能以及能否实现中断处理

    # 从什么层面测试 CPU

    从完整性的层面测试 CPU ,若 CPU 可以打印,说明 CPU 具备了中断处理的能力

    但并没有测试 CPU 的正确性,只是进行了 pokestep ,但没有对任何信号进行 expect 检查。实际验证需要烧板手动测试能否打印。

    # 加载测试程序的方法

    利用定义在 src/main/scala/board/z710/Top.scala 中的模块 Top ,用 RISC-V 程序 src/main/resources/say_goodbye.asmbin 初始化进行仿真

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    由 clock 信号变化可以看出时钟周期为 2 ps

    测试中循环执行了 200 次 step(1000) ,即 200000 个时钟周期,输出的波形图总时长为 400001 ps ,除去半个用于 reset 的时钟周期之外恰好 200000 个时钟周期。

    波形图中 CPU 内信号一直在不断变化,且按照一定序列无限循环周期变化,说明程序进入了死循环

    Timer 模块中的 count 不断自增,但到仿真结束为止页没达到 limit 信号的值 05F5E100H ,所以一直没有进入中断

    # ExecuteTest

    src/test/scala/riscv/singlecycle/ExecuteTest.scala

    avatar

    # 测试用例的功能

    该测试用例用于检查 Execute 模块的正确性,包括:

    # 执行 csrrci 指令

    测试指令 csrci ,即立即数读后清除控制状态寄存器 (csr) 指令,测试其是否将控制状态寄存器 csr 的值与按位取反后的 5 位的零扩展立即数 zimm 按位与的结果写入 csr

    # 执行 csrrsi 指令

    测试指令 csrrsi ,即立即数读后设置控制状态寄存器 (csr) 指令,测试其是否将控制状态寄存器 csr 的值与 5 位的零扩展立即数 zimm 按位或的结果写入 csr

    # 执行 csrrw 指令

    测试指令 csrrw ,即读后写控制状态寄存器 (csr) 指令,测试其是否将对应寄存器中的值写入 csr

    # 执行 csrrs 指令

    测试指令 csrrs ,即读后置位控制状态寄存器 (csr) 指令,测试其是否将控制状态寄存器 csr 的值与对应寄存器中的值按位或的结果写入 csr

    # 从什么层面测试 CPU

    从 CPU 执行处理层面测试了 CPU,主要测试这个阶段对中断相关的 csr 指令的执行处理

    # 加载测试程序的方法

    利用定义在 src/main/scala/riscv/core/Execute.scala 中的模块 Execute ,不使用任何 RISC-V 程序初始化进行仿真,直接仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    测试主要使用了下图中的指令:

    avatar

    # 指令

    测试执行了 1 个时钟周期

    # io_funct3[2:0]

    该信号为 111B ,与 io_opcode 一起指示 ALU 执行 CPU 执行 csr 类型指令 csrrci

    # io_opcode[6:0]

    该信号为 1110011B ,与 io_funct3 一起指示 CPU 执行 csr 类型指令 csrrci

    # zimm[31:0]

    零扩展的 5 位立即数,来自指令,值为 8

    # io_reg1_data[31:0]

    从寄存器读取的值,直接来自 c.io.reg1_data.poke(0x1880L.U) ,值为 1880H

    # io_csr_reg_read_data[31:0]

    从 csr 寄存器读取的值,直接来自 c.io.csr_reg_data.poke(0x1888L.U) ,值为 1888H

    # io_csr_reg_write_data[31:0]

    写入 csr 寄存器的值,在 csrrci 指令中该值等于 io_csr_reg_read_data & ~zimm ,结果为 1880H ,与波形图结果一致,说明执行正确

    # 指令

    测试执行了 1 个时钟周期

    # io_funct3[2:0]

    该信号为 110B ,与 io_opcode 一起指示 ALU 执行 CPU 执行 csr 类型指令 csrrsi

    # io_opcode[6:0]

    该信号为 1110011B ,与 io_funct3 一起指示 CPU 执行 csr 类型指令 csrrsi

    # zimm[31:0]

    零扩展的 5 位立即数,来自指令,值为 8

    # io_reg1_data[31:0]

    从寄存器读取的值,直接来自 c.io.reg1_data.poke(0x1880L.U) ,值为 1880H

    # io_csr_reg_read_data[31:0]

    从 csr 寄存器读取的值,直接来自 c.io.csr_reg_data.poke(0x1880L.U) ,值为 1880H

    # io_csr_reg_write_data[31:0]

    写入 csr 寄存器的值,在 csrrsi 指令中该值等于 io_csr_reg_read_data | zimm ,结果为 1880H ,与波形图结果一致,说明执行正确

    # 指令

    测试执行了 1 个时钟周期

    # io_funct3[2:0]

    该信号为 001B ,与 io_opcode 一起指示 ALU 执行 CPU 执行 csr 类型指令 csrrsw

    # io_opcode[6:0]

    该信号为 1110011B ,与 io_funct3 一起指示 CPU 执行 csr 类型指令 csrrsw

    # zimm[31:0]

    零扩展的 5 位立即数,来自指令,值为 AH

    # io_reg1_data[31:0]

    从寄存器读取的值,直接来自 c.io.reg1_data.poke(0x1888L.U) ,值为 1888H

    # io_csr_reg_read_data[31:0]

    从 csr 寄存器读取的值,直接来自 c.io.csr_reg_data.poke(0.U) ,值为 0

    # io_csr_reg_write_data[31:0]

    写入 csr 寄存器的值,在 csrrsi 指令中该值等于 io_reg1_data ,结果为 1888H ,与波形图结果一致,说明执行正确

    # 指令

    测试执行了 1 个时钟周期

    # io_funct3[2:0]

    该信号为 010B ,与 io_opcode 一起指示 ALU 执行 CPU 执行 csr 类型指令 csrrs

    # io_opcode[6:0]

    该信号为 1110011B ,与 io_funct3 一起指示 CPU 执行 csr 类型指令 csrrs

    # zimm[31:0]

    零扩展的 5 位立即数,来自指令,值为 0

    # io_reg1_data[31:0]

    从寄存器读取的值,直接来自 c.io.reg1_data.poke(0.U) ,值为 0

    # io_csr_reg_read_data[31:0]

    从 csr 寄存器读取的值,直接来自 c.io.csr_reg_data.poke(0x1888L.U) ,值为 1888H

    # io_csr_reg_write_data[31:0]

    写入 csr 寄存器的值,在 csrrs 指令中该值等于 io_csr_reg_read_data | io_reg1_data ,结果为 1888H ,与波形图结果一致,说明执行正确

    # CLINTCSRTest

    src/test/scala/riscv/singlecycle/CLINTCSRTest.scala

    avatar

    avatar

    # 测试用例的功能

    测试了 CLINT 模块和 CSR 模块的正确性,测试了以下几种情况:

    # 无跳转的情况下处理中断请求

    此时应该进入中断:

    • 将 PC 设为中断处理程序入口
    • 写入 CSR 寄存器
      • 将 MIE 置 0
      • 将 PC 的值存入 MEPC ,实际上为中断发生的指令的下一条指令的地址
      • 写入恰当的中断原因存入 MCAUSE

    # 跳转的情况下处理中断请求

    此时应该进入中断,处理与上一种情况相同,但是存入 MEPC 的值不是 PC 而是跳转目标地址

    # 退出中断

    • 将 PC 设为 MEPC
    • 写入 CSR 寄存器
      • 将 MIE 置为 MPIE

    # 不响应中断

    当 MPIE 位为 0 时,禁用中断

    # 从什么层面测试 CPU

    从中断处理层面测试,主要测试对不同情况 CPU 能否正确决定是否响应中断、响应中断时是否正确处理中断、能否正确退出中断

    # 加载测试程序指令的方法

    利用定义在 src/main/scala/riscv/core/CLINT.scala 中的模块 CLINT 和定义在 src/main/scala/riscv/core/CSR.scala 中的模块 CSR ,不使用任何 RISC-V 程序初始化进行仿真,直接仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    # 无跳转的情况下处理中断请求

    avatar

    # io_interrupt_assert

    该信号决定 PC 是否跳转,在输入的 interrupt_flag 信号拉高的同时也拉高,表示进入中断,跳转到中断处理程序

    # io_interrupt_handler_address[31:0]

    该信号在输入的 interrupt_flag 信号拉高的同时变为 00001144H ,表示中断处理程序入口地址,从下一条指令开始进入中断

    # mepc[31:0]

    该信号在进入中断后的下一时钟周期变为 1904H ,等于原 PC 值 1900H + 4 ,表示原本 PC 所在指令的下一条指令地址

    # mcause[31:0]

    该信号主要由引起中断的 io.interrupt_flag 信号决定,在进入中断后的下一时钟周期变为 80000007H ,表示 timer 引起中断

    # mstatus[31:0]

    该信号在进入中断后的下一时钟周期由 00001888H 变为 00001880H ,表示 MPIE 被置 0 ,禁用中断

    # 退出中断

    avatar

    # io_interrupt_assert

    该信号决定 PC 是否跳转,在输入的 io_instruction 变为 30200073H 的同时拉高,即遇到 mret 信号拉高,表示退出中断,跳转到中断前原程序

    # io_interrupt_handler_address[31:0]

    该信号在退出中断的同时保持 00001904H 不变,为 PC 跳转提供目标地址

    # mstatus[31:0]

    该信号在退出中断后的下一时钟周期由 00001880H 变为 00001888H ,表示 MPIE 被置 1 ,启用中断

    # 跳转的情况下处理中断请求

    avatar

    # io_interrupt_assert

    在输入的 interrupt_flag 信号拉高的同时也高,表示进入中断,跳转到中断处理程序

    由于上一个时钟周期控制器在退出中断,所以上一个时钟周期 PC 也发生跳转,该信号也为 1 ,所以波形图中表现为连续高电位

    # io_interrupt_handler_address[31:0]

    该信号在输入的 interrupt_flag 信号拉高的同时变为 00001144H ,表示中断处理程序入口地址,从下一条指令开始进入中断

    # mepc[31:0]

    该信号在进入中断后的下一时钟周期变为 1990H ,等于原跳转指令对应的跳转目标地址,表示原本 PC 所在指令所要跳转的指令地址

    # mcause[31:0]

    该信号主要由引起中断的 io.interrupt_flag 信号决定,在进入中断后的下一时钟周期变为 8000000BH ,表示外部引起中断

    # mstatus[31:0]

    该信号在进入中断后的下一时钟周期由 00001888H 变为 00001880H ,表示 MPIE 被置 0 ,禁用中断

    # 退出中断

    再次退出中断,与 情况一致

    # 不响应中断

    原本测试程序无法显示该处波形,需要加一条 c.clock.step() 才能显示出波形

    avatar

    # io_clint_access_bundle_mstatus[31:0]

    该信号由 c.io.csr_regs_write_data.poke(0x1880L.U) 输入,表示写入 mstatus 的值。其中低位第三位为 0 ,即 MPIE = 0 ,禁用中断

    # interrupt_enable

    在输入的 io_clint_access_bundle_mstatus 信号变为 1880H 的同时拉低,表示禁用中断

    # io_interrupt_flag[31:0]

    由输入 c.io.interrupt_flag.poke(1.U) 拉高该信号,表示中断请求

    # io_interrupt_assert

    io_interrupt_flag 拉高的同时,该信号为低电位,表示拒绝中断请求

    # TimerTest

    src/test/scala/riscv/singlecycle/TimerTest.scala

    avatar

    # 测试用例的功能

    测试了 Timer 模块能否正确处理信号

    # 写 limit

    检测 Timer 模块能否实现正确的写入到 limit 变量操作

    # 写 enabled

    检测 Timer 模块能否实现正确的写入到 enabled 变量操作,以及根据变量输出正确的中断信号

    # 从什么层面测试 CPU

    从外围设备层面测试 CPU 的正确性,准确来说测试 peripheral 设备 Timer 的正确性

    # 加载测试程序指令的方法

    利用定义在 src/main/scala/riscv/core/peripheral/Timer.scala 中的模块 Timer,不使用任何 RISC-V 程序初始化进行仿真,直接仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    # limit

    io_bundle_write_data 变为 00990315H 时,io_bundkle_address 值为 4,表示写入 limit 的地址,写入数据为 00990315 , limit 的值也在写入操作的下一个时钟周期由初始值变为 00990315H ,可见写操作正确执行

    # enabled

    io_bundle_write_data 变为 0 时,io_bundkle_address 值为 8,表示写入 enable 的地址,写入数据为 0 ,enabled 信号关闭不允许启用中断,enabled 的值也在写入操作的下一个时钟周期由 1 变为 0 ,可见写操作正确执行

    # SimpleTrapTest

    src/test/scala/riscv/singlecycle/SimpleTrapTest.scala

    avatar

    csrc/simpletest.c

    avatar

    # 测试用例的功能

    测试了 CPU 能否完整的运行一个中断处理程序,并退出中断

    # 从什么层面测试 CPU

    从 CPU 整体功能层面测试了 CPU ,测试了 CPU 处理自陷的功能

    # 加载测试程序指令的方法

    利用定义在 src/test/scala/riscv/singlecycle/CPUTest.scala 中的模块 TestTopModule ,模拟一个简单的计算机,连接了 CPU 、Memory、InstructionRom 、ROMLoader 几个模块。用 RISC-V 程序 src/main/resources/simpletest.asmbin 初始化进行仿真

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    测试总共执行了 2007 个时钟周期,在前 1000 多个时钟周期测试了 simpletest.asmbin 的程序,即将数据 DEADBEEFH 存到内存地址为 4 的内存空间。后 1000 多个时钟周期产生了一个中断请求信号,完整执行了一遍 simpletest.asmbin 自定义的中断处理程序。中断处理程序将数据 2022 重新写入内存地址为 4 的内存空间覆盖原来的数据,因此测试程序最后从该地址读取的数据为 2022 ,表明中断处理程序正确执行

    波形图表现为前后两段信号不断变化的区间,分别代表 simpletest.asmbin 的 main 函数程序和中断处理程序 trap_handler 的执行。由波形图可见中断处理程序结束后顺利退出了中断

    # 填空涉及到的信号分析

    (src/main/scala/riscv/)

    主要在填空代码处以注释的形式进行了分析

    # 在完成实验的过程中,遇到的关于实验指导不明确或者其他问题,或者改进的建议。

    使用实验板上的数码管、视频输出等外设,输出一个完整程序的运行结果,或者参考硬件调试一节的内容,用硬件波形的方法捕获程序运行结果。给出你的 CPU 可以正确在实验板上运行程序的照片或者硬件调试器波形截图。

    \ No newline at end of file diff --git a/cs/principles-of-computer-composition/yatcpu/lab-reports/lab3/index.html b/cs/principles-of-computer-composition/yatcpu/lab-reports/lab3/index.html index 122f922a..744de3c9 100644 --- a/cs/principles-of-computer-composition/yatcpu/lab-reports/lab3/index.html +++ b/cs/principles-of-computer-composition/yatcpu/lab-reports/lab3/index.html @@ -1 +1 @@ -lab3 - YatCPU - 计算机组成原理 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    YatCPU - - - lab1 - - - lab2 - - - lab3 - - - lab4

    # 测试用例及其波形图分析

    # SayGoodbyeTest

    src/test/scala/riscv/BoardTest.scala

    avatar

    其中 Top 模块来自:

    avatar

    测试执行程序来自 csrc/say_goodbye.c :

    avatar

    # 测试用例的功能

    与 lab2 中 BoardSayGoodbyeTest 功能相似

    测试能否正常执行打印 "Never gonna give you up~ Never gonna let you down~\nNever gonna run around and~ desert you~" ,用于烧板后测试能否实现打印,即测试 CPU 包括 lab1 的功能以及能否实现中断处理

    # 从什么层面测试 CPU

    从完整性的层面测试 CPU ,若 CPU 可以打印,说明 CPU 具备了中断处理的能力

    但并没有测试 CPU 的正确性,只是进行了 pokestep ,但没有对任何信号进行 expect 检查。实际验证需要烧板手动测试能否打印。

    # 加载测试程序的方法

    利用定义在 src/main/scala/board/z710/Top.scala 中的模块 Top ,用 RISC-V 程序 src/main/resources/say_goodbye.asmbin 初始化进行仿真

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    测试中循环执行了 50000 次 step(5) ,即 250000 个时钟周期,输出的波形图总时长为 500001 ps ,除去半个用于 reset 的时钟周期之外恰好 250000 个时钟周期。

    由 io_jump_flag 、io_id_flush 、io_if_flush 、io_if_stall 、io_pc_stall 几个信号变化可以看出,CPU 在执行程序过程中较频繁地出现了控制冒险和数据冒险,需要通过阻塞、清除、转发等操作处理

    # PipelineRegisterTest

    src/test/scala/riscv/PipelineRegisterTest.scala

    avatar

    # 测试用例的功能

    该测试用例用于检查 PipelineRegister 模块的正确性,测试程序模拟了 3 种情况:

    # 既不清除也不阻塞

    在当前指令既不清除也不阻塞的情况,PipelineRegister 模块应该更新内容

    # 阻塞

    在当前指令需要阻塞的情况,PipelineRegister 模块应该保持原本内容不变,实现 “暂停”

    # 清除

    在当前指令需要清除的情况,PipelineRegister 模块的内容应该恢复默认值,默认值代表清除时塞入的 “气泡”,例如:

    • instruction 的默认初始值为 nop
    • instruction_address 的默认初始值为程序入口指令地址

    # 从什么层面测试 CPU

    从辅助寄存器 PipelineRegister 层面测试了 CPU,主要测试流水线 CPU 不同级之间的寄存器能否实现流水线 CPU 所需要的基本功能

    # 加载测试程序的方法

    利用定义在 src/main/scala/riscv/core/PipelineRegister.scala 中的模块 PipelineRegister ,不使用任何 RISC-V 程序初始化进行仿真,直接仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    测试程序模拟了 3 中不同情况,并且重复对三种情况中随机一个进行测试,总共测试了 1000 次,执行了 1000 个时钟周期

    # 既不清除也不阻塞

    avatar

    io_flush = 0 且 io_stall = 0 ,表示既不清除也不阻塞的情况,PipelineRegister 模块应该更新内容,所以存储变量 temp 和输出端口的值 io_out 均变为输入寄存器的新的值 io_in = 59111E2DH

    # 阻塞

    avatar

    io_flush = 0 且 io_stall = 1 ,表示阻塞的情况,PipelineRegister 模块应该维持原本内容不变,所以存储变量 temp 和输出端口的值 io_out 均不受输入寄存器的新的值 io_in = 4846D70DH 的影响,维持 7B035A8H 不变

    # 清除

    avatar

    io_flush = 1 且 io_stall = 0 ,表示清除的情况,PipelineRegister 模块应该重制内容为默认初始值,所以存储变量 temp 和输出端口的值 io_out 均不受输入寄存器的新的值 io_in = 4C52A358H 的影响,变为测试程序随机生成的默认初始值 7B035A8H

    # hazard.S

    后面几个测试都使用了 src/main/resources/hazard.asmbin 进行仿真,其汇编源文件为 csrc/hazard.S ,该文件内容源代码如下

    avatar

    对这段汇编代码出现的冒险分析如下:

    RISCV汇编
    .globl _start
    _start:
      csrr a0, cycle
      addi t0, zero, 1    # 数据冒险:下一条指令 sub 需要使用 ALU 计算结果 t0 的值
      sub t1, zero, t0    # 数据冒险:需要使用上一条指令 addi 的 ALU 计算结果 t0 的值
                          #          下一条指令 and 需要使用 ALU 计算结果 t1 的值
      and t2, t0, t1      # 数据冒险:需要使用上一条指令 sub 的 ALU 计算结果 t1 的值
                          #          下一条指令 sw 需要使用 ALU 计算结果 t2 的值
      sw t2, 4(zero)      # 数据冒险:需要使用上一条指令 and 的 ALU 计算结果 t2 的值
      j skip1             # 控制冒险:无条件跳转指令,后续指令会被抛弃
      or t2, t0, t1       # 控制冒险:将会无条件跳转到 skip1 标签处,此指令将被抛弃
      xor t2, t0, t1      # 控制冒险:将会无条件跳转到 skip1 标签处,此指令将被抛弃
    skip1:
      addi t1, t2, 1      # 数据冒险:下一条指令 add 需要使用 ALU 计算结果 t1 的值
      add t2, t1, t2      # 数据冒险:需要使用上一条指令 addi 的 ALU 计算结果 t1 的值
                          #          下一条指令 and 需要使用 ALU 计算结果 t2 的值
      and t2, t1, t2      # 数据冒险:需要使用上一条指令 and 的 ALU 计算结果 t2 的值
                          #          下一条指令 lw 需要使用 ALU 计算结果 t2 的值
      lw t2, 2(t2)        # 数据冒险:需要使用上一条指令 and 的 ALU 计算结果 t2 的值
      or t3, t1, t2       # 数据冒险:需要使用上一条指令 lw 读取内存得到的 t2 的值
      blt t2, t3, skip2   # 数据冒险:需要使用上一条指令 or 的 ALU 计算结果 t3 的值
                          # 控制冒险:条件分支,后续指令可能会被抛弃
      or t3, t0, t0       # 控制冒险:如果分支条件满足,将会跳转到 skip2 标签处,此指令将被抛弃
      xor t3, t0, t1      # 控制冒险:如果分支条件满足,将会跳转到 skip2 标签处,此指令将被抛弃
    skip2:
      addi t4, zero, 3    # 数据冒险:下一条指令 bne 需要使用 ALU 计算结果 t4 的值
      bne t3, t4, skip1   # 数据冒险:需要使用上一条指令 addi 的 ALU 计算结果 t4 的值
                          # 控制冒险:条件分支,后续指令可能会被抛弃
      sw t3, 8(zero)
      auipc t4, 0         # 控制冒险:如果分支条件满足,将会跳转到 skip1 标签处,此指令将被抛弃
                          # 数据冒险:下一条指令 jalr 需要使用扩展立即数结果 t4 的值
      jalr t4, 8(t4)      # 控制冒险:如果分支条件满足,将会跳转到 skip1 标签处,此指令将被抛弃
                          # 数据冒险:需要使用上一条指令 auipc 扩展立即数写入寄存器 t4 的值
                          # 控制冒险:无条件跳转指令,后续指令会被抛弃
      jalr t4, 4(t4)      # 控制冒险:无条件跳转指令,后续指令会被抛弃
      csrr a1, cycle      # 控制冒险:将会无条件跳转到未知地址,此指令可能被抛弃
                          # 数据冒险:下一条指令 sub 需要使用读取计数器得到的 a1 的值
      sub ra, a1, a0      # 控制冒险:将会无条件跳转到未知地址,此指令可能被抛弃
                          # 数据冒险:需要使用上一条指令 csrr 读取计数器得到的 a1 的值
    loop:
      j loop

    上述分析基于五级流水线 CPU 进行分析,若为三级流水线 CPU,则不存在数据冒险,只存在控制冒险

    若所有冒险都能正确处理,则程序应该按照如下过程执行:

    RISCV汇编
    .globl _start
    _start:
      csrr a0, cycle          # 读取计时器的当前值,存入寄存器 a0
      addi t0, zero, 1        # 将 0 加上立即数 1,结果存入寄存器 t0,t0=1
      sub t1, zero, t0        # 将 0 减去寄存器 t0 的值,结果存入寄存器 t1,t1=-1
      and t2, t0, t1          # 将寄存器 t0 和寄存器 t1 进行按位与操作,结果存入寄存器 t2,t2=1
      sw t2, 4(zero)          # 将寄存器 t2 的值存入内存地址 4
      j skip1                # 无条件跳转到标签 skip1
      or t2, t0, t1          # 控制冒险:此指令将被跳过,不会执行
      xor t2, t0, t1         # 控制冒险:此指令将被跳过,不会执行
    skip1:
      addi t1, t2, 1          # 将寄存器 t2 的值加上立即数 1,结果存入寄存器 t1,t1=2
      add t2, t1, t2          # 将寄存器 t1 和寄存器 t2 相加,结果存入寄存器 t2,t2=3
      and t2, t1, t2          # 将寄存器 t1 和寄存器 t2 进行按位与操作,结果存入寄存器 t2,t2=2
      lw t2, 2(t2)            # 从内存地址 (t2+2) 处读取值,存入寄存器 t2,t2=1
      or t3, t1, t2           # 将寄存器 t1 和寄存器 t2 进行按位或操作,结果存入寄存器 t3,t3=3
      blt t2, t3, skip2       # t2 < t3,跳转到标签 skip2
      or t3, t0, t0           # 控制冒险:此指令将被跳过,不会执行
      xor t3, t0, t1          # 控制冒险:此指令将被跳过,不会执行
    skip2:
      addi t4, zero, 3        # 将 0 加上立即数 3,结果存入寄存器 t4,t4=3
      bne t3, t4, skip1       # t3 和 t4 相等,不跳转,继续执行下面的指令
      sw t3, 8(zero)          # 将寄存器 t3 的值存入内存地址 8
      auipc t4, 0             # 将当前 PC 的高 20 位加上立即数 0 存入寄存器 t4,t4 = 当前指令 PC+4
      jalr t4, 8(t4)          # 跳转到地址 (t4+8),并将返回地址存入寄存器 t4,t4 = 上一条指令 PC+4+8 = 当前指令 PC+8
      jalr t4, 4(t4)          # 控制冒险:此指令将被跳过,不会执行,跳转到下一条指令
      csrr a1, cycle          # 读取计时器的当前值,存入寄存器 a1
      sub ra, a1, a0          # 将寄存器 a1 减去寄存器 a0 的值,结果存入寄存器 ra
    loop:
      j loop                  # 无条件跳转到标签 loop,形成一个无限循环

    程序结束时,寄存器的值如下:

    • a0 :存储了起始时刻的计时器值
    • a1 :存储了结束时刻的计时器值
    • ra :存储了计时器结束时刻减去起始时刻的差值,即进入死循环前程序运行了多少时钟周期
    • t0、t1、t2、t3、t4 寄存器的值如下:
      • t0 : 1
      • t1 : -1
      • t2 :1
      • t3 :3
      • t4 :指令 csrr a1, cycle 的地址

    内存地址中存储的数据如下:

    • 内存地址 4:存储了的值为 1
    • 内存地址 8:存储了的值为 3

    后面几个流水线 CPU 测试使用到的斐波那契数列计算、快排、读写单个字节这 3 个测试均已在 lab1 分析过,与单周期 CPU 无异,所以只分析他们对冒险的处理,即只分析他们对上述程序的执行处理过程

    # ThreeStageCPUTest

    src/test/scala/riscv/ThreeStageCPUTest.scala

    avatar

    # 测试用例的功能

    测试了三级流水线 CPU 的完整功能正确性,检测其是否解决 hazard.asmbin 的所有控制冒险问题

    # 从什么层面测试 CPU

    从对冒险的处理层面测试三级流水线 CPU

    # 加载测试程序指令的方法

    利用定义在 src/main/scala/riscv/TestTopModule.scala 中的模块 TestTopModule ,使用任何 RISC-V 程序 src/main/resources/hazard.asmbin 初始化进行仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    输出波形图中可以看到前面 120 ps 内测试程序执行完毕,此后程序进入死循环,不断重复执行 0000006FH 指令,即 jal 指令,原地循环

    # avatar

    如上图所示,波形图中自上到下三个信号分别为 IF、ID、EX 阶段的指令信号,当信号连续 2 个周期不发生改变,说明该阶段指令阻塞;当信号变为默认初始值 00000013H ,说明该阶段指令清空。

    avatar

    当确保程序执行完毕后,测试程序检测结果,读取寄存器和内存的值,看其是否符合预期。

    # io_regs_debug_read_data[31:0]

    c.io.regs_debug_read_address.poke(1.U) 使信号 io_regs_debug_read_address [4:0] 的值为 1 ,表示读取 1 号寄存器,即读取 ra 寄存器,读取 io_regs_debug_read_data 的值应当为三级流水线 CPU 执行 hazard.asmbin 两次 csrr 指令之间所花费的时钟周期

    两次 csrr 指令之间有 18 条指令(包括两条 csrr )在三级流水线执行过程都进入 EX 阶段过,即没有被跳过;且两次 csrr 指令之间经历了 4 次跳转指令控制冒险,从波形图可以看出 EX 阶段发生了 4 次阻塞,每次阻塞 2 个时钟周期。因此两次 csrr 指令之间所花费的时钟周期数为 18+4*2 = 26

    波形图可见最后读取到 io_regs_debug_read_data 的值等于 26 ,说明三级流水线 CPU 符合预期正确处理了所有控制冒险

    # io_mem_debug_read_data[31:0]

    先后两个时钟周期检测了两个地址所存储的内容

    c.io.mem_debug_read_address.poke(4.U) 读取内存地址为 4 的内容,结果为 1 ,符合预期;

    c.io.mem_debug_read_address.poke(8.U) 读取内存地址为 8 的内容,结果为 3 ,符合预期;

    # FiveStageCPUStallTest

    src/test/scala/riscv/FiveStageCPUStallTest.scala

    avatar

    # 测试用例的功能

    测试了只使用阻塞解决冒险的五级流水线 CPU 的完整功能正确性,检测其是否解决 hazard.asmbin 的所有控制冒险问题

    # 从什么层面测试 CPU

    从对冒险的处理层面测试五级流水线 CPU

    # 加载测试程序指令的方法

    利用定义在 src/main/scala/riscv/TestTopModule.scala 中的模块 TestTopModule ,使用任何 RISC-V 程序 src/main/resources/hazard.asmbin 初始化进行仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    输出波形图中可以看到前面 450 ps 内测试程序执行完毕,此后程序进入死循环,不断重复执行 0000006FH 指令,即 jal 指令,原地循环

    # avatar

    如上图所示,波形图中自上到下三个信号分别为 IF、ID、EX 阶段的指令信号,当信号连续 多 个周期不发生改变,说明该阶段指令阻塞;当信号变为默认初始值 00000013H ,说明该阶段指令清空。

    avatar

    当确保程序执行完毕后,测试程序检测结果,读取寄存器和内存的值,看其是否符合预期。

    # io_regs_debug_read_data[31:0]

    c.io.regs_debug_read_address.poke(1.U) 使信号 io_regs_debug_read_address [4:0] 的值为 1 ,表示读取 1 号寄存器,即读取 ra 寄存器,读取 io_regs_debug_read_data 的值应当为五级流水线 CPU 执行 hazard.asmbin 两次 csrr 指令之间所花费的时钟周期

    两次 csrr 指令之间有 18 条指令(包括第二条 csrr 但不包括第一条 csrr)在五级流水线执行过程都进入 EX 阶段过,即没有被跳过;且两次 csrr 指令之间经历了 4 次跳转指令控制冒险、10 次数据冒险,从波形图可以看出 EX 阶段发生了 14 次阻塞,每次阻塞 2 个时钟周期。因此两次 csrr 指令之间所花费的时钟周期数为 18+14*2 = 46

    波形图可见最后读取到 io_regs_debug_read_data 的值等于 46 ,说明五级流水线 CPU 符合预期正确处理了所有冒险

    # io_mem_debug_read_data[31:0]

    先后两个时钟周期检测了两个地址所存储的内容

    c.io.mem_debug_read_address.poke(4.U) 读取内存地址为 4 的内容,结果为 1 ,符合预期;

    c.io.mem_debug_read_address.poke(8.U) 读取内存地址为 8 的内容,结果为 3 ,符合预期;

    # FiveStageCPUForwardTest

    src/test/scala/riscv/FiveStageCPUForwardTest.scala

    avatar

    # 测试用例的功能

    测试了使用阻塞和转发解决冒险的五级流水线 CPU 的完整功能正确性,检测其是否解决 hazard.asmbin 的所有控制冒险问题

    # 从什么层面测试 CPU

    从对冒险的处理层面测试五级流水线 CPU

    # 加载测试程序指令的方法

    利用定义在 src/main/scala/riscv/TestTopModule.scala 中的模块 TestTopModule ,使用任何 RISC-V 程序 src/main/resources/hazard.asmbin 初始化进行仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    输出波形图中可以看到前面 300 ps 内测试程序执行完毕,此后程序进入死循环,不断重复执行 0000006FH 指令,即 jal 指令,原地循环

    # avatar

    如上图所示,波形图中自上到下三个信号分别为 IF、ID、EX 阶段的指令信号,当信号连续多个周期不发生改变,说明该阶段指令阻塞;当信号变为默认初始值 00000013H ,说明该阶段指令清空。

    avatar

    当确保程序执行完毕后,测试程序检测结果,读取寄存器和内存的值,看其是否符合预期。

    # io_regs_debug_read_data[31:0]

    c.io.regs_debug_read_address.poke(1.U) 使信号 io_regs_debug_read_address [4:0] 的值为 1 ,表示读取 1 号寄存器,即读取 ra 寄存器,读取 io_regs_debug_read_data 的值应当为五级流水线 CPU 执行 hazard.asmbin 两次 csrr 指令之间所花费的时钟周期

    两次 csrr 指令之间有 18 条指令(包括第二条 csrr 但不包括第一条 csrr)在五级流水线执行过程都进入 EX 阶段过,即没有被跳过;且两次 csrr 指令之间经历了 4 次跳转指令控制冒险、1 次无法用转发解决的数据冒险即 lw t2, 2(t2)or t3, t1, t2 之间的数据冒险,读取内存的数据必须等到下一个周期 MEM 阶段才能得到数据。从波形图可以看出 EX 阶段发生了 4 次阻塞,每次阻塞 2 个时钟周期,同时数据冒险还在 IF 和 ID 阶段阻塞了 1 个时钟周期。因此两次 csrr 指令之间所花费的时钟周期数为 18+4*2+1 = 27

    波形图可见最后读取到 io_regs_debug_read_data 的值等于 27 ,说明五级流水线 CPU 符合预期正确处理了所有冒险

    # io_mem_debug_read_data[31:0]

    先后两个时钟周期检测了两个地址所存储的内容

    c.io.mem_debug_read_address.poke(4.U) 读取内存地址为 4 的内容,结果为 1 ,符合预期;

    c.io.mem_debug_read_address.poke(8.U) 读取内存地址为 8 的内容,结果为 3 ,符合预期;

    # FiveStageCPUFinalTest

    src/test/scala/riscv/FiveStageCPUFinalTest.scala

    avatar

    # 测试用例的功能

    测试了使用阻塞、转发以及 ID 阶段判断分支来解决冒险的五级流水线 CPU 的完整功能正确性,检测其是否解决 hazard.asmbin 的所有控制冒险问题

    # 从什么层面测试 CPU

    从对冒险的处理层面测试五级流水线 CPU

    # 加载测试程序指令的方法

    利用定义在 src/main/scala/riscv/TestTopModule.scala 中的模块 TestTopModule ,使用任何 RISC-V 程序 src/main/resources/hazard.asmbin 初始化进行仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    输出波形图中可以看到前面 300 ps 内测试程序执行完毕,此后程序进入死循环,不断重复执行 0000006FH 指令,即 jal 指令,原地循环

    # avatar

    如上图所示,波形图中自上到下三个信号分别为 IF、ID、EX 阶段的指令信号,当信号连续多个周期不发生改变,说明该阶段指令阻塞;当信号变为默认初始值 00000013H ,说明该阶段指令清空。

    avatar

    当确保程序执行完毕后,测试程序检测结果,读取寄存器和内存的值,看其是否符合预期。

    # io_regs_debug_read_data[31:0]

    c.io.regs_debug_read_address.poke(1.U) 使信号 io_regs_debug_read_address [4:0] 的值为 1 ,表示读取 1 号寄存器,即读取 ra 寄存器,读取 io_regs_debug_read_data 的值应当为五级流水线 CPU 执行 hazard.asmbin 两次 csrr 指令之间所花费的时钟周期

    两次 csrr 指令之间有 18 条指令(包括第二条 csrr 但不包括第一条 csrr)在五级流水线执行过程都进入 EX 阶段过,即没有被跳过;且两次 csrr 指令之间经历了 4 次跳转指令控制冒险、1 次无法用转发解决的数据冒险即 lw t2, 2(t2)or t3, t1, t2 之间的数据冒险,读取内存的数据必须等到下一个周期 MEM 阶段才能得到数据。其中第一次控制冒险是由无条件跳转指令引起的,可以通过 ID 阶段只阻塞 1 个周期解决;其余 3 个控制冒险都需要阻塞 2 个时钟周期才能解决;同时数据冒险还在 IF 和 ID 阶段阻塞了 1 个时钟周期。因此两次 csrr 指令之间所花费的时钟周期数为 18+1+3*2+1 = 26

    波形图可见最后读取到 io_regs_debug_read_data 的值等于 26 ,说明五级流水线 CPU 符合预期正确处理了所有冒险

    # io_mem_debug_read_data[31:0]

    先后两个时钟周期检测了两个地址所存储的内容

    c.io.mem_debug_read_address.poke(4.U) 读取内存地址为 4 的内容,结果为 1 ,符合预期;

    c.io.mem_debug_read_address.poke(8.U) 读取内存地址为 8 的内容,结果为 3 ,符合预期;

    # 填空涉及到的信号分析

    (src/main/scala/riscv/)

    主要在填空代码处以注释的形式进行了分析

    # 在完成实验的过程中,遇到的关于实验指导不明确或者其他问题,或者改进的建议。

    使用实验板上的数码管、视频输出等外设,输出一个完整程序的运行结果,或者参考硬件调试一节的内容,用硬件波形的方法捕获程序运行结果。给出你的 CPU 可以正确在实验板上运行程序的照片或者硬件调试器波形截图。

    \ No newline at end of file +lab3 - YatCPU - 计算机组成原理 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    YatCPU - - - lab1 - - - lab2 - - - lab3 - - - lab4

    # 测试用例及其波形图分析

    # SayGoodbyeTest

    src/test/scala/riscv/BoardTest.scala

    avatar

    其中 Top 模块来自:

    avatar

    测试执行程序来自 csrc/say_goodbye.c :

    avatar

    # 测试用例的功能

    与 lab2 中 BoardSayGoodbyeTest 功能相似

    测试能否正常执行打印 "Never gonna give you up~ Never gonna let you down~\nNever gonna run around and~ desert you~" ,用于烧板后测试能否实现打印,即测试 CPU 包括 lab1 的功能以及能否实现中断处理

    # 从什么层面测试 CPU

    从完整性的层面测试 CPU ,若 CPU 可以打印,说明 CPU 具备了中断处理的能力

    但并没有测试 CPU 的正确性,只是进行了 pokestep ,但没有对任何信号进行 expect 检查。实际验证需要烧板手动测试能否打印。

    # 加载测试程序的方法

    利用定义在 src/main/scala/board/z710/Top.scala 中的模块 Top ,用 RISC-V 程序 src/main/resources/say_goodbye.asmbin 初始化进行仿真

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    测试中循环执行了 50000 次 step(5) ,即 250000 个时钟周期,输出的波形图总时长为 500001 ps ,除去半个用于 reset 的时钟周期之外恰好 250000 个时钟周期。

    由 io_jump_flag 、io_id_flush 、io_if_flush 、io_if_stall 、io_pc_stall 几个信号变化可以看出,CPU 在执行程序过程中较频繁地出现了控制冒险和数据冒险,需要通过阻塞、清除、转发等操作处理

    # PipelineRegisterTest

    src/test/scala/riscv/PipelineRegisterTest.scala

    avatar

    # 测试用例的功能

    该测试用例用于检查 PipelineRegister 模块的正确性,测试程序模拟了 3 种情况:

    # 既不清除也不阻塞

    在当前指令既不清除也不阻塞的情况,PipelineRegister 模块应该更新内容

    # 阻塞

    在当前指令需要阻塞的情况,PipelineRegister 模块应该保持原本内容不变,实现 “暂停”

    # 清除

    在当前指令需要清除的情况,PipelineRegister 模块的内容应该恢复默认值,默认值代表清除时塞入的 “气泡”,例如:

    • instruction 的默认初始值为 nop
    • instruction_address 的默认初始值为程序入口指令地址

    # 从什么层面测试 CPU

    从辅助寄存器 PipelineRegister 层面测试了 CPU,主要测试流水线 CPU 不同级之间的寄存器能否实现流水线 CPU 所需要的基本功能

    # 加载测试程序的方法

    利用定义在 src/main/scala/riscv/core/PipelineRegister.scala 中的模块 PipelineRegister ,不使用任何 RISC-V 程序初始化进行仿真,直接仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    测试程序模拟了 3 中不同情况,并且重复对三种情况中随机一个进行测试,总共测试了 1000 次,执行了 1000 个时钟周期

    # 既不清除也不阻塞

    avatar

    io_flush = 0 且 io_stall = 0 ,表示既不清除也不阻塞的情况,PipelineRegister 模块应该更新内容,所以存储变量 temp 和输出端口的值 io_out 均变为输入寄存器的新的值 io_in = 59111E2DH

    # 阻塞

    avatar

    io_flush = 0 且 io_stall = 1 ,表示阻塞的情况,PipelineRegister 模块应该维持原本内容不变,所以存储变量 temp 和输出端口的值 io_out 均不受输入寄存器的新的值 io_in = 4846D70DH 的影响,维持 7B035A8H 不变

    # 清除

    avatar

    io_flush = 1 且 io_stall = 0 ,表示清除的情况,PipelineRegister 模块应该重制内容为默认初始值,所以存储变量 temp 和输出端口的值 io_out 均不受输入寄存器的新的值 io_in = 4C52A358H 的影响,变为测试程序随机生成的默认初始值 7B035A8H

    # hazard.S

    后面几个测试都使用了 src/main/resources/hazard.asmbin 进行仿真,其汇编源文件为 csrc/hazard.S ,该文件内容源代码如下

    avatar

    对这段汇编代码出现的冒险分析如下:

    RISCV汇编
    .globl _start
    _start:
      csrr a0, cycle
      addi t0, zero, 1    # 数据冒险:下一条指令 sub 需要使用 ALU 计算结果 t0 的值
      sub t1, zero, t0    # 数据冒险:需要使用上一条指令 addi 的 ALU 计算结果 t0 的值
                          #          下一条指令 and 需要使用 ALU 计算结果 t1 的值
      and t2, t0, t1      # 数据冒险:需要使用上一条指令 sub 的 ALU 计算结果 t1 的值
                          #          下一条指令 sw 需要使用 ALU 计算结果 t2 的值
      sw t2, 4(zero)      # 数据冒险:需要使用上一条指令 and 的 ALU 计算结果 t2 的值
      j skip1             # 控制冒险:无条件跳转指令,后续指令会被抛弃
      or t2, t0, t1       # 控制冒险:将会无条件跳转到 skip1 标签处,此指令将被抛弃
      xor t2, t0, t1      # 控制冒险:将会无条件跳转到 skip1 标签处,此指令将被抛弃
    skip1:
      addi t1, t2, 1      # 数据冒险:下一条指令 add 需要使用 ALU 计算结果 t1 的值
      add t2, t1, t2      # 数据冒险:需要使用上一条指令 addi 的 ALU 计算结果 t1 的值
                          #          下一条指令 and 需要使用 ALU 计算结果 t2 的值
      and t2, t1, t2      # 数据冒险:需要使用上一条指令 and 的 ALU 计算结果 t2 的值
                          #          下一条指令 lw 需要使用 ALU 计算结果 t2 的值
      lw t2, 2(t2)        # 数据冒险:需要使用上一条指令 and 的 ALU 计算结果 t2 的值
      or t3, t1, t2       # 数据冒险:需要使用上一条指令 lw 读取内存得到的 t2 的值
      blt t2, t3, skip2   # 数据冒险:需要使用上一条指令 or 的 ALU 计算结果 t3 的值
                          # 控制冒险:条件分支,后续指令可能会被抛弃
      or t3, t0, t0       # 控制冒险:如果分支条件满足,将会跳转到 skip2 标签处,此指令将被抛弃
      xor t3, t0, t1      # 控制冒险:如果分支条件满足,将会跳转到 skip2 标签处,此指令将被抛弃
    skip2:
      addi t4, zero, 3    # 数据冒险:下一条指令 bne 需要使用 ALU 计算结果 t4 的值
      bne t3, t4, skip1   # 数据冒险:需要使用上一条指令 addi 的 ALU 计算结果 t4 的值
                          # 控制冒险:条件分支,后续指令可能会被抛弃
      sw t3, 8(zero)
      auipc t4, 0         # 控制冒险:如果分支条件满足,将会跳转到 skip1 标签处,此指令将被抛弃
                          # 数据冒险:下一条指令 jalr 需要使用扩展立即数结果 t4 的值
      jalr t4, 8(t4)      # 控制冒险:如果分支条件满足,将会跳转到 skip1 标签处,此指令将被抛弃
                          # 数据冒险:需要使用上一条指令 auipc 扩展立即数写入寄存器 t4 的值
                          # 控制冒险:无条件跳转指令,后续指令会被抛弃
      jalr t4, 4(t4)      # 控制冒险:无条件跳转指令,后续指令会被抛弃
      csrr a1, cycle      # 控制冒险:将会无条件跳转到未知地址,此指令可能被抛弃
                          # 数据冒险:下一条指令 sub 需要使用读取计数器得到的 a1 的值
      sub ra, a1, a0      # 控制冒险:将会无条件跳转到未知地址,此指令可能被抛弃
                          # 数据冒险:需要使用上一条指令 csrr 读取计数器得到的 a1 的值
    loop:
      j loop

    上述分析基于五级流水线 CPU 进行分析,若为三级流水线 CPU,则不存在数据冒险,只存在控制冒险

    若所有冒险都能正确处理,则程序应该按照如下过程执行:

    RISCV汇编
    .globl _start
    _start:
      csrr a0, cycle          # 读取计时器的当前值,存入寄存器 a0
      addi t0, zero, 1        # 将 0 加上立即数 1,结果存入寄存器 t0,t0=1
      sub t1, zero, t0        # 将 0 减去寄存器 t0 的值,结果存入寄存器 t1,t1=-1
      and t2, t0, t1          # 将寄存器 t0 和寄存器 t1 进行按位与操作,结果存入寄存器 t2,t2=1
      sw t2, 4(zero)          # 将寄存器 t2 的值存入内存地址 4
      j skip1                # 无条件跳转到标签 skip1
      or t2, t0, t1          # 控制冒险:此指令将被跳过,不会执行
      xor t2, t0, t1         # 控制冒险:此指令将被跳过,不会执行
    skip1:
      addi t1, t2, 1          # 将寄存器 t2 的值加上立即数 1,结果存入寄存器 t1,t1=2
      add t2, t1, t2          # 将寄存器 t1 和寄存器 t2 相加,结果存入寄存器 t2,t2=3
      and t2, t1, t2          # 将寄存器 t1 和寄存器 t2 进行按位与操作,结果存入寄存器 t2,t2=2
      lw t2, 2(t2)            # 从内存地址 (t2+2) 处读取值,存入寄存器 t2,t2=1
      or t3, t1, t2           # 将寄存器 t1 和寄存器 t2 进行按位或操作,结果存入寄存器 t3,t3=3
      blt t2, t3, skip2       # t2 < t3,跳转到标签 skip2
      or t3, t0, t0           # 控制冒险:此指令将被跳过,不会执行
      xor t3, t0, t1          # 控制冒险:此指令将被跳过,不会执行
    skip2:
      addi t4, zero, 3        # 将 0 加上立即数 3,结果存入寄存器 t4,t4=3
      bne t3, t4, skip1       # t3 和 t4 相等,不跳转,继续执行下面的指令
      sw t3, 8(zero)          # 将寄存器 t3 的值存入内存地址 8
      auipc t4, 0             # 将当前 PC 的高 20 位加上立即数 0 存入寄存器 t4,t4 = 当前指令 PC+4
      jalr t4, 8(t4)          # 跳转到地址 (t4+8),并将返回地址存入寄存器 t4,t4 = 上一条指令 PC+4+8 = 当前指令 PC+8
      jalr t4, 4(t4)          # 控制冒险:此指令将被跳过,不会执行,跳转到下一条指令
      csrr a1, cycle          # 读取计时器的当前值,存入寄存器 a1
      sub ra, a1, a0          # 将寄存器 a1 减去寄存器 a0 的值,结果存入寄存器 ra
    loop:
      j loop                  # 无条件跳转到标签 loop,形成一个无限循环

    程序结束时,寄存器的值如下:

    • a0 :存储了起始时刻的计时器值
    • a1 :存储了结束时刻的计时器值
    • ra :存储了计时器结束时刻减去起始时刻的差值,即进入死循环前程序运行了多少时钟周期
    • t0、t1、t2、t3、t4 寄存器的值如下:
      • t0 : 1
      • t1 : -1
      • t2 :1
      • t3 :3
      • t4 :指令 csrr a1, cycle 的地址

    内存地址中存储的数据如下:

    • 内存地址 4:存储了的值为 1
    • 内存地址 8:存储了的值为 3

    后面几个流水线 CPU 测试使用到的斐波那契数列计算、快排、读写单个字节这 3 个测试均已在 lab1 分析过,与单周期 CPU 无异,所以只分析他们对冒险的处理,即只分析他们对上述程序的执行处理过程

    # ThreeStageCPUTest

    src/test/scala/riscv/ThreeStageCPUTest.scala

    avatar

    # 测试用例的功能

    测试了三级流水线 CPU 的完整功能正确性,检测其是否解决 hazard.asmbin 的所有控制冒险问题

    # 从什么层面测试 CPU

    从对冒险的处理层面测试三级流水线 CPU

    # 加载测试程序指令的方法

    利用定义在 src/main/scala/riscv/TestTopModule.scala 中的模块 TestTopModule ,使用任何 RISC-V 程序 src/main/resources/hazard.asmbin 初始化进行仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    输出波形图中可以看到前面 120 ps 内测试程序执行完毕,此后程序进入死循环,不断重复执行 0000006FH 指令,即 jal 指令,原地循环

    # avatar

    如上图所示,波形图中自上到下三个信号分别为 IF、ID、EX 阶段的指令信号,当信号连续 2 个周期不发生改变,说明该阶段指令阻塞;当信号变为默认初始值 00000013H ,说明该阶段指令清空。

    avatar

    当确保程序执行完毕后,测试程序检测结果,读取寄存器和内存的值,看其是否符合预期。

    # io_regs_debug_read_data[31:0]

    c.io.regs_debug_read_address.poke(1.U) 使信号 io_regs_debug_read_address [4:0] 的值为 1 ,表示读取 1 号寄存器,即读取 ra 寄存器,读取 io_regs_debug_read_data 的值应当为三级流水线 CPU 执行 hazard.asmbin 两次 csrr 指令之间所花费的时钟周期

    两次 csrr 指令之间有 18 条指令(包括两条 csrr )在三级流水线执行过程都进入 EX 阶段过,即没有被跳过;且两次 csrr 指令之间经历了 4 次跳转指令控制冒险,从波形图可以看出 EX 阶段发生了 4 次阻塞,每次阻塞 2 个时钟周期。因此两次 csrr 指令之间所花费的时钟周期数为 18+4*2 = 26

    波形图可见最后读取到 io_regs_debug_read_data 的值等于 26 ,说明三级流水线 CPU 符合预期正确处理了所有控制冒险

    # io_mem_debug_read_data[31:0]

    先后两个时钟周期检测了两个地址所存储的内容

    c.io.mem_debug_read_address.poke(4.U) 读取内存地址为 4 的内容,结果为 1 ,符合预期;

    c.io.mem_debug_read_address.poke(8.U) 读取内存地址为 8 的内容,结果为 3 ,符合预期;

    # FiveStageCPUStallTest

    src/test/scala/riscv/FiveStageCPUStallTest.scala

    avatar

    # 测试用例的功能

    测试了只使用阻塞解决冒险的五级流水线 CPU 的完整功能正确性,检测其是否解决 hazard.asmbin 的所有控制冒险问题

    # 从什么层面测试 CPU

    从对冒险的处理层面测试五级流水线 CPU

    # 加载测试程序指令的方法

    利用定义在 src/main/scala/riscv/TestTopModule.scala 中的模块 TestTopModule ,使用任何 RISC-V 程序 src/main/resources/hazard.asmbin 初始化进行仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    输出波形图中可以看到前面 450 ps 内测试程序执行完毕,此后程序进入死循环,不断重复执行 0000006FH 指令,即 jal 指令,原地循环

    # avatar

    如上图所示,波形图中自上到下三个信号分别为 IF、ID、EX 阶段的指令信号,当信号连续 多 个周期不发生改变,说明该阶段指令阻塞;当信号变为默认初始值 00000013H ,说明该阶段指令清空。

    avatar

    当确保程序执行完毕后,测试程序检测结果,读取寄存器和内存的值,看其是否符合预期。

    # io_regs_debug_read_data[31:0]

    c.io.regs_debug_read_address.poke(1.U) 使信号 io_regs_debug_read_address [4:0] 的值为 1 ,表示读取 1 号寄存器,即读取 ra 寄存器,读取 io_regs_debug_read_data 的值应当为五级流水线 CPU 执行 hazard.asmbin 两次 csrr 指令之间所花费的时钟周期

    两次 csrr 指令之间有 18 条指令(包括第二条 csrr 但不包括第一条 csrr)在五级流水线执行过程都进入 EX 阶段过,即没有被跳过;且两次 csrr 指令之间经历了 4 次跳转指令控制冒险、10 次数据冒险,从波形图可以看出 EX 阶段发生了 14 次阻塞,每次阻塞 2 个时钟周期。因此两次 csrr 指令之间所花费的时钟周期数为 18+14*2 = 46

    波形图可见最后读取到 io_regs_debug_read_data 的值等于 46 ,说明五级流水线 CPU 符合预期正确处理了所有冒险

    # io_mem_debug_read_data[31:0]

    先后两个时钟周期检测了两个地址所存储的内容

    c.io.mem_debug_read_address.poke(4.U) 读取内存地址为 4 的内容,结果为 1 ,符合预期;

    c.io.mem_debug_read_address.poke(8.U) 读取内存地址为 8 的内容,结果为 3 ,符合预期;

    # FiveStageCPUForwardTest

    src/test/scala/riscv/FiveStageCPUForwardTest.scala

    avatar

    # 测试用例的功能

    测试了使用阻塞和转发解决冒险的五级流水线 CPU 的完整功能正确性,检测其是否解决 hazard.asmbin 的所有控制冒险问题

    # 从什么层面测试 CPU

    从对冒险的处理层面测试五级流水线 CPU

    # 加载测试程序指令的方法

    利用定义在 src/main/scala/riscv/TestTopModule.scala 中的模块 TestTopModule ,使用任何 RISC-V 程序 src/main/resources/hazard.asmbin 初始化进行仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    输出波形图中可以看到前面 300 ps 内测试程序执行完毕,此后程序进入死循环,不断重复执行 0000006FH 指令,即 jal 指令,原地循环

    # avatar

    如上图所示,波形图中自上到下三个信号分别为 IF、ID、EX 阶段的指令信号,当信号连续多个周期不发生改变,说明该阶段指令阻塞;当信号变为默认初始值 00000013H ,说明该阶段指令清空。

    avatar

    当确保程序执行完毕后,测试程序检测结果,读取寄存器和内存的值,看其是否符合预期。

    # io_regs_debug_read_data[31:0]

    c.io.regs_debug_read_address.poke(1.U) 使信号 io_regs_debug_read_address [4:0] 的值为 1 ,表示读取 1 号寄存器,即读取 ra 寄存器,读取 io_regs_debug_read_data 的值应当为五级流水线 CPU 执行 hazard.asmbin 两次 csrr 指令之间所花费的时钟周期

    两次 csrr 指令之间有 18 条指令(包括第二条 csrr 但不包括第一条 csrr)在五级流水线执行过程都进入 EX 阶段过,即没有被跳过;且两次 csrr 指令之间经历了 4 次跳转指令控制冒险、1 次无法用转发解决的数据冒险即 lw t2, 2(t2)or t3, t1, t2 之间的数据冒险,读取内存的数据必须等到下一个周期 MEM 阶段才能得到数据。从波形图可以看出 EX 阶段发生了 4 次阻塞,每次阻塞 2 个时钟周期,同时数据冒险还在 IF 和 ID 阶段阻塞了 1 个时钟周期。因此两次 csrr 指令之间所花费的时钟周期数为 18+4*2+1 = 27

    波形图可见最后读取到 io_regs_debug_read_data 的值等于 27 ,说明五级流水线 CPU 符合预期正确处理了所有冒险

    # io_mem_debug_read_data[31:0]

    先后两个时钟周期检测了两个地址所存储的内容

    c.io.mem_debug_read_address.poke(4.U) 读取内存地址为 4 的内容,结果为 1 ,符合预期;

    c.io.mem_debug_read_address.poke(8.U) 读取内存地址为 8 的内容,结果为 3 ,符合预期;

    # FiveStageCPUFinalTest

    src/test/scala/riscv/FiveStageCPUFinalTest.scala

    avatar

    # 测试用例的功能

    测试了使用阻塞、转发以及 ID 阶段判断分支来解决冒险的五级流水线 CPU 的完整功能正确性,检测其是否解决 hazard.asmbin 的所有控制冒险问题

    # 从什么层面测试 CPU

    从对冒险的处理层面测试五级流水线 CPU

    # 加载测试程序指令的方法

    利用定义在 src/main/scala/riscv/TestTopModule.scala 中的模块 TestTopModule ,使用任何 RISC-V 程序 src/main/resources/hazard.asmbin 初始化进行仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    输出波形图中可以看到前面 300 ps 内测试程序执行完毕,此后程序进入死循环,不断重复执行 0000006FH 指令,即 jal 指令,原地循环

    # avatar

    如上图所示,波形图中自上到下三个信号分别为 IF、ID、EX 阶段的指令信号,当信号连续多个周期不发生改变,说明该阶段指令阻塞;当信号变为默认初始值 00000013H ,说明该阶段指令清空。

    avatar

    当确保程序执行完毕后,测试程序检测结果,读取寄存器和内存的值,看其是否符合预期。

    # io_regs_debug_read_data[31:0]

    c.io.regs_debug_read_address.poke(1.U) 使信号 io_regs_debug_read_address [4:0] 的值为 1 ,表示读取 1 号寄存器,即读取 ra 寄存器,读取 io_regs_debug_read_data 的值应当为五级流水线 CPU 执行 hazard.asmbin 两次 csrr 指令之间所花费的时钟周期

    两次 csrr 指令之间有 18 条指令(包括第二条 csrr 但不包括第一条 csrr)在五级流水线执行过程都进入 EX 阶段过,即没有被跳过;且两次 csrr 指令之间经历了 4 次跳转指令控制冒险、1 次无法用转发解决的数据冒险即 lw t2, 2(t2)or t3, t1, t2 之间的数据冒险,读取内存的数据必须等到下一个周期 MEM 阶段才能得到数据。其中第一次控制冒险是由无条件跳转指令引起的,可以通过 ID 阶段只阻塞 1 个周期解决;其余 3 个控制冒险都需要阻塞 2 个时钟周期才能解决;同时数据冒险还在 IF 和 ID 阶段阻塞了 1 个时钟周期。因此两次 csrr 指令之间所花费的时钟周期数为 18+1+3*2+1 = 26

    波形图可见最后读取到 io_regs_debug_read_data 的值等于 26 ,说明五级流水线 CPU 符合预期正确处理了所有冒险

    # io_mem_debug_read_data[31:0]

    先后两个时钟周期检测了两个地址所存储的内容

    c.io.mem_debug_read_address.poke(4.U) 读取内存地址为 4 的内容,结果为 1 ,符合预期;

    c.io.mem_debug_read_address.poke(8.U) 读取内存地址为 8 的内容,结果为 3 ,符合预期;

    # 填空涉及到的信号分析

    (src/main/scala/riscv/)

    主要在填空代码处以注释的形式进行了分析

    # 在完成实验的过程中,遇到的关于实验指导不明确或者其他问题,或者改进的建议。

    使用实验板上的数码管、视频输出等外设,输出一个完整程序的运行结果,或者参考硬件调试一节的内容,用硬件波形的方法捕获程序运行结果。给出你的 CPU 可以正确在实验板上运行程序的照片或者硬件调试器波形截图。

    \ No newline at end of file diff --git a/cs/principles-of-computer-composition/yatcpu/lab-reports/lab4/index.html b/cs/principles-of-computer-composition/yatcpu/lab-reports/lab4/index.html index bf319d95..c139c626 100644 --- a/cs/principles-of-computer-composition/yatcpu/lab-reports/lab4/index.html +++ b/cs/principles-of-computer-composition/yatcpu/lab-reports/lab4/index.html @@ -1 +1 @@ -lab4 - YatCPU - 计算机组成原理 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    YatCPU - - - lab1 - - - lab2 - - - lab3 - - - lab4

    # 测试用例及其波形图分析

    # SayGoodbyeTest

    src/test/scala/riscv/BoardTest.scala

    avatar

    其中 Top 模块来自:

    avatar

    测试执行程序来自 csrc/say_goodbye.c :

    avatar

    # 测试用例的功能

    测试能否正常执行打印 "Never gonna give you up~ Never gonna let you down~\nNever gonna run around and~ desert you~" ,用于烧板后测试能否实现打印,即测试 CPU 包括 lab1 的功能以及能否实现中断处理

    此外,测试用例还通过波形图展示了总线连接下各个设备之间的通信

    # 从什么层面测试 CPU

    从完整性的层面测试 CPU ,若 CPU 可以打印,则说明 CPU 实现了总线以及各个设备通过总线协议的通信

    但并没有测试 CPU 的正确性,只是进行了 pokestep ,但没有对任何信号进行 expect 检查。实际验证需要烧板手动测试能否打印

    # 加载测试程序的方法

    利用定义在 src/main/scala/board/z710/Top.scala 中的模块 Top ,用 RISC-V 程序 src/main/resources/say_goodbye.asmbin 初始化进行仿真

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    所展示波形图里的信号自上而下分别表示:dummy 主设备、CPU、dummy 从设备、instruction_rom 从设备、mem 从设备、timer 从设备、uart 从设备的状态信号 state 变化情况

    instruction_rom 设备状态在一开始不断变化的过程即为加载程序的过程,在 ROMLoaderTest 中体现了这一过程

    instruction_rom 设备加载结束后,程序被加载到内存,CPU 的状态信号开始不断变化,表示 CPU 开始工作,执行程序

    # TimerTest

    src/test/scala/riscv/TimerTest.scala

    avatar

    # 测试用例的功能

    该测试用例用于检查总线模块的正确性,利用 Timer 模块作为从设备,与自定义的 master 模块作为主设备进行通信

    # 从什么层面测试 CPU

    从总线上设备的通信层面测试了 CPU

    # 加载测试程序的方法

    利用定义在 src/main/scala/riscv/core/peripheral/Timer.scala 中的模块 Timer、定义在 src/main/scala/bus/AXI4Lite.scala 中的模块 AXI4LiteMaster 、AXI4LiteSlave 等模块 ,不使用任何 RISC-V 程序初始化进行仿真,直接仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    测试程序模拟了先后向 Timer 写数据、读数据的信号变化不同情况:

    # 写数据

    信号 io_bundle_write = 1 被拉高,向 Timer 内地址为 4 的空间写入数据 io_bundle_write_data = 00990315H ,即向 Timer 的 limit 写入 00990315H 的数据

    写入操作开始后 io_bundle_busy 信号变为 1 ,表示 Timer 忙碌,现在正在与主设备通信实现写操作

    # 写数据成功

    向 Timer 写数据开始的 7 个时钟周期后 Timer 的变量 limit 才变为 00990315H ,此后又花了 1 个时钟周期回复确认,也即 Timer 和主设备之间花了 8 个时钟周期才实现握手、写操作、写回复等。写入 limit 成功后,io_bundle_busy 随即被拉低,表示 Timer 空闲,io_write_valid 被拉高表示写操作请求完成

    # 数据

    信号 io_bundle_read = 1 被拉高,向 Timer 申请读取地址为 4 的数据 limit = 00990315H

    读数据操作开始后 io_bundle_busy 信号变为 1 ,表示 Timer 忙碌,现在正在与主设备通信实现读操作

    # 读数据成功

    向 Timer 写数据开始的 5 个时钟周期后 io_bundle_read_data 才变为 00990315H ,此后又花了 1 个时钟周期回复确认,也即 Timer 和主设备之间花了 6 个时钟周期才实现握手、读操作等。读取 limit 成功后,io_bundle_busy 随即被拉低,表示 Timer 空闲,io_read_valid 被拉高表示读操作成功执行

    # MemoryTest

    src/test/scala/riscv/MemoryTest.scala

    avatar

    # 测试用例的功能

    该测试用例用于检查总线模块的正确性,利用 Memory 模块作为从设备,与自定义的 master 模块作为主设备进行通信

    # 从什么层面测试 CPU

    从总线上设备的通信层面测试了 CPU

    # 加载测试程序的方法

    利用定义在 src/main/scala/riscv/core/peripheral/Memory.scala 中的模块 Memory、定义在 src/main/scala/bus/AXI4Lite.scala 中的 AXI4LiteMaster 、AXI4LiteSlave 等模块,不使用任何 RISC-V 程序初始化进行仿真,直接仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    测试程序模拟了先后向 Memory 写数据、读数据的信号变化不同情况:

    # 写数据

    信号 io_bundle_write = 1 被拉高,向 Memory 内地址为 4 的空间写入数据 io_bundle_write_data = DEADBEEFH

    写入操作开始后 io_bundle_busy 信号变为 1 ,表示 Memory 忙碌,现在正在与主设备通信实现写操作

    # 写数据成功

    向 Memory 写数据开始的 8 个时钟周期后 io_bundle_busy 被拉低,表示 Memory 空闲,io_write_valid 被拉高表示写操作请求完成,也即 Timer 和主设备之间花了 8 个时钟周期才实现握手、写操作、写回复等。

    # 数据

    信号 io_bundle_read = 1 被拉高,向 Memory 申请读取地址为 4 的数据

    读数据操作开始后 io_bundle_busy 信号变为 1 ,表示 Memory 忙碌,现在正在与主设备通信实现读操作

    # 读数据成功

    向 Timer 写数据开始的 6 个时钟周期后读取 limit 成功,io_bundle_busy 被拉低,表示 Timer 空闲,io_read_valid 被拉高表示读操作成功执行,io_bundle_read_data 信号为 DEADBEEFH ,符合预期

    # ROMLoaderTest

    src/test/scala/riscv/ROMLoaderTest.scala

    avatar

    # 测试用例的功能

    该测试用例用于检查总线模块的正确性,利用 ROMLoader 模块作为从设备,与自定义的 slave 模块作为主设备进行通信

    # 从什么层面测试 CPU

    从总线上设备的通信层面测试了 CPU

    # 加载测试程序的方法

    利用定义在 src/main/scala/peripheral/ROMLoader.scala 中的模块 ROMLoader、定义在 src/main/scala/bus/AXI4Lite.scala 中的 AXI4LiteMaster 、AXI4LiteSlave 等模块,不使用任何 RISC-V 程序初始化进行仿真,直接仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    测试程序模拟检测了 ROMLoader 加载程序,即将 ROM 数据写入内存 (从设备) 过程的信号变化过程

    波形图中最底下 2 个信号从上到下分别表示主设备和从设备的 state 状态变量信号,它们的变化展现了 ROMLoader 加载程序过程的不断

    # 写入从设备数据

    信号 io_load_start = 1 被拉高,下一个时钟周期开始 ROMLoader 按顺序将数据写入从设备

    io_load_address 被设置为 100H ,表示从内存地址 100H 开始写数据

    一开始 io_rom_address 的值为 0 ,表示用 rom 内容最开始处的数据写到从设备

    # 写入数据成功

    写入操作开始后 8 个时钟周期后 io_bundle_write 被拉高,表示成功写入数据

    # 加载程序结束

    又进行一次写操作后,由于已经遍历了 ROM 的空间为 2 ,达到了测试程序定义的 ROMLoader 模块设置的最大容量 capacity = 2 ,所以加载程序过程停止。此时 io_load_finish 信号拉高变为 1 ,表明程序加载结束,符合预期

    # threestage.MMIOTest

    src/test/scala/riscv/threestage/CPUTest.scala

    avatar

    csrc/mmio.S

    avatar

    对该程序分析如下:

    RISCV汇编
    .globl _start
    _start:
      li a0, 0x80000000   # 将立即数 0x80000000 加载到寄存器 a0 中,a0=80000000H
      lw t0, 4 (a0)        # 从地址 a0+4=80000004H 处加载一个字(32 位)到寄存器 t0 中,t0=05F5E100H(limit 默认初始值)
      li a1, 0xBEEF       # 将立即数 0xBEEF 加载到寄存器 a1 中,a1=BEEFH
      sw a1, 4 (a0)        # 将寄存器 a1 的内容存储到地址 a0+4=80000004H 处,limit=BEEFH
      nop                 # 空指令,不执行任何操作
      lw t1, 4 (a0)        # 从地址 a0+4 处加载一个字(32 位)到寄存器 t1 中,t1=BEEFH
    loop:
      j loop              # 无条件跳转到标签 loop 处

    可知程序执行结束后,各个寄存器的内容如下:

    • a0:80000000H
    • a1:BEEFH
    • t0:05F5E100H
    • t1:BEEFH

    且主存地址为 80000004H 处的 Timer 内部数据 limit 变为 BEEFH

    # 测试用例的功能

    测试了 CPU 能否与计时器 Timer 通信,并实现 CPU 作为主设备对 Timer 的读写功能

    # 从什么层面测试 CPU

    从总线上设备的通信层面测试了 CPU

    # 加载测试程序指令的方法

    利用定义在 src/main/scala/riscv/TestTopModule.scala 中的模块 TestTopModule ,使用任何 RISC-V 程序 src/main/resources/mmio.asmbin 初始化进行仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    测试程序总共执行了 200 个时钟周期:

    • 前面几乎一半时间用来加载程序,波形图上表现为 ROMLoader 的状态变量 state 在这一段时间不断变化,然后稳定

    • 后面剩余时间用于 CPU 执行程序,波形图上表现为 IF 阶段的 io_id_instruction 变两在这段时间不断变化

    当确保程序执行完毕后,测试程序检测结果,读取寄存器和内存的值,看其是否符合预期

    # register_5[31:0]

    CPU 5 号寄存器的值,即 t0 寄存器的值,最后结果为 05F5E100H,符合预期

    # register_6[31:0]

    CPU 6 号寄存器的值,即 t1 寄存器的值,最后结果为 BEEFH ,符合预期

    # register_10[31:0]

    CPU 10 号寄存器的值,即 a0 寄存器的值,最后结果为 80000000H,符合预期

    # register_11[31:0]

    CPU 11 号寄存器的值,即 a1 寄存器的值,最后结果为 BEEFH,符合预期

    # limit[31:0]

    计时器 Timer 的变量,即内存地址空间为 80000004H 的值,最后结果为 BEEFH,符合预期

    # fivestage.MMIOTest

    该测试与 threestage.MMIOTest 基本一致,不再重复分析

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 填空涉及到的信号分析

    (src/main/scala/riscv/)

    主要在填空代码处以注释的形式进行了分析

    状态转移图:

    avatar

    # 在完成实验的过程中,遇到的关于实验指导不明确或者其他问题,或者改进的建议。

    使用实验板上的数码管、视频输出等外设,输出一个完整程序的运行结果,或者参考硬件调试一节的内容,用硬件波形的方法捕获程序运行结果。给出你的 CPU 可以正确在实验板上运行程序的照片或者硬件调试器波形截图。

    \ No newline at end of file +lab4 - YatCPU - 计算机组成原理 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    YatCPU - - - lab1 - - - lab2 - - - lab3 - - - lab4

    # 测试用例及其波形图分析

    # SayGoodbyeTest

    src/test/scala/riscv/BoardTest.scala

    avatar

    其中 Top 模块来自:

    avatar

    测试执行程序来自 csrc/say_goodbye.c :

    avatar

    # 测试用例的功能

    测试能否正常执行打印 "Never gonna give you up~ Never gonna let you down~\nNever gonna run around and~ desert you~" ,用于烧板后测试能否实现打印,即测试 CPU 包括 lab1 的功能以及能否实现中断处理

    此外,测试用例还通过波形图展示了总线连接下各个设备之间的通信

    # 从什么层面测试 CPU

    从完整性的层面测试 CPU ,若 CPU 可以打印,则说明 CPU 实现了总线以及各个设备通过总线协议的通信

    但并没有测试 CPU 的正确性,只是进行了 pokestep ,但没有对任何信号进行 expect 检查。实际验证需要烧板手动测试能否打印

    # 加载测试程序的方法

    利用定义在 src/main/scala/board/z710/Top.scala 中的模块 Top ,用 RISC-V 程序 src/main/resources/say_goodbye.asmbin 初始化进行仿真

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    所展示波形图里的信号自上而下分别表示:dummy 主设备、CPU、dummy 从设备、instruction_rom 从设备、mem 从设备、timer 从设备、uart 从设备的状态信号 state 变化情况

    instruction_rom 设备状态在一开始不断变化的过程即为加载程序的过程,在 ROMLoaderTest 中体现了这一过程

    instruction_rom 设备加载结束后,程序被加载到内存,CPU 的状态信号开始不断变化,表示 CPU 开始工作,执行程序

    # TimerTest

    src/test/scala/riscv/TimerTest.scala

    avatar

    # 测试用例的功能

    该测试用例用于检查总线模块的正确性,利用 Timer 模块作为从设备,与自定义的 master 模块作为主设备进行通信

    # 从什么层面测试 CPU

    从总线上设备的通信层面测试了 CPU

    # 加载测试程序的方法

    利用定义在 src/main/scala/riscv/core/peripheral/Timer.scala 中的模块 Timer、定义在 src/main/scala/bus/AXI4Lite.scala 中的模块 AXI4LiteMaster 、AXI4LiteSlave 等模块 ,不使用任何 RISC-V 程序初始化进行仿真,直接仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    测试程序模拟了先后向 Timer 写数据、读数据的信号变化不同情况:

    # 写数据

    信号 io_bundle_write = 1 被拉高,向 Timer 内地址为 4 的空间写入数据 io_bundle_write_data = 00990315H ,即向 Timer 的 limit 写入 00990315H 的数据

    写入操作开始后 io_bundle_busy 信号变为 1 ,表示 Timer 忙碌,现在正在与主设备通信实现写操作

    # 写数据成功

    向 Timer 写数据开始的 7 个时钟周期后 Timer 的变量 limit 才变为 00990315H ,此后又花了 1 个时钟周期回复确认,也即 Timer 和主设备之间花了 8 个时钟周期才实现握手、写操作、写回复等。写入 limit 成功后,io_bundle_busy 随即被拉低,表示 Timer 空闲,io_write_valid 被拉高表示写操作请求完成

    # 数据

    信号 io_bundle_read = 1 被拉高,向 Timer 申请读取地址为 4 的数据 limit = 00990315H

    读数据操作开始后 io_bundle_busy 信号变为 1 ,表示 Timer 忙碌,现在正在与主设备通信实现读操作

    # 读数据成功

    向 Timer 写数据开始的 5 个时钟周期后 io_bundle_read_data 才变为 00990315H ,此后又花了 1 个时钟周期回复确认,也即 Timer 和主设备之间花了 6 个时钟周期才实现握手、读操作等。读取 limit 成功后,io_bundle_busy 随即被拉低,表示 Timer 空闲,io_read_valid 被拉高表示读操作成功执行

    # MemoryTest

    src/test/scala/riscv/MemoryTest.scala

    avatar

    # 测试用例的功能

    该测试用例用于检查总线模块的正确性,利用 Memory 模块作为从设备,与自定义的 master 模块作为主设备进行通信

    # 从什么层面测试 CPU

    从总线上设备的通信层面测试了 CPU

    # 加载测试程序的方法

    利用定义在 src/main/scala/riscv/core/peripheral/Memory.scala 中的模块 Memory、定义在 src/main/scala/bus/AXI4Lite.scala 中的 AXI4LiteMaster 、AXI4LiteSlave 等模块,不使用任何 RISC-V 程序初始化进行仿真,直接仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    测试程序模拟了先后向 Memory 写数据、读数据的信号变化不同情况:

    # 写数据

    信号 io_bundle_write = 1 被拉高,向 Memory 内地址为 4 的空间写入数据 io_bundle_write_data = DEADBEEFH

    写入操作开始后 io_bundle_busy 信号变为 1 ,表示 Memory 忙碌,现在正在与主设备通信实现写操作

    # 写数据成功

    向 Memory 写数据开始的 8 个时钟周期后 io_bundle_busy 被拉低,表示 Memory 空闲,io_write_valid 被拉高表示写操作请求完成,也即 Timer 和主设备之间花了 8 个时钟周期才实现握手、写操作、写回复等。

    # 数据

    信号 io_bundle_read = 1 被拉高,向 Memory 申请读取地址为 4 的数据

    读数据操作开始后 io_bundle_busy 信号变为 1 ,表示 Memory 忙碌,现在正在与主设备通信实现读操作

    # 读数据成功

    向 Timer 写数据开始的 6 个时钟周期后读取 limit 成功,io_bundle_busy 被拉低,表示 Timer 空闲,io_read_valid 被拉高表示读操作成功执行,io_bundle_read_data 信号为 DEADBEEFH ,符合预期

    # ROMLoaderTest

    src/test/scala/riscv/ROMLoaderTest.scala

    avatar

    # 测试用例的功能

    该测试用例用于检查总线模块的正确性,利用 ROMLoader 模块作为从设备,与自定义的 slave 模块作为主设备进行通信

    # 从什么层面测试 CPU

    从总线上设备的通信层面测试了 CPU

    # 加载测试程序的方法

    利用定义在 src/main/scala/peripheral/ROMLoader.scala 中的模块 ROMLoader、定义在 src/main/scala/bus/AXI4Lite.scala 中的 AXI4LiteMaster 、AXI4LiteSlave 等模块,不使用任何 RISC-V 程序初始化进行仿真,直接仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    测试程序模拟检测了 ROMLoader 加载程序,即将 ROM 数据写入内存 (从设备) 过程的信号变化过程

    波形图中最底下 2 个信号从上到下分别表示主设备和从设备的 state 状态变量信号,它们的变化展现了 ROMLoader 加载程序过程的不断

    # 写入从设备数据

    信号 io_load_start = 1 被拉高,下一个时钟周期开始 ROMLoader 按顺序将数据写入从设备

    io_load_address 被设置为 100H ,表示从内存地址 100H 开始写数据

    一开始 io_rom_address 的值为 0 ,表示用 rom 内容最开始处的数据写到从设备

    # 写入数据成功

    写入操作开始后 8 个时钟周期后 io_bundle_write 被拉高,表示成功写入数据

    # 加载程序结束

    又进行一次写操作后,由于已经遍历了 ROM 的空间为 2 ,达到了测试程序定义的 ROMLoader 模块设置的最大容量 capacity = 2 ,所以加载程序过程停止。此时 io_load_finish 信号拉高变为 1 ,表明程序加载结束,符合预期

    # threestage.MMIOTest

    src/test/scala/riscv/threestage/CPUTest.scala

    avatar

    csrc/mmio.S

    avatar

    对该程序分析如下:

    RISCV汇编
    .globl _start
    _start:
      li a0, 0x80000000   # 将立即数 0x80000000 加载到寄存器 a0 中,a0=80000000H
      lw t0, 4 (a0)        # 从地址 a0+4=80000004H 处加载一个字(32 位)到寄存器 t0 中,t0=05F5E100H(limit 默认初始值)
      li a1, 0xBEEF       # 将立即数 0xBEEF 加载到寄存器 a1 中,a1=BEEFH
      sw a1, 4 (a0)        # 将寄存器 a1 的内容存储到地址 a0+4=80000004H 处,limit=BEEFH
      nop                 # 空指令,不执行任何操作
      lw t1, 4 (a0)        # 从地址 a0+4 处加载一个字(32 位)到寄存器 t1 中,t1=BEEFH
    loop:
      j loop              # 无条件跳转到标签 loop 处

    可知程序执行结束后,各个寄存器的内容如下:

    • a0:80000000H
    • a1:BEEFH
    • t0:05F5E100H
    • t1:BEEFH

    且主存地址为 80000004H 处的 Timer 内部数据 limit 变为 BEEFH

    # 测试用例的功能

    测试了 CPU 能否与计时器 Timer 通信,并实现 CPU 作为主设备对 Timer 的读写功能

    # 从什么层面测试 CPU

    从总线上设备的通信层面测试了 CPU

    # 加载测试程序指令的方法

    利用定义在 src/main/scala/riscv/TestTopModule.scala 中的模块 TestTopModule ,使用任何 RISC-V 程序 src/main/resources/mmio.asmbin 初始化进行仿真测试

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 执行不同指令时候对应的部件的关键信号的变化情况

    测试程序总共执行了 200 个时钟周期:

    • 前面几乎一半时间用来加载程序,波形图上表现为 ROMLoader 的状态变量 state 在这一段时间不断变化,然后稳定

    • 后面剩余时间用于 CPU 执行程序,波形图上表现为 IF 阶段的 io_id_instruction 变两在这段时间不断变化

    当确保程序执行完毕后,测试程序检测结果,读取寄存器和内存的值,看其是否符合预期

    # register_5[31:0]

    CPU 5 号寄存器的值,即 t0 寄存器的值,最后结果为 05F5E100H,符合预期

    # register_6[31:0]

    CPU 6 号寄存器的值,即 t1 寄存器的值,最后结果为 BEEFH ,符合预期

    # register_10[31:0]

    CPU 10 号寄存器的值,即 a0 寄存器的值,最后结果为 80000000H,符合预期

    # register_11[31:0]

    CPU 11 号寄存器的值,即 a1 寄存器的值,最后结果为 BEEFH,符合预期

    # limit[31:0]

    计时器 Timer 的变量,即内存地址空间为 80000004H 的值,最后结果为 BEEFH,符合预期

    # fivestage.MMIOTest

    该测试与 threestage.MMIOTest 基本一致,不再重复分析

    # 测试用例的执行结果

    avatar

    # 输出波形图

    avatar

    # 填空涉及到的信号分析

    (src/main/scala/riscv/)

    主要在填空代码处以注释的形式进行了分析

    状态转移图:

    avatar

    # 在完成实验的过程中,遇到的关于实验指导不明确或者其他问题,或者改进的建议。

    使用实验板上的数码管、视频输出等外设,输出一个完整程序的运行结果,或者参考硬件调试一节的内容,用硬件波形的方法捕获程序运行结果。给出你的 CPU 可以正确在实验板上运行程序的照片或者硬件调试器波形截图。

    \ No newline at end of file diff --git a/cs/principles-of-computer-composition/yatcpu/yatcpu-report/index.html b/cs/principles-of-computer-composition/yatcpu/yatcpu-report/index.html index b41b2c05..7dba8a03 100644 --- a/cs/principles-of-computer-composition/yatcpu/yatcpu-report/index.html +++ b/cs/principles-of-computer-composition/yatcpu/yatcpu-report/index.html @@ -1 +1 @@ -YatCPU 实验报告 - YatCPU - 计算机组成原理 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # YatCPU 实验报告

    2023 年秋季学期的计算机组成原理实验项目 YatCPU 的实验报告

    # YatCPU 相关资料

    • YatCPU 的教学文档仓库

    • 2023 年秋季学期由 Tokisakix 整理的文档仓库,新纳入了 docker 环境搭建脚本、zybo-z710 开发板移植及烧板脚本

    # 实验报告

    YatCPU - - - lab1 - - - lab2 - - - lab3 - - - lab4

    # 关于挑战实验

    阅读了 egos-2000 操作系统的文档和仓库,按照其教程实现了操作系统映像文件的编译和写入 microSD 卡的过程

    但是 egos-2000 操作系统暂时只支持 Arty A7-35T board、A7-100T board 和 S7-50 board,移植到 z710 上比较困难,所以没能成功

    关于开发板移植的问题仓库有人咨询过,作者给出的回答也很有参考价值:

    avatar

    目前只成功在 qemu 上运行了 egos-2000 操作系统:

    avatar

    附:我们的板子 z710 上原本有一个 ARM 处理器,在官方网站和文档可以查到官方配套的在 z710 上运行 linux 操作系统的方法和工具。但是这个方法使用的是已有的 ARM 处理器

    \ No newline at end of file +YatCPU 实验报告 - YatCPU - 计算机组成原理 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # YatCPU 实验报告

    2023 年秋季学期的计算机组成原理实验项目 YatCPU 的实验报告

    # YatCPU 相关资料

    • YatCPU 的教学文档仓库

    • 2023 年秋季学期由 Tokisakix 整理的文档仓库,新纳入了 docker 环境搭建脚本、zybo-z710 开发板移植及烧板脚本

    # 实验报告

    YatCPU - - - lab1 - - - lab2 - - - lab3 - - - lab4

    # 关于挑战实验

    阅读了 egos-2000 操作系统的文档和仓库,按照其教程实现了操作系统映像文件的编译和写入 microSD 卡的过程

    但是 egos-2000 操作系统暂时只支持 Arty A7-35T board、A7-100T board 和 S7-50 board,移植到 z710 上比较困难,所以没能成功

    关于开发板移植的问题仓库有人咨询过,作者给出的回答也很有参考价值:

    avatar

    目前只成功在 qemu 上运行了 egos-2000 操作系统:

    avatar

    附:我们的板子 z710 上原本有一个 ARM 处理器,在官方网站和文档可以查到官方配套的在 z710 上运行 linux 操作系统的方法和工具。但是这个方法使用的是已有的 ARM 处理器

    \ No newline at end of file diff --git a/cs/raspberrypi/static-ip/index.html b/cs/raspberrypi/static-ip/index.html index 82e25f7b..d82507c7 100644 --- a/cs/raspberrypi/static-ip/index.html +++ b/cs/raspberrypi/static-ip/index.html @@ -1 +1 @@ -树莓派配置静态 ip - 树莓派 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    树莓派配置静态 ip

    h bash
    sudo nano /etc/dhcpcd.conf

    文件末尾添加:

    h bash
    interface eth0
    static ip_address=192.168.134.32/24
    static routers=192.168.134.1
    static domain_name_servers=192.168.134.1 8.8.8.8

    其中 192.168.134.32 可以直接是路由器给树莓派分配的 ip 地址,作为静态 ip 地址

    ctrl+s 保存,ctrl+x 退出,重启树莓派即可

    \ No newline at end of file +树莓派配置静态 ip - 树莓派 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    树莓派配置静态 ip

    h bash
    sudo nano /etc/dhcpcd.conf

    文件末尾添加:

    h bash
    interface eth0
    static ip_address=192.168.134.32/24
    static routers=192.168.134.1
    static domain_name_servers=192.168.134.1 8.8.8.8

    其中 192.168.134.32 可以直接是路由器给树莓派分配的 ip 地址,作为静态 ip 地址

    ctrl+s 保存,ctrl+x 退出,重启树莓派即可

    \ No newline at end of file diff --git a/cs/tools/cmake/index.html b/cs/tools/cmake/index.html index 9d632db7..16d8b829 100644 --- a/cs/tools/cmake/index.html +++ b/cs/tools/cmake/index.html @@ -1 +1 @@ -CMake 基础 - 工具 - 编程语言 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # CMake 简介

    CMake 官网:https://cmake.org

    WIKI:https://en.wikipedia.org/wiki/CMake

    CMake 自身也是一种编程语言,可以用它实现基本的程序逻辑,此如循环、条件分支等等,其中最典型的应用是根据工程的配置来选择性编译部分源代码,比如有些功能只有在测试版本中才会开启。

    和其它的构建工具一样,CMake 的本质依然是定义各个目标之间的关联,比如您的项目由哪些可执行文件、动态库、资源文件组成,然后它们又是通过哪些源文件编译,用到了哪些外部关联等等,本身并没有编译和链接的功能。

    除了基本的构建功能之外,CMake 还允许您添加单元测试、创建安装包、或者将工程拆分成若干个子目录更利于大型项目的管理。

    CMake 与 (GNU) Make 类似,不同之处在于 Make 只是一个命令行工具,用于自动化构建和编译软件项目。Make 使用 Makefile 文件来定义构建过程和依赖关系而 CMake 使用 CMakeLists.txt 。Make 最初是为 Unix 系统设计的,但现在也可用于其他操作系统,且不同平台需要编写不同的 Makefile 文件,而 CMake 则支持跨平台,无需为不同平台分别编写 CMakeLists.txt 文件,还可以更好的实现复杂庞大的项目,功能更加强大,更加方便。

    # CMakeLists.txt

    一个比较完整的 CMakeLists.txt 内容如下:

    工程所需最低 cmake 版本

    cmake
    cmake_minimum_required(VERSION 3.10)

    工程名字 Example

    cmake
    project (Example)

    需要的第三方库 example ,其中 REQUIRED 表示这个库是必须的
    cmake 会自动在机器中寻找符合要求、支持 cmake 构建的第三方库

    cmake
    find_package(example REQUIRED)

    用 GLOB 指令将所有源文件存到变量 SRC_FILE 中

    cmake
    File(GLOB SRC_FILES
    	"${PROJECT_SOURCE_DIR}/src/*.h"
    	"${PROJECT_SOURCE_DIR}/src/*.cpp"
    	"${PROJECT_ SOURCE_DIR}/src/*.c"
    	"${ PROJECT_SOURCE_DIR}/src/*.cc")	# 用正则表达式匹配所有源文件

    构建一个可执行文件,名字为 CMAKE_PROJECT_NAME 即为 Example 工程名字,由 SRC_FILES 里所有文件编译而成

    cmake
    add_executable(${CMAKE_PROJECT_NAME} ${SRC_FILES})

    使用了多个第三方库,需要进行第三方库的链接,否则会出现经典的符号无法解析的报错

    cmake
    target_link_libraries(${CMAKE_PROJECT_NAME} PRIVATE imgui::imgui)
    target_link_libraries(${CMAKE_PROJECT_NAME} PRIVATE glfw)
    target_link_libraries(${CMAKE_PROJECT_NAME} PRIVATE GLEW: :GLEW)
    target_link_libraries(${CMAKE_PROJECT_NAME} PRIVATE gim)

    打开对 c++17 的语法支持

    cmake
    target_compile features (${CMAKE_PROJECT_MAME} PRIVATE cxx_std_17)

    简单的自动化脚本,在构建结束后复制所有 文件

    cmake
    add_custom_command(
      TARGET ${CMAKE_ PROJECT_NAME}
      POST_BUILD		# 表示在【构建】完成后执行
      COMMAND ${CMAKE_COMMAND} -E copy_directory
      	"${PROJECT_SOURCE_DIR}/assets"
      	"$<TARGET_FILE_DIR:${CMAKE_PROJECT_NAME}>/assets")
    cmake
    add_custom_command (
      TARGET ${CMAKE_PROJECT_NAME}
      POST_BUILD
      COMMAND ${CMAKE_COMMAND}-E copy_directory
        "${PROJECT_SOURCE_DIR) /shader/"
        "$<TARGET_FILE_DIR:${CMAKE_PROJECT_NAME}>/shader")

    # 配置 (Configure)

    根据 CMakeLists 文件生成目标平台下的原生工程

    bash
    cmake -S . -B build

    使用了第三方库时,若使用 vcpkg 管理第三方库,则需要:

    bash
    cmake -S . -B build -DCMAKE_TOOLCHAIN_FILE=<vcpkg安装路径:.../vcpkg>/scripts/buildsystems/vcpkg.cmake

    使用 VScode 则需在 settings.json 中添加:

    json
    {
      "make configureSettings":{"MAKE_TOOLCHAIN_FILE": "<vcpkg安装路径:.../vcpkg>/scripts/buildsystems/vcpkg.cmake"
    	}
    }

    # 构建 (Build)

    构建生成可执行文件

    bash
    cmake --build build

    # 第三方库

    使用微软的跨平台开源工具 vcpkg 可以更方便的安装和管理 c/c++ 第三方库,类似 python 的 pip

    Edited on Views times
    \ No newline at end of file +CMake 基础 - 工具 - 编程语言 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # CMake 简介

    CMake 官网:https://cmake.org

    WIKI:https://en.wikipedia.org/wiki/CMake

    CMake 自身也是一种编程语言,可以用它实现基本的程序逻辑,此如循环、条件分支等等,其中最典型的应用是根据工程的配置来选择性编译部分源代码,比如有些功能只有在测试版本中才会开启。

    和其它的构建工具一样,CMake 的本质依然是定义各个目标之间的关联,比如您的项目由哪些可执行文件、动态库、资源文件组成,然后它们又是通过哪些源文件编译,用到了哪些外部关联等等,本身并没有编译和链接的功能。

    除了基本的构建功能之外,CMake 还允许您添加单元测试、创建安装包、或者将工程拆分成若干个子目录更利于大型项目的管理。

    CMake 与 (GNU) Make 类似,不同之处在于 Make 只是一个命令行工具,用于自动化构建和编译软件项目。Make 使用 Makefile 文件来定义构建过程和依赖关系而 CMake 使用 CMakeLists.txt 。Make 最初是为 Unix 系统设计的,但现在也可用于其他操作系统,且不同平台需要编写不同的 Makefile 文件,而 CMake 则支持跨平台,无需为不同平台分别编写 CMakeLists.txt 文件,还可以更好的实现复杂庞大的项目,功能更加强大,更加方便。

    # CMakeLists.txt

    一个比较完整的 CMakeLists.txt 内容如下:

    工程所需最低 cmake 版本

    cmake
    cmake_minimum_required(VERSION 3.10)

    工程名字 Example

    cmake
    project (Example)

    需要的第三方库 example ,其中 REQUIRED 表示这个库是必须的
    cmake 会自动在机器中寻找符合要求、支持 cmake 构建的第三方库

    cmake
    find_package(example REQUIRED)

    用 GLOB 指令将所有源文件存到变量 SRC_FILE 中

    cmake
    File(GLOB SRC_FILES
    	"${PROJECT_SOURCE_DIR}/src/*.h"
    	"${PROJECT_SOURCE_DIR}/src/*.cpp"
    	"${PROJECT_ SOURCE_DIR}/src/*.c"
    	"${ PROJECT_SOURCE_DIR}/src/*.cc")	# 用正则表达式匹配所有源文件

    构建一个可执行文件,名字为 CMAKE_PROJECT_NAME 即为 Example 工程名字,由 SRC_FILES 里所有文件编译而成

    cmake
    add_executable(${CMAKE_PROJECT_NAME} ${SRC_FILES})

    使用了多个第三方库,需要进行第三方库的链接,否则会出现经典的符号无法解析的报错

    cmake
    target_link_libraries(${CMAKE_PROJECT_NAME} PRIVATE imgui::imgui)
    target_link_libraries(${CMAKE_PROJECT_NAME} PRIVATE glfw)
    target_link_libraries(${CMAKE_PROJECT_NAME} PRIVATE GLEW: :GLEW)
    target_link_libraries(${CMAKE_PROJECT_NAME} PRIVATE gim)

    打开对 c++17 的语法支持

    cmake
    target_compile features (${CMAKE_PROJECT_MAME} PRIVATE cxx_std_17)

    简单的自动化脚本,在构建结束后复制所有 文件

    cmake
    add_custom_command(
      TARGET ${CMAKE_ PROJECT_NAME}
      POST_BUILD		# 表示在【构建】完成后执行
      COMMAND ${CMAKE_COMMAND} -E copy_directory
      	"${PROJECT_SOURCE_DIR}/assets"
      	"$<TARGET_FILE_DIR:${CMAKE_PROJECT_NAME}>/assets")
    cmake
    add_custom_command (
      TARGET ${CMAKE_PROJECT_NAME}
      POST_BUILD
      COMMAND ${CMAKE_COMMAND}-E copy_directory
        "${PROJECT_SOURCE_DIR) /shader/"
        "$<TARGET_FILE_DIR:${CMAKE_PROJECT_NAME}>/shader")

    # 配置 (Configure)

    根据 CMakeLists 文件生成目标平台下的原生工程

    bash
    cmake -S . -B build

    使用了第三方库时,若使用 vcpkg 管理第三方库,则需要:

    bash
    cmake -S . -B build -DCMAKE_TOOLCHAIN_FILE=<vcpkg安装路径:.../vcpkg>/scripts/buildsystems/vcpkg.cmake

    使用 VScode 则需在 settings.json 中添加:

    json
    {
      "make configureSettings":{"MAKE_TOOLCHAIN_FILE": "<vcpkg安装路径:.../vcpkg>/scripts/buildsystems/vcpkg.cmake"
    	}
    }

    # 构建 (Build)

    构建生成可执行文件

    bash
    cmake --build build

    # 第三方库

    使用微软的跨平台开源工具 vcpkg 可以更方便的安装和管理 c/c++ 第三方库,类似 python 的 pip

    Edited on Views times
    \ No newline at end of file diff --git a/cs/tools/vscode-remote-ssh-login-shell/index.html b/cs/tools/vscode-remote-ssh-login-shell/index.html index e91511c1..9e7992c7 100644 --- a/cs/tools/vscode-remote-ssh-login-shell/index.html +++ b/cs/tools/vscode-remote-ssh-login-shell/index.html @@ -1 +1 @@ -VSCode 远程 SSH 连接的 shell 相关一个 bug - 工具 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 环境 / 情景

    MacOS 14.2 M1 用 VSCode 远程连接 Ubuntu 22.04

    # 问题

    在虚拟机上安装了 Enviroment Modules ,并按照官方配置文档配置了 bash 启动时自动加载 module 指令,ssh 远程登录虚拟机 'module' 指令可用,但是 VSCode 远程连接虚拟机发现 module 指令使用不了。

    # 原因

    在 VSCode 的 bash 中检查发现当前 shell 不是登录式的:

    shell
    shopt login_shell
    # 结果为 off 不是登陆式
    login_shell    	off

    分析发现由于 VSCode 启动 shell 是非登陆式,参考这篇,所以没有加载 /etc/profile 文件,而该文件下包含如下内容:

    shell
    if [ -d /etc/profile.d ]; then
      for i in /etc/profile.d/*.sh; do
        if [ -r $i ]; then
          . $i
        fi
      done
      unset i
    fi

    因此也没有加载 /etc/profile.d/ 目录下的文件,自然也没有加载 Enviroment Modules 官方文档配置的 /etc/profile.d/modules.sh/etc/profile.d/modules.csh 这些 Linux Link。导致了 VSCode 的终端无法使用 'module' 指令。

    去官方文档搜索找到了一个 open 的 issue :https://github.com/microsoft/vscode-remote-release/issues/8484 ,也是同样的问题,暂时没有解决。

    # 临时解决方案

    暂时无法从 VSCode 这一远程连接服务端解决,那就只能:

    • 要么从控制端解决,每次 VSCode 打开一个 shell 后手动输指令加载 /etc/profile 文件
    • 要么从受控端解决
      • 在 Ubuntu 里编辑 /etc/bash.bashrc~/.bashrc 让它加载 /etc/profile 等 login shell 才能加载的文件,虽然保证了 VSCode 的 shell 环境能够与 login shell 一致更方便了,但是这么做修改了系统设计可能会导致其他问题。
      • 更安全的做法是单独在 /etc/bash.bashrc~/.bashrc 里添加单独加载指定的需要的配置文件,还可以单独自定义一个目录存放这样的配置文件并用 iffor 加载这个目录所有文件(模仿 /etc/profile 文件里的操作)。
    \ No newline at end of file +VSCode 远程 SSH 连接的 shell 相关一个 bug - 工具 - 计算机科学 | Balloon Party = 風船のパーティー = fuusen no party

    # 环境 / 情景

    MacOS 14.2 M1 用 VSCode 远程连接 Ubuntu 22.04

    # 问题

    在虚拟机上安装了 Enviroment Modules ,并按照官方配置文档配置了 bash 启动时自动加载 module 指令,ssh 远程登录虚拟机 'module' 指令可用,但是 VSCode 远程连接虚拟机发现 module 指令使用不了。

    # 原因

    在 VSCode 的 bash 中检查发现当前 shell 不是登录式的:

    shell
    shopt login_shell
    # 结果为 off 不是登陆式
    login_shell    	off

    分析发现由于 VSCode 启动 shell 是非登陆式,参考这篇,所以没有加载 /etc/profile 文件,而该文件下包含如下内容:

    shell
    if [ -d /etc/profile.d ]; then
      for i in /etc/profile.d/*.sh; do
        if [ -r $i ]; then
          . $i
        fi
      done
      unset i
    fi

    因此也没有加载 /etc/profile.d/ 目录下的文件,自然也没有加载 Enviroment Modules 官方文档配置的 /etc/profile.d/modules.sh/etc/profile.d/modules.csh 这些 Linux Link。导致了 VSCode 的终端无法使用 'module' 指令。

    去官方文档搜索找到了一个 open 的 issue :https://github.com/microsoft/vscode-remote-release/issues/8484 ,也是同样的问题,暂时没有解决。

    # 临时解决方案

    暂时无法从 VSCode 这一远程连接服务端解决,那就只能:

    • 要么从控制端解决,每次 VSCode 打开一个 shell 后手动输指令加载 /etc/profile 文件
    • 要么从受控端解决
      • 在 Ubuntu 里编辑 /etc/bash.bashrc~/.bashrc 让它加载 /etc/profile 等 login shell 才能加载的文件,虽然保证了 VSCode 的 shell 环境能够与 login shell 一致更方便了,但是这么做修改了系统设计可能会导致其他问题。
      • 更安全的做法是单独在 /etc/bash.bashrc~/.bashrc 里添加单独加载指定的需要的配置文件,还可以单独自定义一个目录存放这样的配置文件并用 iffor 加载这个目录所有文件(模仿 /etc/profile 文件里的操作)。
    \ No newline at end of file diff --git a/css/app.css b/css/app.css index 1a1629ad..6f90dfb6 100644 --- a/css/app.css +++ b/css/app.css @@ -1,3 +1,3 @@ -/* build time:Fri Jun 28 2024 18:11:52 GMT+0800 (中国标准时间)*/ +/* build time:Mon Jul 01 2024 11:40:16 GMT+0800 (中国标准时间)*/ #nav .menu .item:not(.title) a::before{content:"";position:absolute;width:0;height:.1875rem;bottom:0;border-radius:.125rem;left:50%;transform:translateX(-50%);background-color:currentColor}#nav .menu .submenu .item.active,#nav .menu .submenu .item:hover,#search-pagination .current .page-number,#sidebar .tab .item.active,#sidebar .tab .item:hover,.overview .menu .item.active,.pagination .next:hover,.pagination .page-number.current,.pagination .page-number:hover,.pagination .prev:hover{color:var(--grey-0);background-image:linear-gradient(to right,var(--color-pink) 0,var(--color-orange) 100%);box-shadow:0 0 .75rem var(--color-pink-a3)}#loading,#neko{position:fixed;left:0;right:0;top:0;bottom:0;z-index:9999}:root{--grey-0:#fff;--grey-1:#fdfdfd;--grey-2:#f7f7f7;--grey-3:#eff2f3;--grey-4:#ccc;--grey-5:#999;--grey-6:#666;--grey-7:#333;--grey-8:#222;--grey-9:#000;--grey-1-a0:rgba(253,253,253,0);--grey-1-a7:rgba(253,253,253,0.7);--grey-1-a5:rgba(253,253,253,0.5);--grey-1-a3:rgba(253,253,253,0.3);--grey-9-a1:rgba(0,0,0,0.1);--grey-9-a5:rgba(0,0,0,0.5);--grey-2-a0:rgba(247,247,247,0);--color-pink-light:#ffe6fa;--color-cyan-light:#e3fdf5;--color-red:#e9546b;--color-pink:#ed6ea0;--color-orange:#ec8c69;--color-yellow:#eab700;--color-green:#0a7426;--color-aqua:#3e999f;--color-blue:#38a1db;--color-purple:#9d5b8b;--color-grey:#869194;--color-red-a1:rgba(233,84,107,0.1);--color-red-a3:rgba(233,84,107,0.3);--color-pink-a3:rgba(237,110,160,0.3);--color-pink-light-a3:rgba(255,230,250,0.3);--color-pink-light-a5:rgba(255,230,250,0.5);--color-pink-light-a7:rgba(255,230,250,0.7);--body-bg-shadow:var(--grey-2);--box-bg-shadow:var(--grey-9-a1);--text-color:var(--grey-7);--header-text-color:var(--grey-0);--primary-color:var(--color-red);--nav-bg:linear-gradient(-225deg, var(--color-cyan-light) 0, var(--color-pink-light) 100%)}.primary{--note-border:#cda0c7;--note-bg:#fdf8ff;--note-text:#8a51c0;--note-hover:#935aca}.info{--note-border:#8fa4dc;--note-bg:#f1f9ff;--note-text:#1d4974;--note-hover:#1d5fa0}.success{--note-border:#a3c293;--note-bg:#fcfff5;--note-text:#2c662d;--note-hover:#3b883c}.warning{--note-border:#c9ba9b;--note-bg:#fffbeb;--note-text:#947600;--note-hover:#ccb045}.danger{--note-border:#f4b3c1;--note-bg:#fff2f5;--note-text:#cc0f35;--note-hover:#f14668}[data-theme=dark]:root{--grey-0:#222;--grey-1:#21252b;--grey-2:#363636;--grey-3:#444;--grey-4:#666;--grey-5:#aaa;--grey-6:#ccc;--grey-7:#ddd;--grey-8:#eee;--grey-9:#f7f7f7;--grey-1-a7:rgba(34,34,34,0.7);--grey-1-a5:rgba(34,34,34,0.5);--grey-1-a3:rgba(34,34,34,0.3);--grey-1-a0:rgba(34,34,34,0);--grey-9-a1:rgba(51,51,51,0.1);--grey-2-a0:rgba(54,54,54,0);--color-pink-light:#322d31;--color-cyan-light:#2d3230;--color-red:rgba(237,118,137,0.9);--color-pink:rgba(241,139,179,0.8);--color-orange:rgba(240,163,135,0.8);--color-yellow:#ffe175;--color-green:#86c59d;--color-aqua:#97d3d6;--color-blue:#9cd0ed;--color-purple:#cfacc5;--color-grey:#c3c8ca;--body-bg-shadow:#000;--box-bg-shadow:#000;--text-color:var(--grey-5);--header-text-color:var(--grey-9)}[data-theme=dark] .primary{--note-border:rgba(123,96,119,0.8);--note-bg:rgba(50,49,50,0.8);--note-text:rgba(161,116,205,0.8);--note-hover:rgba(117,72,161,0.8)}[data-theme=dark] .info{--note-border:rgba(85,98,132,0.8);--note-bg:rgba(48,49,50,0.8);--note-text:rgba(109,164,219,0.8);--note-hover:rgba(39,127,214,0.8)}[data-theme=dark] .success{--note-border:rgba(97,116,88,0.8);--note-bg:rgba(50,50,48,0.8);--note-text:rgba(128,200,129,0.8);--note-hover:rgba(41,95,42,0.8)}[data-theme=dark] .warning{--note-border:rgba(120,111,93,0.8);--note-bg:rgba(50,50,46,0.8);--note-text:rgba(220,176,0,0.8);--note-hover:rgba(163,140,55,0.8)}[data-theme=dark] .danger{--note-border:rgba(146,107,115,0.8);--note-bg:rgba(50,48,48,0.8);--note-text:rgba(239,38,79,0.8);--note-hover:rgba(168,49,72,0.8)}[data-theme=dark] .index.wrap .card .message .btn,[data-theme=dark] .overview .menu .item,[data-theme=dark] .sidebar .tab li,[data-theme=dark] [data-background-image],[data-theme=dark] img{transition:all .2s ease-in-out 0s;opacity:.75}[data-theme=dark] .index.wrap .card .message .btn:hover,[data-theme=dark] .overview .menu .item:hover,[data-theme=dark] .sidebar .tab li:hover,[data-theme=dark] [data-background-image]:hover,[data-theme=dark] img:hover{opacity:.9}[data-theme=dark] #imgs::before{background-color:rgba(0,0,0,.5)}.red{color:var(--color-red)}.pink{color:var(--color-pink)}.orange{color:var(--color-orange)}.yellow{color:var(--color-yellow)}.green{color:var(--color-green)}.aqua{color:var(--color-aqua)}.blue{color:var(--color-blue)}.purple{color:var(--color-purple)}.grey{color:var(--color-grey)}.i-bilibili:before{content:"\e652"}.i-volume-off:before,.player-info .volume.off::before{content:"\e61e"}.i-volume-on:before,.player-info .volume.on::before{content:"\e62c"}.i-circle-play:before,.player-info .play-pause::before{content:"\e647"}.i-forward:before,.player-info .forward::before{content:"\e648"}.i-backward:before,.player-info .backward::before{content:"\e649"}.i-circle-pause:before,.playing .player-info .play-pause::before{content:"\e64a"}.i-loop:before,.player-info .mode.loop::before{content:"\e64b"}.i-order:before,.player-info .mode.order::before{content:"\e64c"}.i-random:before,.player-info .mode.random::before{content:"\e64d"}.i-douban:before{content:"\e75f"}.i-linux:before{content:"\f1e8"}.i-opera:before{content:"\f205"}.i-qq:before{content:"\f216"}.i-safari:before{content:"\f229"}.i-snapchat-ghost:before{content:"\f234"}.i-weixin:before{content:"\f262"}.i-windows:before{content:"\f266"}.i-stars:before{content:"\e8c4"}.i-apple:before{content:"\e600"}.i-blackberry:before{content:"\e601"}.i-centos:before{content:"\e602"}.i-fedora:before{content:"\e603"}.i-redhat:before{content:"\e604"}.i-ubuntu:before{content:"\e605"}.i-suse:before{content:"\e606"}.i-mobile-alt:before{content:"\f052"}.i-paw:before{content:"\f06b"}.i-android:before{content:"\f161"}.i-chrome:before{content:"\f178"}.i-edge:before{content:"\f195"}.i-firefox:before{content:"\f1a1"}.i-internet-explorer:before{content:"\f1d4"}.i-markdown:before{content:"\f1eb"}.i-smile:before{content:"\f2a5"}.i-preview:before{content:"\e901"}#copyright .license::before,.i-share:before{content:"\e61b"}#copyright .link::before,.i-link-circle:before{content:"\e67b"}#copyright .author::before,.i-person:before{content:"\e69d"}.i-sun:before{content:"\e6d1"}.i-moon:before{content:"\e71e"}.i-compress:before{content:"\ef82"}.i-expand:before{content:"\efb4"}.i-align-justify:before{content:"\ef13"}.i-align-left:before{content:"\ef14"}.i-eye:before{content:"\efb8"}.i-pen:before{content:"\f071"}.i-clock:before{content:"\ef75"}.i-flag:before{content:"\e680"}.i-at:before{content:"\e619"}.i-file:before{content:"\e68d"}.i-clipboard:before{content:"\e651"}.i-feather:before{content:"\efbd"}#tool .player .music::before,.i-music:before{content:"\f059"}#tool.playing .player .play-pause::before,.i-pause:before{content:"\f06a"}.i-comments:before{content:"\ef7f"}#tool .player .play-pause::before,.i-play:before,.player-info ol>li.current::before{content:"\f07f"}.i-calendar-check:before{content:"\ef5b"}.i-angle-up:before{content:"\ef1b"}.i-facebook:before{content:"\f19d"}.i-instagram:before{content:"\f1d3"}.i-skype:before{content:"\f231"}.i-stack-overflow:before{content:"\f239"}.i-youtube:before{content:"\f274"}.i-list-alt:before{content:"\e6c1"}.i-star:before,.page .body h3 .anchor::before{content:"\f0d4"}.i-link-alt:before{content:"\f037"}.i-paper-plane:before{content:"\f063"}.i-user:before{content:"\f2dd"}.i-link:before{content:"\e8fc"}.i-angle-down:before,details[open] summary::before{content:"\ef1a"}.i-calendar:before{content:"\e812"}#sidebar .tab .item.overview::before,.i-home:before{content:"\e8ed"}.i-magic:before{content:"\f03e"}.i-sakura:before,.page .body h1 .anchor::before{content:"\e695"}.i-tag:before{content:"\e759"}.i-angle-left:before{content:"\ef19"}.i-arrow-circle-right:before,.md .note.default::before{content:"\ef23"}.i-check-circle:before,.md .note.success::before,.md ul li.task-list-item input[type=checkbox]:checked+label::before{content:"\ef66"}.i-exclamation-circle:before,.md .note.warning::before{content:"\efb5"}.i-info-circle:before,.md .note.info::before{content:"\f02b"}.i-minus-circle:before,.md .note.danger::before{content:"\f050"}.i-plus-circle:before,.md .note.primary::before{content:"\f082"}.i-file-word:before{content:"\f299"}.i-check:before,.md ol>li.quiz.show.true::after,.md ol>li.quiz>ul.options li.right::after{content:"\ef65"}.i-times:before,.md ol>li.quiz.show.false::after,.md ol>li.quiz>ul.options li.wrong::after{content:"\f109"}#sidebar .tab .item.contents::before,.i-list-ol:before{content:"\f039"}.i-archive:before{content:"\ef1c"}.i-angle-right:before,details summary::before{content:"\ef1f"}.i-arrow-down:before{content:"\ef25"}.i-arrow-up:before,.tabs .show-btn::before{content:"\ef2a"}.i-chart-area:before{content:"\ef64"}.i-chevron-left:before{content:"\ef6e"}.i-chevron-right:before{content:"\ef6f"}.i-coffee:before{content:"\ef7a"}.i-envelope:before{content:"\efae"}.i-external-link-alt:before{content:"\efb6"}.i-heart:before{content:"\f013"}.i-heartbeat:before,.page .body h2 .anchor::before{content:"\f017"}.i-search:before{content:"\f0a8"}#sidebar .tab .item.related::before,.i-sitemap:before{content:"\f0bd"}.i-tags:before{content:"\f0f3"}.i-th:before{content:"\f0fc"}.i-thumbtack:before{content:"\f107"}.i-times-circle:before,.md ul li.task-list-item input[type=checkbox]+label::before{content:"\f10a"}.i-creative-commons:before{content:"\f17e"}.i-github:before{content:"\f1b4"}.i-twitter:before{content:"\f24d"}.i-weibo:before{content:"\f261"}.i-address-card:before{content:"\f278"}.i-zhihu:before{content:"\e765"}.i-cloud-music:before{content:"\e76a"}html{line-height:1.15;-webkit-text-size-adjust:100%}body{margin:0}main{display:block}h1{font-size:2em;margin:.67em 0}hr{box-sizing:content-box;height:0;overflow:visible}pre{font-family:monospace,monospace;font-size:1em}a{background:0 0}abbr[title]{border-bottom:none;text-decoration:underline;-webkit-text-decoration:underline dotted;text-decoration:underline dotted}b,strong{font-weight:bolder}code,kbd,samp{font-family:monospace,monospace;font-size:1em}small{font-size:80%}sub,sup{font-size:75%;line-height:0;position:relative;vertical-align:baseline}sub{bottom:-.25em}sup{top:-.5em}img{border-style:none}button,input,optgroup,select,textarea{font-family:inherit;font-size:100%;line-height:1.15;margin:0}button,input{overflow:visible}button,select{text-transform:none}[type=button],[type=reset],[type=submit],button{-webkit-appearance:button}[type=button]::-moz-focus-inner,[type=reset]::-moz-focus-inner,[type=submit]::-moz-focus-inner,button::-moz-focus-inner{border-style:none;padding:0}[type=button]:-moz-focusring,[type=reset]:-moz-focusring,[type=submit]:-moz-focusring,button:-moz-focusring{outline:.0625rem dotted ButtonText}fieldset{padding:.35em .75em .625em}legend{box-sizing:border-box;color:inherit;display:table;max-width:100%;padding:0;white-space:normal}progress{vertical-align:baseline}textarea{overflow:auto}[type=checkbox],[type=radio]{box-sizing:border-box;padding:0}[type=number]::-webkit-inner-spin-button,[type=number]::-webkit-outer-spin-button{height:auto}[type=search]{outline-offset:-0.125rem;-webkit-appearance:textfield}[type=search]::-webkit-search-decoration{-webkit-appearance:none}::-webkit-file-upload-button{font:inherit;-webkit-appearance:button}details{display:block}summary{display:list-item}template{display:none}[hidden]{display:none}::-webkit-scrollbar{width:.3125rem;height:.3125rem;background:0 0}::-webkit-scrollbar-track{border-radius:.3125rem;background:0 0}::-webkit-scrollbar-track-piece{background:0 0}::-webkit-scrollbar-button{background:0 0;height:0}::-webkit-scrollbar-thumb{border-radius:1.25rem;background-clip:padding-box;background-color:var(--grey-3);background-image:linear-gradient(45deg,var(--grey-1-a5) 25%,transparent 0,transparent 50%,var(--grey-1-a5) 0,var(--grey-1-a5) 75%,transparent 0,transparent)}body.loaded::-webkit-scrollbar{width:.5rem;height:.5rem}body.loaded::-webkit-scrollbar-thumb{background-color:var(--color-pink-light)}.page .body h1 .anchor::before,.rotate{animation:rotate 6s linear infinite}.beat,.page .body h2 .anchor::before{animation:beat 1.33s ease-in-out infinite}.flash,.page .body h3 .anchor::before{animation:flash 6s cubic-bezier(.22,.61,.36,1) infinite}.overview .author:hover .image,.ribbon a:hover,.shake{animation:shake 1s}.fade-in,.tabs .tab,.tip,details[open]>div{animation:fadeIn .5s}#tool.affix .player-info.show.hide,.fade-out,.tip.hide{animation:fadeOut .3s}.code-container .show-btn .ic,.up-down{animation:UpDown 2s infinite}.code-container .show-btn.open .ic,.down-up{animation:DownUp 2s infinite}.md ol>li.quiz.show .note,.md ol>li.quiz.show blockquote,.slide{animation:slide .5s}#nav .menu .submenu,.slide-up-in{animation:slideUpIn .3s}.cards .item.show,.segments>.item.show,.slide-up-big-in,body.loaded #main .wrap,body.loaded #sidebar .panel.active{animation:slideUpBigIn .5s}#tool.affix .player-info.show,.slide-right-in{animation:slideRightIn .3s}.slide-left-in{animation:slideLeftIn .3s}.overview .menu .item .submenu,.slide-down-in,body.loaded #brand .pjax{animation:slideDownIn .3s}.blur,.lozaded{animation:blur .8s ease-in-out forwards}.elastic,.highlight.fullscreen{animation:elastic 1s}@keyframes rotate{from{transform:rotate(0)}to{transform:rotate(360deg)}}@keyframes rotating{from{transform:rotate(720deg)}to{transform:none}}@keyframes rotate-needle-pause{0%{transform:rotateZ(-35deg)}100%{transform:rotateZ(-60deg)}}@keyframes rotate-needle-resume{0%{transform:rotateZ(-60deg)}100%{transform:rotateZ(-35deg)}}@keyframes beat{0%,100%{transform:scale(1)}10%,30%{transform:scale(.9)}20%,40%,60%,80%{transform:scale(1.1)}50%,70%{transform:scale(1.1)}}@keyframes flash{0%,50%,to{opacity:1}25%,75%{opacity:0}}@keyframes shake{from,to{transform:translate3d(0,0,0)}10%,30%,50%,70%,90%{transform:translate3d(-10px,0,0)}20%,40%,60%,80%{transform:translate3d(10px,0,0)}}@keyframes fadeIn{0%{opacity:0}100%{opacity:1}}@keyframes fadeOut{0%{opacity:1}100%{opacity:0}}@keyframes blur{0%{filter:blur(10px)}to{filter:blur(0)}}@keyframes blur-dark{0%{filter:blur(10px) brightness(.9)}to{filter:blur(0) brightness(.9)}}@keyframes UpDown{0%,100%{opacity:.8;transform:translateY(10px)}50%{opacity:.4;transform:translateY(0)}}@keyframes DownUp{0%,100%{opacity:.8;transform:rotate(180deg) translateY(0)}50%{opacity:.4;transform:rotate(180deg) translateY(-10px)}}@keyframes slide{0%{opacity:0;transform:scaleY(0)}100%{opacity:1;transform:scaleY(1)}}@keyframes slideRightIn{0%{opacity:0;transform:translateX(50%)}to{opacity:1;transform:translateX(0)}}@keyframes slideLeftIn{0%{opacity:0;transform:translateX(-50%)}to{opacity:1;transform:translateX(0)}}@keyframes slideUpIn{0%{opacity:0;transform:translateY(10px)}to{opacity:1;transform:translateY(0)}}@keyframes slideUpBigIn{0%{opacity:0;transform:translateY(80px)}100%{opacity:1;transform:translateY(0)}}@keyframes slideDownIn{0%{opacity:0;transform:translateY(-18px)}100%{opacity:1;transform:translateY(0)}}@keyframes elastic{0%{transform:scale(0)}55%{transform:scale(1)}70%{transform:scale(.98)}100%{transform:scale(1)}}::-moz-selection{background:var(--color-orange);color:var(--grey-2)}::selection{background:var(--color-orange);color:var(--grey-2)}*,:after,:before{box-sizing:border-box;margin:0;padding:0}body,html{height:100%}body{background:linear-gradient(to top,var(--body-bg-shadow) 0,var(--grey-1) 20%) no-repeat bottom;color:var(--text-color);font-family:Mulish,-apple-system,"PingFang SC","Microsoft YaHei",sans-serif;font-size:1em;line-height:2;-webkit-font-smoothing:antialiased;overflow:hidden}body.loaded{overflow:auto}body.loaded #sidebar .panel.active{display:block}@media (max-width:991px){body{padding-left:0!important;padding-right:0!important}}body.fullscreen{overflow:hidden}body.fullscreen #sidebar>.inner,body.fullscreen #tool{z-index:0}h1,h2,h3,h4,h5,h6{font-family:Mulish,'Noto Serif SC','Noto Serif JP',Mulish,-apple-system,"PingFang SC","Microsoft YaHei",sans-serif;font-weight:700;line-height:1.5;margin:1.25rem 0 .9375rem}h1.title,h2.title,h3.title,h4.title,h5.title,h6.title{font-family:Mulish,'Noto Serif JP','Noto Serif SC',Mulish,-apple-system,"PingFang SC","Microsoft YaHei",sans-serif}a{border:none;color:currentColor;outline:0;text-decoration:none;overflow-wrap:break-word;word-wrap:break-word;transition:all .2s ease-in-out 0s;cursor:pointer}a:hover{border-bottom-color:var(--color-blue);color:var(--color-blue)}a::after,a::before{transition:all .4s ease-in-out 0s}a,div,li{-webkit-tap-highlight-color:transparent}li{list-style:none}iframe,img,video{display:block;margin-left:auto;margin-right:auto;max-width:100%}hr{background-image:repeating-linear-gradient(-45deg,var(--grey-3),var(--grey-4) .25rem,transparent .25rem,transparent .5rem);border:none;height:.125rem;margin:1.5rem 0}blockquote{border-left:.25rem solid var(--grey-4);color:var(--grey-6);margin:1.25rem 0;padding:.625rem 1.25rem}blockquote cite::before{content:'-';padding:0 .3125rem}dt{font-weight:700}dd{margin:0;padding:0}input,textarea{color:var(--text-color)}@font-face{font-family:ic;src:url("//at.alicdn.com/t/font_1832207_igi8uaupcus.eot");src:url("//at.alicdn.com/t/font_1832207_igi8uaupcus.eot?#iefix") format('embedded-opentype'),url("//at.alicdn.com/t/font_1832207_igi8uaupcus.woff2") format('woff2'),url("//at.alicdn.com/t/font_1832207_igi8uaupcus.woff") format('woff'),url("//at.alicdn.com/t/font_1832207_igi8uaupcus.ttf") format('truetype'),url("//at.alicdn.com/t/font_1832207_igi8uaupcus.svg#ic") format('svg')}.ic{font-family:ic!important;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;text-align:center;width:1.25em;display:inline-block;font-style:normal;font-variant:normal;text-rendering:auto;line-height:1}.ic em{font-size:0}.ic-lg{font-size:1.33333em;line-height:.75em;vertical-align:-.0667em}.table-container{overflow:auto}.table-container table{border-collapse:collapse;border-spacing:0;font-size:.875em;margin:0 0 1.25rem 0;width:100%;overflow:auto}.table-container table tbody tr:nth-of-type(2n){background:var(--grey-0)}.table-container table tbody tr:hover{background:var(--grey-2)}.table-container table caption,.table-container table td,.table-container table th{font-weight:400;padding:.5rem;text-align:left;vertical-align:middle}.table-container table td,.table-container table th{border:.0625rem solid var(--grey-3);border-bottom:.1875rem solid var(--grey-3)}.table-container table th{font-weight:700;padding-bottom:.625rem;text-align:center}.table-container table td{border-bottom-width:.0625rem}.table-container table td .highlight:last-child,.table-container table td p:last-child,.table-container table td pre:last-child{margin:0}.button{display:inline-block;cursor:pointer;touch-action:manipulation;text-align:center;text-decoration:none;vertical-align:middle;white-space:nowrap;border-radius:.3rem;border:.0625rem solid var(--grey-3);color:var(--grey-6);font-size:.875em;font-weight:400;line-height:1.5;background:0 0;margin-bottom:0;min-height:1em;padding:.5em 1.25em;-webkit-user-select:none;-moz-user-select:none;user-select:none;outline:0;will-change:auto;transition:all .2s ease-in-out 0s}.button+.button{margin-left:1.25em}.button:active,.button:hover{color:var(--grey-0);background-color:var(--primary-color);border-color:var(--primary-color)}.button:disabled{border-color:var(--grey-4);color:var(--grey-4);background-color:var(--grey-2);cursor:not-allowed}.button .ic{text-align:left;width:1.285714285714286em}.toggle{line-height:0;cursor:pointer}.toggle .line{background:var(--header-text-color);display:inline-block;height:.125rem;left:0;position:relative;border-radius:.0625rem;top:0;transition:all .4s;vertical-align:top;width:100%;box-shadow:0 0 .5rem rgba(0,0,0,.5)}.toggle .line:not(:first-child){margin-top:.1875rem}.toggle.toggle-arrow .toggle-line-first{left:50%;top:.125rem;transform:rotate(45deg);width:50%}.toggle.toggle-arrow .toggle-line-middle{left:.125rem;width:90%}.toggle.toggle-arrow .toggle-line-last{left:50%;top:-.125rem;transform:rotate(-45deg);width:50%}.toggle.close .line:first-child{transform:rotate(-45deg);top:.3125rem}.toggle.close .line:nth-child(2){opacity:0}.toggle.close .line:nth-child(3){transform:rotate(45deg);top:-.3125rem}.pagination .next,.pagination .page-number,.pagination .prev,.pagination .space{display:inline-block;margin:0 .5rem;padding:0 .75rem;position:relative;border-radius:.3125rem}@media (max-width:767px){.pagination .next,.pagination .page-number,.pagination .prev,.pagination .space{margin:0 .3125rem}}.pagination{width:100%;padding:1.25rem 3.125rem;text-align:center;display:inline-block;color:var(--grey-5)}@media (max-width:767px){.pagination{padding:1.25rem .625rem}}.pagination .inner{width:auto;border-radius:.9375rem}.pagination .next,.pagination .page-number,.pagination .prev{transition:all .2s ease-in-out 0s}.pagination .space{margin:0;padding:0}.pagination .prev{margin-left:0}.pagination .next{margin-right:0}#search-pagination .current .page-number:hover,.pagination .page-number.current:hover{box-shadow:0 0 .3125rem var(--primary-color)}.tip{position:fixed;background:var(--grey-9-a5);color:#fff;top:50%;left:50%;z-index:9999;padding:.625rem 1rem;border-radius:.625rem;transform:translate(-50%,-50%);text-align:center;font-size:.875em;-webkit-backdrop-filter:blur(0.625rem);backdrop-filter:blur(0.625rem)}.ribbon{display:inline-block;align-self:flex-start;position:relative;padding:0 1rem 0 2rem;border-radius:0 .3rem .3rem 0;background-image:linear-gradient(to right,var(--color-orange) 0,var(--color-pink) 100%);color:var(--grey-0)}.ribbon::after{content:"";position:absolute;top:100%;left:0;width:0;height:0;background-color:transparent;border-style:solid;border-width:0 1rem 1rem 0;border-color:transparent;border-right-color:var(--color-orange);filter:brightness(.9)}.ribbon a{display:block;margin:0;text-align:center;overflow:hidden;white-space:nowrap;text-overflow:ellipsis}.ribbon a:hover{color:currentColor}.divider{margin:1rem 0;line-height:1;height:0;font-weight:700;text-transform:uppercase;letter-spacing:.05rem;-webkit-user-select:none;-moz-user-select:none;user-select:none;color:var(--grey-4);display:table;white-space:nowrap;height:auto;line-height:1;text-align:center}.divider::after,.divider::before{content:'';display:table-cell;position:relative;top:50%;width:50%;background-repeat:no-repeat;background-image:url("")}.divider::before{background-position:right 1rem top 50%}.divider::after{background-position:left 1rem top 50%}#container{min-height:100%;min-width:100%;position:relative;display:flex;flex-direction:column}.inner{margin:0 auto;width:100%}main{background:linear-gradient(to top,var(--body-bg-shadow) 0,var(--grey-1) 20%) no-repeat bottom}main>.inner{width:calc(100% - .625rem);align-items:flex-start;display:flex;justify-content:space-between;flex-direction:row-reverse}@media (min-width:1200px){main>.inner{width:72.5rem}}@media (min-width:1600px){main>.inner{width:73%}}#main{background:linear-gradient(to top,var(--grey-0) 0,var(--grey-1) 20%) no-repeat top;box-shadow:0 1.25rem 1rem .3125rem var(--body-bg-shadow);width:calc(100% - 15.75rem);min-height:37.5rem}#main .cat{margin-top:10rem}#main .wrap{position:relative;padding:1.25rem}#main .wrap:first-child{margin-bottom:1.25rem}@media (max-width:991px){#main{width:100%}#main .wrap{padding:.625rem}}@media (max-width:767px){#main .wrap{padding:.5rem}}#header{margin:0 auto;position:relative;width:100%;height:50vh;text-shadow:0 .2rem .3rem rgba(0,0,0,.5);color:var(--header-text-color)}#header a:hover{color:currentColor}#imgs{display:block;position:fixed;top:0;left:0;width:100%;height:70vh;min-height:25rem;z-index:-9;background-color:#363636}#imgs img{width:100%;height:100%;position:absolute;top:0;left:0;-o-object-fit:cover;object-fit:cover}#imgs .item{width:100%;height:100%;position:absolute;top:0;left:0;background-size:cover;background-position:50% 50%;background-repeat:no-repeat;opacity:0;z-index:0;animation:imageAnimation 36s linear infinite 0s;backface-visibility:hidden;transform-style:preserve-3d}#imgs .item:nth-child(2){animation-delay:6s}#imgs .item:nth-child(3){animation-delay:12s}#imgs .item:nth-child(4){animation-delay:18s}#imgs .item:nth-child(5){animation-delay:24s}#imgs .item:nth-child(6){animation-delay:30s}#imgs .item:nth-child(7){animation-delay:36s}#imgs::before{content:'';display:block;position:absolute;top:0;left:0;width:100%;height:100%;background-color:rgba(0,0,0,.2);z-index:1;transition:all .2s ease-in-out 0s}@keyframes imageAnimation{0%{opacity:0;animation-timing-function:ease-in}2%{opacity:1}8%{opacity:1;transform:scale(1.05);animation-timing-function:ease-out}17%{opacity:1;transform:scale(1.1)}25%{opacity:0;transform:scale(1.1)}100%{opacity:0}}#nav{position:fixed;z-index:9;width:100%;height:3.125rem;transition:all .2s ease-in-out 0s}#nav.up{transform:translateY(0)}#nav.down{transform:translateY(-100%)}#nav.show{background:var(--nav-bg);box-shadow:.1rem .1rem .2rem var(--grey-9-a1);text-shadow:0 0 .0625rem var(--grey-9-a1);color:var(--text-color)}#nav.show .line{background:var(--text-color);box-shadow:0 0 .0625rem var(--grey-9-a1)}#nav.show .item.active>a,#nav.show .item.expand>a{color:var(--color-aqua);opacity:1}#nav.show .menu .submenu{background-color:var(--grey-1)}#nav.show .menu .submenu .item.active a{color:var(--grey-0);opacity:1}#nav .inner{height:100%;display:flex;width:calc(100% - .625rem);flex-wrap:nowrap}@media (min-width:1200px){#nav .inner{width:72.5rem}}@media (min-width:1600px){#nav .inner{width:73%}}#nav .toggle{display:none}@media (max-width:991px){#nav .toggle{display:flex;flex-direction:column;justify-content:center;align-items:center}}#nav .toggle .lines{padding:1.25rem;width:1.375rem;box-sizing:unset}#nav .menu{padding:.625rem 0;margin:0;width:100%}#nav .menu .item{display:inline-block;position:relative;padding:0 .625rem;letter-spacing:.0625rem;text-align:center}@media (max-width:767px){#nav .menu .item{display:none}#nav .menu .item.title{display:block}}#nav .menu .item .ic{margin-right:.5rem}#nav .menu .item:not(.title) a{display:block;font-size:1em}#nav .menu .item.active:not(.dropdown) a::before,#nav .menu .item:not(.dropdown):hover a::before{width:70%}#nav .menu .submenu{display:none;position:absolute;margin-top:.5rem;padding:0;width:-moz-max-content;width:max-content;background-color:var(--grey-9-a5);box-shadow:0 .3125rem 1.25rem -.25rem var(--grey-9-a1);border-radius:.625rem 0}#nav .menu .submenu::before{position:absolute;top:-1.25rem;left:0;width:100%;height:2.5rem;content:''}#nav .menu .submenu:hover{display:block}#nav .menu .submenu .item{display:block}#nav .menu .submenu .item:first-child{border-radius:.625rem 0 0 0}#nav .menu .submenu .item:last-child{border-radius:0 0 .625rem 0}#nav .menu .submenu .item a{display:inline-block;padding:.3rem .7rem;width:100%;text-shadow:none}#nav .menu .submenu .item a::before{content:none}#nav .menu .submenu .item:hover a{transform:translateX(.3rem)}#nav .menu .submenu .item.active a,#nav .menu .submenu .item:hover a{opacity:1}#nav .menu .item.dropdown>a::after{content:"";display:inline-block;margin-left:.3rem;vertical-align:middle;border:.3rem solid transparent;border-top-color:currentColor;border-bottom:0}#nav .menu .item.dropdown:hover .submenu{display:block}#nav .right{display:inline-flex;align-items:center;justify-content:center}#nav .right .item{padding:.625rem .5rem;cursor:pointer}#nav .right .i-sun{font-size:1.125em}#brand{position:fixed;padding:3rem 5rem 0;text-align:center;width:100%;height:50vh;min-height:10rem}#brand.affix{z-index:-1}#brand,#brand .pjax{display:flex;flex-direction:column;justify-content:center;align-items:center}#brand .artboard{font-family:'Fredericka the Great',Mulish,-apple-system,"PingFang SC","Microsoft YaHei",sans-serif;font-size:3.5em;line-height:1.2}#brand h1{font-size:2.5em;letter-spacing:.125rem}#brand .artboard+h1{margin:.625rem 0}#brand .sticky{font-size:.75em;display:inline-block;transform:rotate(30deg);backface-visibility:hidden}#brand .meta{display:flex;font-size:.875em;margin:0}#brand .meta .item+.item{margin-left:.625rem}@media (min-width:768px) and (max-width:991px){#brand .meta{font-size:.8125em}}@media (max-width:767px){#brand{padding:3rem .5rem 0}#brand h1{font-size:1.5em}#brand .meta{font-size:.75em}#brand .meta .text{display:none}}@media (max-width:413px){#brand .artboard{font-size:2.5em}}#tool{position:fixed;right:1rem;top:50vh;z-index:9}@media (max-width:991px){#tool{right:1rem;left:auto}}#tool .item{display:none;width:1.875rem;height:1.875rem;opacity:.5;cursor:pointer;align-items:center;justify-content:center;flex-direction:column;transition:all .2s ease-in-out 0s}#tool .item:hover{opacity:.9}#tool .player{display:inline-flex;flex-direction:row-reverse;font-size:1.5em;width:auto;height:auto}#tool .player>.btn{font-family:ic;font-weight:400;width:1.875rem;text-align:center}#tool .player>.btn+.btn{margin-right:.625rem}#tool .player .music{display:none}#tool .player-info{border-radius:.5rem;border:.0625rem solid var(--grey-2);box-shadow:0 .625rem 1.875rem -.9375rem var(--box-bg-shadow);background:var(--grey-1-a7);-webkit-backdrop-filter:blur(0.625rem);backdrop-filter:blur(0.625rem);position:fixed;display:none;bottom:0;overflow:hidden;right:3.5rem;width:50vw;z-index:9;max-width:25rem}@media (max-width:767px){#tool .player-info{right:2.5rem}}#tool .player-info .nav{font-size:.75em;height:2.1875rem}#tool .back-to-top span{font-size:.75em}#tool.affix{text-shadow:none;top:auto;box-shadow:0 0 .5rem rgba(0,0,0,.1);background:var(--grey-1-a3);border-radius:.3125rem;right:0;bottom:0}#tool.affix .item{display:flex;color:var(--primary-color)}#tool.affix .player{font-size:1em;flex-direction:column-reverse}#tool.affix .player>.btn+.btn{margin-right:0}#tool.affix .player .music{display:block}#tool.affix .player-info{right:2.5rem}@media (min-width:1200px){#tool.affix .player-info{right:3.5rem}}@media (max-width:991px){#tool.affix .player-info{right:2.5rem;left:auto}}#tool.affix .player-info.show{display:block}#tool.affix .back-to-top{align-items:center;justify-content:center;height:auto;padding:.3125rem 0 0}#tool.affix .contents{display:none}@media (max-width:991px){#tool.affix .contents{display:flex}}@media (min-width:1200px){#tool.affix{right:1rem;bottom:1rem}}@media (max-width:991px){#tool.affix{right:0;bottom:0;left:auto}}.waves{width:100%;height:15vh;margin-bottom:-.6875rem;min-height:3.125rem;max-height:9.375rem;position:relative}@media (max-width:767px){.waves{height:10vh}}.parallax>use{animation:wave 25s cubic-bezier(.55,.5,.45,.5) infinite}.parallax>use:first-child{animation-delay:-2s;animation-duration:7s;fill:var(--grey-1-a7)}.parallax>use:nth-child(2){animation-delay:-3s;animation-duration:10s;fill:var(--grey-1-a5)}.parallax>use:nth-child(3){animation-delay:-4s;animation-duration:13s;fill:var(--grey-1-a3)}.parallax>use:nth-child(4){animation-delay:-5s;animation-duration:20s;fill:var(--grey-1)}@keyframes wave{0%{transform:translate3d(-90px,0,0)}100%{transform:translate3d(85px,0,0)}}#sidebar{position:static;width:15rem;top:0;bottom:0;transition:all .2s ease-in-out 0s}@media (max-width:991px){#sidebar{display:none;position:fixed;right:0;background:var(--grey-1);box-shadow:0 .375rem .9375rem .3125rem rgba(0,0,0,.2);z-index:99}}#sidebar.affix>.inner{position:fixed;top:0}#sidebar.affix .panels{padding-top:3.625rem;height:100vh}#sidebar>.inner{position:relative;width:15rem;color:var(--grey-6);text-align:center;display:flex;justify-content:space-around;align-items:flex-start;flex-wrap:wrap;z-index:1}.panels{padding:4.6875rem 0 2rem;width:100%;overflow:hidden}.panels .inner{overflow-x:hidden;overflow-y:auto;-webkit-overflow-scrolling:touch;width:auto;height:100%}.panels .inner::-webkit-scrollbar{display:none}.panels .panel{display:none;padding:.875rem .9375rem 2rem}.dimmer{display:none}@media (max-width:991px){.dimmer{background:#000;height:100%;left:100%;opacity:0;top:0;width:100%;z-index:9;transition:opacity 1s}#sidebar.on+.dimmer{position:fixed;display:block;opacity:.3;transform:translateX(-100%)}}.overview .author .image{border:.0625rem solid var(--body-bg-shadow);display:block;margin:0 auto;max-width:10rem;padding:.125rem;box-shadow:0 0 1rem .625rem var(--body-bg-shadow);border-radius:50%}.overview .author .name{color:var(--grey-7);font-weight:400;margin:.3125rem 0 0;text-align:center}.overview .author .description{color:var(--grey-5);font-size:1em;margin-top:.3125rem;text-align:center}@keyframes shake{0%{transform:scale(1)}10%,20%{transform:scale(.9) rotate(3deg)}30%,50%,70%,90%{transform:scale(1.1) rotate(-3deg)}40%,60%,80%{transform:scale(1.1) rotate(3deg)}100%{transform:scale(1)}}.overview .menu{padding:1.25rem;margin:0;background-color:transparent}.overview .menu .item{border-radius:.9375rem;margin-bottom:.625rem;display:block;color:var(--grey-5);transition:all .2s ease-in-out 0s}.overview .menu .item a{color:inherit;display:block;line-height:3}.overview .menu .item .submenu{display:none;padding:0}.overview .menu .item:hover{background-color:rgba(0,0,0,.1);color:inherit}.overview .menu .item:hover .submenu{display:block}.overview .menu .item .ic{margin-right:.625rem}.overview .menu .item.active:hover{box-shadow:0 0 .75rem var(--color-pink);color:var(--grey-0)}.overview .menu .item.active .item{color:currentColor}.overview .menu .item.expand{background-color:rgba(0,0,0,.05)}.overview .menu .item.expand .submenu{display:block}.social{margin-top:.9375rem;text-align:center}.social .item{display:inline-block;width:1.875rem;height:1.875rem;line-height:1.875rem;text-align:center;position:relative;overflow:hidden;border-radius:38%}.social .item i{font-size:1.4em;vertical-align:middle;transform:scale(.8)}.social .item::before{top:90%;left:-110%;content:"";width:120%;height:120%;position:absolute;transform:rotate(45deg)}.social .item i,.social .item::before{transition:all .35s cubic-bezier(.31, -.105, .43, 1.59) 0s}.social .item:focus::before,.social .item:hover::before{top:-10%;left:-10%}.social .item.github::before{background-color:#191717}.social .item.github i{color:#191717}.social .item.telegram::before{background-color:#55acd5}.social .item.telegram i{color:#55acd5}.social .item.about::before{background-color:#3b5998}.social .item.about i{color:#3b5998}.social .item.music::before{background-color:#e60026}.social .item.music i{color:#e60026}.social .item.bilibili::before{background-color:#55acd5}.social .item.bilibili i{color:#55acd5}.social .item:focus i,.social .item:hover i{color:var(--grey-0);transform:scale(1)}#quick{display:none;align-items:center;flex-wrap:wrap;width:15rem;margin:0;padding:0;position:fixed;bottom:.125rem}#quick li{width:25%;min-height:1.875rem;transition:all .2s ease-in-out 0s}#quick li i{cursor:pointer}#quick li a{width:100%;display:block}#quick li:hover{color:var(--primary-color)}#quick li.percent{display:block;background:var(--primary-color);width:0;min-height:.125rem}#sidebar.affix #quick,#sidebar.on #quick{display:flex}#sidebar .tab{position:absolute;display:inline-flex;padding:1.875rem 0 .625rem;margin:0;min-height:1.875rem}#sidebar .tab .item{cursor:pointer;display:inline-flex;font-size:.875em;padding:.3125rem .9375rem;color:var(--grey-5);border-radius:.625rem;text-align:center;text-decoration:none;background-color:rgba(0,0,0,.08);transition:all .2s ease-out 0s}#sidebar .tab .item:nth-child(2){margin:auto .625rem}#sidebar .tab .item span{display:none;word-break:keep-all}#sidebar .tab .item.active span{display:inherit}#sidebar .tab .item.active:hover{box-shadow:0 0 .75rem var(--color-pink)}#sidebar .tab .item::before{font-family:ic;font-weight:400}#sidebar .tab .item.active::before{margin-right:.3125rem}#sidebar.affix .tab{padding-top:.625rem}.contents ol{padding:0 .125rem .3125rem .625rem;text-align:left}.contents ol>ol{padding-left:0}.contents .toc-item{font-size:1em;line-height:1.8;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.contents .toc-child{display:none}.contents .active>.toc-child{display:block}.contents .current>.toc-child{display:block}.contents .current>.toc-child>.toc-item{display:block}.contents .active>a{color:var(--primary-color)}.contents .current>a{color:var(--primary-color)}.contents .current>a:hover{color:var(--primary-color)}.related{font-size:.875em}.related ul{padding:0 .125rem .3125rem 1.25rem;text-align:left}.related ul li{position:relative;line-height:1.8;padding-bottom:.625rem}.related ul li a{text-overflow:ellipsis;white-space:nowrap;overflow:hidden;width:100%;display:inline-block}.related ul li.active a{color:var(--primary-color)}.related ul li::before{content:"";width:.5rem;height:.5rem;background:var(--primary-color);box-sizing:unset;left:-1.25rem;top:.3125rem;border-radius:100%;position:absolute;border:.1875rem solid var(--grey-1);z-index:1;transition:all .2s ease-in-out 0s}.related ul li:hover::before{background:var(--color-blue)}.related ul li:not(:last-child)::after{content:"";height:100%;width:.125rem;background:var(--color-red-a3);position:absolute;left:-.875rem;top:.5rem}.state{display:flex;justify-content:center;line-height:1.4;margin-top:.625rem;overflow:hidden;text-align:center;white-space:nowrap}.state .item{padding:0 .9375rem}.state .item:not(:first-child){border-left:.0625rem solid var(--grey-4)}.state .item a{border-bottom:none}.state .item .count{display:block;font-size:1.25em;font-weight:600;text-align:center}.state .item .name{color:inherit;font-size:.875em}#footer{color:var(--grey-5);font-size:.875em;background:var(--body-bg-shadow)}#footer .inner{margin:0 auto;width:calc(100% - .625rem);position:relative;padding-left:16.25rem}@media (min-width:1200px){#footer .inner{width:72.5rem}}@media (min-width:1600px){#footer .inner{width:73%}}@media (max-width:991px){#footer .inner{padding-left:0;padding-right:0;width:auto}}.status{width:100%;text-align:center;margin-top:2rem}.languages{display:inline-block;font-size:1.125em;position:relative}.languages .lang-select-label span{margin:0 .5rem}.languages .lang-select{height:100%;left:0;opacity:0;position:absolute;top:0;width:100%}.with-love{color:pink;display:inline-block;margin:0 .3125rem 0 .125rem}.powered-by,.theme-info{display:inline-block;margin-bottom:.625rem}@media (max-width:567px){.post.block{padding:initial!important}.md h1,.md h2,.md h3,.md h4,.md h5,.md h6{padding-left:1.25rem}.md .note h1,.md .note h2,.md .note h3,.md .note h4,.md .note h5,.md .note h6,.md .tabs .tab-content .tab-pane h1,.md .tabs .tab-content .tab-pane h2,.md .tabs .tab-content .tab-pane h3,.md .tabs .tab-content .tab-pane h4,.md .tabs .tab-content .tab-pane h5,.md .tabs .tab-content .tab-pane h6{margin:0 .3125rem}.md img,.md video{margin-bottom:.625rem!important}.md .tabs .tab-content .tab-pane{padding:.625rem .625rem 0 .625rem!important}.post .eof{margin:2.5rem auto 1.25rem!important}}.post.block{padding:0 2rem}.md{font-family:Mulish,-apple-system,"PingFang SC","Microsoft YaHei",sans-serif;overflow-wrap:break-word;word-wrap:break-word}.md .exturl .ic{font-size:.875em;margin-left:.25rem}.md .button{margin-top:2.5rem;text-align:center}.breadcrumb{display:inline-flex;font-size:.8125em;align-items:center;margin:1.25rem 0;flex-wrap:wrap}.breadcrumb .ic{margin:0 .125rem;color:var(--grey-4)}.breadcrumb .ic:first-child{margin-left:0;margin-right:.3125rem}.breadcrumb span{white-space:pre}.breadcrumb span.current{background-color:var(--color-red-a1);border-radius:.625rem;padding:0 .625rem;transition:all .2s ease-in-out 0s}.breadcrumb span.current span{white-space:normal}.breadcrumb span.current a{color:var(--primary-color)}.breadcrumb span.current:hover{background-color:var(--color-red-a3)}.post header{font-size:1.125em;margin-bottom:.625rem}.post header .title{font-size:1.5em;margin:initial;text-align:center;overflow-wrap:break-word;word-wrap:break-word;padding-bottom:.625rem}.post header .link{display:inline-block;position:relative;vertical-align:top}.post header .link .i-link-alt{font-size:.875em;margin-left:.3125rem}.post header .meta{text-align:center;border-top:.0625rem dashed var(--grey-9-a1);font-family:Mulish,-apple-system,"PingFang SC","Microsoft YaHei",sans-serif}.post .meta{color:var(--grey-5);font-size:.75em;text-align:right}.post .meta .item{display:inline-block;margin-right:.625rem}.post .meta .icon{margin-right:.1875rem}@media (max-width:991px){.post .meta .icon{display:inline-block}}.post-nav{display:flex;margin-bottom:2.5rem}@media (max-width:767px){.post-nav{flex-direction:column}}.post-nav .item{width:50%}@media (max-width:767px){.post-nav .item{width:100%}}.post-nav .item a{display:flex;flex-direction:column;height:100%;color:var(--header-text-color);padding:1.25rem 2.5rem;background-size:cover;position:relative}.post-nav .item a::before{content:"";position:absolute;width:100%;height:100%;background:linear-gradient(135deg,#434343,#000);opacity:.5;transition:all .2s ease-in-out 0s;z-index:-1;top:0;left:0}.post-nav .item a:hover::before{opacity:.4}.post-nav .item span{font-size:.8125em}.post-nav .item.left h3,.post-nav .item.left span{align-self:flex-start}.post-nav .item.right h3,.post-nav .item.right span{align-self:flex-end;text-align:right}.md .links:last-child,.md .tabs:last-child,.md blockquote:last-child,.md img:last-child,.md p:last-child,.md pre:last-child,.md table:last-child{margin-bottom:0}.md dd ol,.md dd ul,.md ol ol,.md ol ul,.md ul ol,.md ul ul{margin-top:0}.md a{color:var(--primary-color)}.md a:hover{color:var(--color-blue)}.md h1{font-size:1.5em}.md h2{font-size:1.375em}.md h3{font-size:1.25em}.md h4{font-size:1.125em}.md h5{font-size:1em}.md h6{font-size:.875em}.md h1,.md h2,.md h3,.md h4,.md h5,.md h6{position:relative;padding-top:.625rem}.md h1:hover .anchor::after,.md h1:hover .anchor::before,.md h2:hover .anchor::after,.md h2:hover .anchor::before,.md h3:hover .anchor::after,.md h3:hover .anchor::before,.md h4:hover .anchor::after,.md h4:hover .anchor::before,.md h5:hover .anchor::after,.md h5:hover .anchor::before,.md h6:hover .anchor::after,.md h6:hover .anchor::before{color:var(--primary-color)}.md h1::after{content:"";display:block;box-sizing:unset;width:100%;height:.0625rem;background:var(--grey-3);padding-right:1.25rem;margin-left:-1.25rem;margin-top:.3125rem}.md .anchor{border-bottom-style:none;color:var(--grey-4);float:right;margin-left:.625rem}.md .anchor:hover{color:inherit}.md .anchor::after,.md .anchor::before{color:var(--grey-4);position:absolute;font-weight:400;transition:all .2s ease-out 0s}.md .active .anchor::after,.md .active .anchor::before{color:var(--primary-color)}.md p{margin:0 0 .8rem}.md blockquote{font-size:90%;background-color:var(--grey-2);margin:1.25rem 0;border-radius:.1875rem}.md blockquote ul{margin:.625rem 0!important}.md blockquote ul>li::before{width:.375rem!important;height:.375rem!important;top:.6875rem!important}.md blockquote ol:last-child,.md blockquote p:last-child,.md blockquote ul:last-child{margin-bottom:0!important}.md>blockquote{border-left-color:var(--primary-color)}.md iframe{margin-bottom:1.25rem}.md .image-info{display:block;text-align:center;font-size:.8125em;color:var(--grey-4)}.md .video-container{height:0;margin-bottom:1.25rem;overflow:hidden;padding-top:75%;position:relative;width:100%}.md .video-container embed,.md .video-container iframe,.md .video-container object{height:100%;left:0;margin:0;position:absolute;top:0;width:100%}.md .kbd,.md kbd{background-color:var(--grey-1);background-image:linear-gradient(var(--grey-2),var(--grey-0),var(--grey-2));border:.0625rem solid var(--grey-4);border-radius:.2rem;box-shadow:.1rem .1rem .2rem var(--grey-9-a1);font-family:inherit;padding:.1rem .3rem;white-space:nowrap}.md mark{background-color:#dbfdad}.md ins{--line-color:var(--note-hover, var(--primary-color));text-decoration:none;border-bottom:.125rem solid var(--line-color)}.md ins.wavy{-webkit-text-decoration-style:wavy;text-decoration-style:wavy;-webkit-text-decoration-line:underline;text-decoration-line:underline;-webkit-text-decoration-color:var(--line-color);text-decoration-color:var(--line-color);border-bottom:none}.md ins.dot{border-bottom:.2rem dotted var(--line-color)}.md s{color:var(--grey-5);-webkit-text-decoration-color:var(--note-hover,var(--grey-5));text-decoration-color:var(--note-hover,var(--grey-5))}.md ruby{padding:0 .3125rem}.md .katex-display{overflow-x:scroll;overflow-y:hidden}.md .spoiler:not(.bulr){background-color:var(--text-color);color:var(--text-color);text-shadow:none;transition:color .3s;padding:0 .1875rem}.md .spoiler:not(.bulr):hover{color:#fff}.md .bulr{text-shadow:rgba(0,0,0,.7) 0 0 .625rem;color:transparent}.md .rainbow{background-image:linear-gradient(to left,#ff4500,orange,gold,#90ee90,#0ff,#1e90ff,#9370db,#ff69b4,#ff4500);background-size:110vw;-webkit-background-clip:text;background-clip:text;color:transparent;animation:rainbow 60s linear infinite}.article .md .anchor::after,.article .md .anchor::before{content:"H";font-family:Mulish,-apple-system,"PingFang SC","Microsoft YaHei",sans-serif;left:-1.875rem;top:1rem;width:1.25rem;height:1.5625rem;text-align:right;visibility:visible;font-size:80%}@media (max-width:567px){.article .md .anchor::after,.article .md .anchor::before{left:-.625rem}}.article .md .anchor::after{font-size:50%;left:-1.375rem;line-height:3}@media (max-width:567px){.article .md .anchor::after{left:-.1875rem}}.article .md h1 .anchor::after{content:" 1 "}.article .md h2 .anchor::after{content:" 2 "}.article .md h3 .anchor::after{content:" 3 "}.article .md h4 .anchor::after{content:" 4 "}.article .md h5 .anchor::after{content:" 5 "}.article .md h6 .anchor::after{content:" 6 "}@keyframes rainbow{to{background-position:-2000vw}}.rtl.body a,.rtl.body h1,.rtl.body h2,.rtl.body h3,.rtl.body h4,.rtl.body h5,.rtl.body h6,.rtl.body li,.rtl.body ol,.rtl.body p,.rtl.body ul{direction:rtl;font-family:UKIJ Ekran}.rtl.title{font-family:UKIJ Ekran}.post footer::before{content:"";width:100%;height:.0625rem;background:var(--grey-3);display:block;margin:1.25rem auto 0}.post .tags{text-align:left;margin-top:.625rem;font-size:.75em}.post .tags a{display:inline-block;position:relative;padding:0 .3125rem;border-radius:.3125rem;background:var(--note-bg);color:var(--note-text)}.post .tags a:not(:last-child){margin-right:.625rem}.post .tags a:before{content:'';position:absolute;bottom:0;height:100%;width:0;right:0;background:var(--color-red-a1);border-radius:.25rem;transition:all .2s ease-in-out 0s}.post .tags a:hover{color:var(--primary-color)}.post .tags a:hover:before{width:104%;left:-2%}.reward{margin:1.25rem auto;padding:.625rem 0;text-align:center}.reward button{background:var(--primary-color);border:0;border-radius:.3125rem;color:var(--grey-0);cursor:pointer;line-height:2;outline:0;padding:0 .9375rem;vertical-align:text-top}.reward button:hover{background:var(--primary-color)}.reward button i{margin-right:.3125rem}.reward p{font-size:.8125em;color:var(--grey-5);margin:0}#qr{padding-top:1.25rem;display:none}#qr a{border:0}#qr img{display:inline-block;max-width:100%}#qr p{text-align:center}#copyright{background:var(--grey-2);padding:1rem 2rem;position:relative;font-size:.75em;border-radius:.625rem;color:var(--grey-6)}#copyright li::before{font-family:ic;font-weight:400;color:var(--grey-5);margin-right:.3125rem;font-size:1.1rem;line-height:.75rem;vertical-align:-.0667rem}.cards .item,.segments>.item{position:relative;color:inherit;width:calc(50% - 2rem);min-width:calc(50% - 2rem);height:14rem;margin:1rem;opacity:0}.cards .item.show,.segments>.item.show{opacity:1}@media (max-width:767px){.cards .item,.segments>.item{width:calc(100% - 1rem)!important;min-width:calc(100% - 1rem)!important;margin:1rem .5rem!important}}.index.wrap .btn{position:absolute;bottom:0;right:0;padding:.3rem 1rem;border-radius:1rem 0;color:var(--grey-0);background-image:linear-gradient(to right,var(--color-pink) 0,var(--color-orange) 100%)}.index.wrap .btn::before{position:absolute;display:block;content:'';height:calc(100% - 1rem);width:calc(100% - 1rem);border-radius:5rem;left:.5rem;top:.8rem;box-shadow:0 0 .6rem .6rem var(--color-pink-a3);background-color:var(--color-pink-a3)}.index.wrap .btn:hover{transform:translateZ(2.5rem)}.index.wrap .btn:hover::before{transform:translateZ(-2.5rem)}.index.wrap .active .btn{transform-style:preserve-3d;transform:translateZ(2rem);backface-visibility:hidden}.index.wrap .active .btn::before{transform-style:preserve-3d;transform:translateZ(-2rem);backface-visibility:hidden}.index.wrap .meta{font-size:.8125em;color:var(--grey-5)}.index.wrap .meta .ic{margin-right:.0625rem}.index.wrap .meta .item+.item{margin-left:.625rem}.index.wrap .meta.footer{position:absolute;bottom:.5rem;max-width:calc(100% - 7rem);white-space:nowrap;overflow:hidden;text-overflow:ellipsis;justify-content:flex-start}.index.wrap .meta.footer span{margin-right:.5rem}.cards{display:flex;margin:0 auto;align-items:center;flex-wrap:wrap;justify-content:space-between}.cards.hide .item{display:none}.cards .item{perspective:62.5rem}.cards .item.show{display:block!important}.cards .item .cover,.cards .item .info{position:absolute;display:flex;flex-direction:column;justify-content:center;align-items:center;border-radius:.375rem;height:100%;width:100%;backface-visibility:hidden;transform-style:preserve-3d;transition:ease-in-out .6s}.cards .item .cover{background-position:center;background-size:cover;padding:.5rem 1rem;font-size:1em;color:var(--header-text-color);overflow:hidden;transform:rotateY(0)}.cards .item .cover .title{margin:0;white-space:normal;text-align:center}.cards .item .cover::before{position:absolute;display:block;content:'';top:0;left:0;right:0;bottom:0;background-image:linear-gradient(135deg,#434343 0,#000 100%);opacity:.25;z-index:-1}.cards .item .cover span{font-size:.75em;position:absolute;right:.9375rem;top:.625rem;padding:0 .3125rem;border-radius:.3125rem;box-shadow:0 0 .3125rem .0625rem rgba(0,0,0,.6);background:rgba(0,0,0,.5)}.cards .item .info{background-color:var(--grey-0);transform:rotateY(-180deg);padding:1rem 1.5rem 4rem;justify-content:space-between}@media (max-width:767px){.cards .item .info{padding:1rem 1rem 4rem}}.cards .item .info .ribbon{left:-2.5rem;margin-bottom:.8rem;max-width:calc(100% + 2rem)}@media (max-width:767px){.cards .item .info .ribbon{left:-2rem}}.cards .item:nth-child(2n) .info{transform:rotateY(180deg)}.cards .item.active .cover{transform:rotateY(180deg)}.cards .item.active .info{transform:rotateY(0);box-shadow:0 0 2rem var(--box-bg-shadow)}.cards .item:nth-child(2n).active .cover{transform:rotateY(-180deg)}.cards .item .title .ic{font-size:80%}.cards .item ul.posts{display:flex;flex-wrap:wrap;justify-content:space-between;align-content:baseline;min-height:5rem;overflow:hidden}.cards .item ul.posts li{width:45%;overflow:hidden;white-space:nowrap;text-overflow:ellipsis;color:var(--primary-color)}.segments{display:flex;flex-direction:column;justify-content:center;align-items:center}.segments>.item{display:flex;border-radius:.625rem;width:calc(100% - 2rem);min-width:calc(100% - 2rem);border-radius:.5rem;box-shadow:0 .625rem 1.875rem -.9375rem var(--box-bg-shadow);transition:all .2s ease-in-out 0s}@media (max-width:767px){.segments>.item{flex-direction:column;height:-moz-fit-content;height:fit-content;max-height:-moz-fit-content;max-height:fit-content}}.segments>.item:hover{box-shadow:0 0 2rem var(--box-bg-shadow)}.segments>.item:hover .cover img{transform:scale(1.05) rotate(1deg)}.segments .cover{width:50%;margin-right:1.5rem;-webkit-clip-path:polygon(0 0,92% 0%,100% 100%,0% 100%);clip-path:polygon(0 0,92% 0%,100% 100%,0% 100%);border-radius:.625rem 0 0 .625rem;overflow:hidden}@media (max-width:767px){.segments .cover{width:100%;height:14rem;margin:auto;-webkit-clip-path:polygon(0 0,100% 0,100% 92%,0 100%);clip-path:polygon(0 0,100% 0,100% 92%,0 100%);border-radius:.625rem .625rem 0 0}}.segments .cover img{-o-object-fit:cover;object-fit:cover;width:100%;height:100%;transition:all .2s ease-in-out 0s}.segments .info{position:relative;width:50%;padding:1rem 1.5rem 3rem 0;perspective:62.5rem}@media (max-width:767px){.segments .info{width:100%;height:14rem;padding:0 1rem 3rem}}.segments .info .meta{display:flex;justify-content:flex-end;margin:0}@media (max-width:767px){.segments .info .meta .item:not(:first-child){display:none}}.segments .info h3{text-overflow:ellipsis;overflow:hidden;white-space:nowrap;margin:.625rem 0;color:var(--primary-color)}.segments .info .excerpt{overflow:hidden;font-size:.875em;max-height:5rem;display:-webkit-box;-webkit-box-orient:vertical;-webkit-line-clamp:3;text-overflow:ellipsis}.segments>.item:nth-child(2n){flex-direction:row-reverse}@media (max-width:767px){.segments>.item:nth-child(2n){flex-direction:column}}.segments>.item:nth-child(2n) .cover{margin-right:auto;margin-left:1.5rem;-webkit-clip-path:polygon(0 0%,100% 0%,100% 100%,8% 100%);clip-path:polygon(0 0%,100% 0%,100% 100%,8% 100%);border-radius:0 .625rem .625rem 0}@media (max-width:767px){.segments>.item:nth-child(2n) .cover{width:100%;margin:auto;-webkit-clip-path:polygon(0 0,100% 0,100% 100%,0 92%);clip-path:polygon(0 0,100% 0,100% 100%,0 92%);border-radius:.625rem .625rem 0 0}}.segments>.item:nth-child(2n) .info{padding:1rem 0 3rem 1.5rem}@media (max-width:767px){.segments>.item:nth-child(2n) .info{padding:0 1rem 3rem}}.segments>.item:nth-child(2n) .info .meta{justify-content:flex-start}.segments>.item:nth-child(2n) .btn{left:0;right:auto;border-radius:0 1rem;background-image:linear-gradient(to right,var(--color-orange) 0,var(--color-pink) 100%)}.segments>.item:nth-child(2n) .meta.footer{right:.5rem;justify-content:flex-start}.segments>.item:nth-child(2n):hover .cover img{transform:scale(1.05) rotate(-1deg)}.collapse small{color:var(--grey-4);margin:auto .3125rem}.collapse .item{position:relative;padding:1.25rem 1.875rem;margin:0}.collapse .item::before{content:"";position:absolute;z-index:1;transition:all .2s ease-in-out 0s;box-sizing:unset;top:1.9rem;left:0;width:.6rem;height:.6rem;border:.15rem solid var(--primary-color);border-radius:50%;background:var(--grey-1)}.collapse .item:not(:last-child):not(.title)::after{content:"";position:absolute;top:1.9rem;bottom:-1.9rem;left:.35rem;border-left:.2rem solid var(--color-red-a3)}.collapse .item:hover::before{border-color:var(--color-blue)}.collapse .item.header::after{border-left-style:dashed!important}.collapse .item.header .cheers{display:block}.collapse .item.section::before{width:.4rem;height:.4rem;margin-left:.1rem}.collapse .item.header a,.collapse .item.section a,.collapse .item.title a{border-bottom:.0625rem dashed var(--grey-4)}.collapse .item.header:hover a,.collapse .item.section:hover a,.collapse .item.title:hover a{border-bottom-color:var(--color-blue)}.collapse .item.normal{display:flex;flex-wrap:wrap;align-items:center;padding:.625rem 1.875rem}.collapse .item.normal::before{top:1.4rem;width:.2rem;height:.2rem;margin-left:.2rem;background:var(--primary-color)}.collapse .item.normal:hover::before{background:var(--color-blue)}.collapse .item.normal .meta{display:inline;font-size:.75em;margin-right:.625rem}.collapse .item.normal .meta time{color:var(--grey-4)}.collapse .item.normal .title{display:inline}.collapse .item.normal .title a{color:var(--primary-color)}.collapse .item.normal .title a:hover{color:var(--color-blue)}.collapse .item.normal .title .i-link-alt{font-size:.875em;margin-left:.3125rem}.tag.cloud{text-align:center}.tag.cloud a{display:inline-block;margin:.625rem}.tag.cloud a:hover{color:var(--primary-color)!important}.page .notfound{width:18.75rem;height:22.625rem;background:url("../images/404.png") no-repeat center bottom;text-align:center;margin:6.25rem auto}.page .anchor::before{font-family:ic;font-weight:400;color:var(--grey-4);position:absolute;left:-1.5625rem;margin-top:.25rem;visibility:visible}@media (max-width:567px){.page .anchor::before{left:-.1875rem}}.page .body h1 .anchor::before{font-size:.875em;margin-top:.15rem;color:pink;left:-1.875rem}@media (max-width:567px){.page .body h1 .anchor::before{left:-.3125rem}}.page .body h2 .anchor::before{font-size:.8125em;color:var(--color-pink)}.page .body h3 .anchor::before{font-size:.75em;color:var(--color-orange)}.pace{pointer-events:none;-webkit-user-select:none;-moz-user-select:none;user-select:none}.pace-inactive{display:none}.pace .pace-progress{background:var(--primary-color);position:fixed;z-index:2000;top:0;right:100%;width:100%;height:.125rem}#loading{background-color:var(--grey-1)}.cat{position:relative;display:block;width:15em;height:100%;font-size:10px;margin:auto;animation:2.74s linear infinite loading-cat}.cat *{box-sizing:content-box}.cat .body,.cat .foot,.cat .head,.cat .paw{position:absolute;top:0;right:0;bottom:0;left:0;margin:auto;border-radius:50%;width:15em;height:15em}.cat .body{background-image:radial-gradient(transparent 0,transparent 35%,#383c4b 35%,#383c4b 39%,#eda65d 39%,#eda65d 46%,#f2c089 46%,#f2c089 60%,#eda65d 60%,#eda65d 67%,#383c4b 67%,#383c4b 100%)}.cat .foot:before,.cat .head:before{background-image:radial-gradient(transparent 0,transparent 35%,#383c4b 35%,#383c4b 39%,#eda65d 39%,#eda65d 67%,#383c4b 67%,#383c4b 100%)}.cat .head:before{content:'';width:100%;height:100%;position:absolute;border-radius:50%;clip-path:polygon(100% 20%,50% 50%,70% -10%);-webkit-clip-path:polygon(100% 20%,50% 50%,70% -10%)}.cat .head:after{content:'';width:4.125em;height:2.5em;position:absolute;top:.8125em;right:3.9375em;background-image:linear-gradient(var(--grey-1) 65%,transparent 65%),radial-gradient(var(--grey-1) 51%,#383c4b 55%,#383c4b 68%,transparent 70%);transform:rotate(-66deg)}.cat .head .face{width:5em;height:3.75em;left:9.0625em;top:1.8125em;position:absolute;transform:rotate(-47deg);background:radial-gradient(circle,#f2c089 0,#f2c089 23%,transparent 23%) -.1875em 1.0625em no-repeat,radial-gradient(circle,#383c4b 0,#383c4b 6%,transparent 6%) .75em -.75em no-repeat,radial-gradient(circle,#383c4b 0,#383c4b 6%,transparent 6%) -.75em -.75em no-repeat,radial-gradient(#eda65d 0,#eda65d 15%,transparent 15%) 0 -.6875em no-repeat,radial-gradient(circle,transparent 5%,#383c4b 5%,#383c4b 10%,transparent 10%) -.1875em -.3125em no-repeat,radial-gradient(circle,transparent 5%,#383c4b 5%,#383c4b 10%,transparent 10%) .1875em -.3125em no-repeat,radial-gradient(circle,#eda65d 45%,transparent 45%) 0 -.1875em,linear-gradient(transparent 35%,#383c4b 35%,#383c4b 41%,transparent 41%,transparent 44%,#383c4b 44%,#383c4b 50%,transparent 50%,transparent 53%,#383c4b 53%,#383c4b 59%,transparent 59%)}.cat .foot:after,.cat .foot:before{content:'';width:100%;height:100%;position:absolute}.cat .foot:before{border-radius:50%;clip-path:polygon(50% 50%,0% 50%,0% 25%);-webkit-clip-path:polygon(50% 50%,0% 50%,0% 25%)}.cat .foot .tummy-end{width:1.5em;height:1.5em;position:absolute;border-radius:50%;background-color:#f2c089;left:1.1875em;top:6.5625em}.cat .foot .bottom{width:2.1875em;height:.9375em;position:absolute;top:4.875em;left:.75em;border:.375em solid #383c4b;border-bottom:0;border-radius:50%;transform:rotate(21deg);background:#eda65d}.cat .foot:after,.cat .hands,.cat .legs{width:.625em;height:1.5625em;position:absolute;border:.375em solid #383c4b;background-color:#eda65d}.cat .hands{border-top:0;border-radius:0 0 .75em .75em}.cat .hands.left{top:4.3em;left:13.1875em;transform:rotate(-20deg)}.cat .hands.right{top:5.125em;left:10.975em;transform:rotate(-25deg)}.cat .legs{border-bottom:0;border-radius:.75em .75em 0 0}.cat .legs.left{top:4.0625em;left:3.125em;transform:rotate(25deg)}.cat .legs.right{top:3.3125em;left:.75em;transform:rotate(22deg)}.cat .foot:after{width:.9em;height:2.5em;top:2.625em;left:2.5em;z-index:-1;transform:rotate(25deg);background-color:#c6823b;border-bottom:0;border-radius:.75em .75em 0 0}.cat .body{animation:2.74s linear infinite body}.cat .foot{animation:2.74s linear infinite foot}.cat:hover{animation-play-state:paused}.cat:hover .body,.cat:hover .foot{animation-play-state:paused}.cat:active{animation-play-state:running}.cat:active .body,.cat:active .foot{animation-play-state:running}@keyframes body{0%{-webkit-clip-path:polygon(50% 50%,0% 50%,0% 100%,100% 100%,100% 20%);clip-path:polygon(50% 50%,0% 50%,0% 100%,100% 100%,100% 20%)}10%{-webkit-clip-path:polygon(50% 50%,30% 120%,50% 100%,100% 100%,100% 20%);clip-path:polygon(50% 50%,30% 120%,50% 100%,100% 100%,100% 20%)}20%{-webkit-clip-path:polygon(50% 50%,100% 90%,120% 90%,100% 100%,100% 20%);clip-path:polygon(50% 50%,100% 90%,120% 90%,100% 100%,100% 20%)}40%{-webkit-clip-path:polygon(50% 50%,100% 45%,120% 45%,120% 50%,100% 20%);clip-path:polygon(50% 50%,100% 45%,120% 45%,120% 50%,100% 20%)}50%{-webkit-clip-path:polygon(50% 50%,100% 45%,120% 45%,120% 50%,100% 20%);clip-path:polygon(50% 50%,100% 45%,120% 45%,120% 50%,100% 20%)}65%{-webkit-clip-path:polygon(50% 50%,100% 65%,120% 65%,120% 50%,100% 20%);clip-path:polygon(50% 50%,100% 65%,120% 65%,120% 50%,100% 20%)}80%{-webkit-clip-path:polygon(50% 50%,75% 130%,120% 65%,120% 50%,100% 20%);clip-path:polygon(50% 50%,75% 130%,120% 65%,120% 50%,100% 20%)}90%{-webkit-clip-path:polygon(50% 50%,-20% 110%,50% 120%,100% 100%,100% 20%);clip-path:polygon(50% 50%,-20% 110%,50% 120%,100% 100%,100% 20%)}100%{-webkit-clip-path:polygon(50% 50%,0% 50%,0% 100%,100% 100%,100% 20%);clip-path:polygon(50% 50%,0% 50%,0% 100%,100% 100%,100% 20%)}}@keyframes loading-cat{0%{transform:rotate(0)}10%{transform:rotate(-80deg)}20%{transform:rotate(-180deg)}40%{transform:rotate(-245deg)}50%{transform:rotate(-250deg)}68%{transform:rotate(-300deg)}90%{transform:rotate(-560deg)}100%{transform:rotate(-720deg)}}@keyframes foot{0%{transform:rotate(-10deg)}10%{transform:rotate(-100deg)}20%{transform:rotate(-145deg)}35%{transform:rotate(-190deg)}50%{transform:rotate(-195deg)}70%{transform:rotate(-165deg)}100%{transform:rotate(-10deg)}}#search{position:fixed;background:var(--nav-bg);left:0;top:0;width:100%;height:100%;padding:1.25rem;z-index:999;display:none}#search>.inner{border-radius:0;height:100%;margin:0 auto;width:43.75rem;text-shadow:none}@media (max-width:767px){#search>.inner{width:100%}}#search>.inner .close-btn,#search>.inner .icon{color:var(--grey-5);font-size:1.125rem;padding:0 .625rem}#search>.inner .close-btn{cursor:pointer}#search>.inner .close-btn:hover i{color:var(--grey-7)}#search>.inner .header{display:flex;background:var(--grey-1-a5);border-radius:3rem;padding:.5rem 1.5rem;margin-bottom:1.25rem;border:.0625rem solid var(--grey-5);font-size:1.125em;align-items:center}#search>.inner .search-input-container{flex-grow:1}#search>.inner .search-input-container form{padding:.125rem}#search>.inner .search-input{background:0 0;border:0;outline:0;width:100%}#search>.inner .search-input::-webkit-search-cancel-button{display:none}#search .results{height:calc(100% - 6.25rem);padding:1.875rem 1.875rem .3125rem;border-radius:.3125rem;background:var(--grey-1-a7) url("../images/search.png") no-repeat bottom right;color:var(--text-color)}#search .results .inner{position:relative;height:100%;overflow:hidden}#search .results hr{margin:.625rem 0}.algolia-powered{float:right;background:url("../images/algolia_logo.svg") no-repeat;display:inline-block;height:1.125rem;width:4.25rem;margin:.5rem auto}#search-hits{overflow-y:scroll;height:calc(100% - 8.125rem)}#search-hits ol{padding:0}#search-hits .item{margin:.9375rem 0}#search-hits .item a{border-bottom:.0625rem dashed var(--grey-4);display:block;transition:all .2s ease-in-out 0s}#search-hits .item span{font-size:70%;display:block}#search-hits .item span i{color:var(--grey-4);margin:0 .3125rem}#search-pagination ul{padding:0;margin:1.25rem 0}#search-pagination .pagination{opacity:1;padding:0}#search-pagination .pagination-item{display:inline-block}#search-pagination .page-number{transition:all .2s ease-in-out 0s}#search-pagination .current .page-number{cursor:default}#search-pagination .disabled-item{color:var(--grey-4);cursor:default}#search-pagination .disabled-item .page-number:hover{color:var(--grey-4);background:0 0;box-shadow:none}#neko{display:none;background:linear-gradient(to top,#fddb92 0,#d1fdff 80%)}#neko .planet{position:fixed;left:-50%;top:-50%;width:200%;height:200%;animation:rotate 2s cubic-bezier(.7,0,0,1);transform-origin:center bottom}#neko:before{content:"";position:absolute;left:0;right:0;top:0;bottom:0;opacity:0;background:linear-gradient(to top,#30cfd0 0,#330867 100%);transition:2s ease all}#neko .moon,#neko .sun{position:absolute;border-radius:100%;left:55%;top:32%}#neko .sun{height:40px;width:40px;background:#ffee94;box-shadow:0 0 40px #ffee94;opacity:1}#neko .moon{height:24px;width:24px;background:#eee;box-shadow:0 0 20px #fff;opacity:0}#neko .body{display:block;position:absolute;bottom:-20px;height:140px;width:135px;left:50%;margin-left:-100px;background:#777;transition:all .25s ease-in-out;animation:slideUpBigIn 1s}#neko .body:after,#neko .body:before{position:absolute;content:"";width:0;height:0;border-bottom:20px solid #777;top:-20px;transition:all .25s ease-in-out}#neko .body:before{border-left:0 solid transparent;border-right:30px solid transparent;left:0}#neko .body:after{border-right:0px solid transparent;border-left:30px solid transparent;right:0}#neko .body .eyes{display:block;position:absolute;background:#ffee94;height:40px;width:40px;border-radius:100%;bottom:80px}#neko .body .eyes.left{left:12px}#neko .body .eyes.right{right:12px}#neko .body .eyes .pupil,#neko .body .nose{display:block;position:relative;background:#ffb399;border-radius:100%;margin:0 auto}#neko .body .eyes .pupil{height:100%;width:5px;transition:width 1s .5s ease-in-out}#neko .body .nose{top:45px;height:10px;width:10px}#neko.dark:before{opacity:1}#neko.dark .sun{opacity:0}#neko.dark .moon{opacity:1}#neko.dark .body{background:#444}#neko.dark .body:before{border-bottom:20px solid #444}#neko.dark .body:after{border-bottom:20px solid #444}#neko.dark .body .eyes .pupil{height:90%;width:34px;margin:5% auto}.widgets{display:flex;z-index:1;background:var(--body-bg-shadow);justify-content:space-around}.widgets>div{width:calc(50% - 2rem);padding:1rem}@media (max-width:767px){.widgets{flex-direction:column-reverse}.widgets>div{width:calc(100% - 1rem)!important}}.widgets ul{counter-reset:counter}.widgets .item{padding:.5rem 0 .5rem 2rem;border-bottom:.0625rem dashed var(--grey-4);position:relative}.widgets .item::before{counter-increment:counter;content:counter(counter);position:absolute;left:0;font-size:1.5em;color:var(--grey-4);line-height:1.2;text-align:right;width:1em}.widgets .item .breadcrumb,.widgets .item span{display:block;text-overflow:ellipsis;overflow:hidden;max-height:2rem;white-space:nowrap}.widgets .item .breadcrumb{margin:0;display:flex;max-height:1.2rem}.katex,.katex-display{-webkit-user-select:all;-moz-user-select:all;user-select:all}.medium-zoom-overlay{z-index:9998}.medium-zoom-image--opened{z-index:9999}.operation{color:var(--grey-5);cursor:pointer;position:absolute;padding:.125rem .375rem;right:.125rem;top:.45rem;font-size:.8125em}.operation span{transition:all .2s ease-in-out 0s;margin:0 .3125rem}.operation span:hover{color:var(--grey-7)}.highlight{background:var(--grey-2);color:var(--grey-7);line-height:1.6;margin:1.25rem auto}.code-container,code,pre{font-family:Inconsolata,consolas,Menlo,-apple-system,"PingFang SC","Microsoft YaHei";font-size:1em}:not(td)>pre{background:var(--grey-1);border:.0625rem solid var(--grey-3);border-radius:.3125rem;padding:1rem;margin-bottom:.8rem;line-break:anywhere;white-space:break-spaces}:not(td)>pre code{background:0 0;color:currentColor}:not(pre)>code{color:var(--primary-color);border-radius:.3rem;border:.0625rem solid rgba(0,0,0,.1);background-color:var(--grey-0);padding:.2rem .3rem;overflow-wrap:break-word;word-wrap:break-word}.highlight{position:relative;overflow:hidden;border-radius:.5rem;box-shadow:0 .3125rem .625rem -.125rem var(--grey-9-a1)}.highlight ::-moz-selection{background:var(--grey-4)}.highlight ::selection{background:var(--grey-4)}.highlight::before{content:" ";position:absolute;border-radius:50%;background:#fc625d;width:.75rem;height:.75rem;left:.75rem;top:.8125rem;box-shadow:1.25rem 0 #fdbc40,2.5rem 0 #35cd4b}.highlight figcaption{color:var(--grey-4);display:inline-flex;font-size:.875em;font-weight:700;padding:0 6rem 0 5rem;min-height:2.5rem;width:100%;text-align:center;align-items:center;justify-content:space-between;background-color:var(--grey-3);margin-bottom:.625rem}.highlight figcaption::before{content:attr(data-lang);text-transform:Capitalize}.highlight figcaption a,.highlight figcaption span{display:block;margin-left:.625rem;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.highlight figcaption a{color:var(--grey-5)}.highlight figcaption a:hover{color:var(--grey-6)}.highlight.fullscreen{position:fixed;top:0;left:0;width:100%;height:100%;min-width:100%;z-index:9999;margin:0;border-radius:0;overflow-y:scroll;overflow-x:hidden}.highlight.fullscreen .show-btn{position:fixed}.highlight.breakline .code-container table{line-break:anywhere;white-space:break-spaces}.highlight.breakline .code-container table pre{white-space:break-spaces}.code-container{overflow-x:scroll;overflow-y:hidden}.code-container::after{content:"";display:block;height:.625rem;width:100%}.code-container table{white-space:pre;border-spacing:0;width:100%}.code-container tr{background-color:inherit}.code-container tr:hover td{background-color:var(--grey-3)}.code-container td{position:relative;padding:unset;vertical-align:unset;border:unset;transition:all .2s ease-in-out 0s}.code-container td pre::before{content:" "}.code-container td:first-child{position:absolute;background:var(--grey-2);text-align:right;overflow-x:visible!important;overflow-y:hidden;left:0;width:2.5rem;padding-right:.9375rem;color:var(--color-grey);z-index:1}.code-container td:nth-child(2){padding-left:3rem}.code-container td:last-child{padding-right:.9375rem}.code-container td[data-num]::before{content:attr(data-num)}.code-container td[data-command]::before{content:attr(data-command) " ";color:var(--grey-5);display:block;letter-spacing:-1px;padding-right:.5rem;text-align:right;white-space:nowrap}.code-container td[data-command]+td{padding-left:.5rem;border-left:1px solid var(--grey-4)}.code-container .marked td{background-color:var(--grey-3)}.code-container .marked:hover td:last-child{background-color:var(--color-red-a1)}.code-container .show-btn{position:absolute;cursor:pointer;left:0;bottom:0;width:100%;height:2.875rem;text-align:center;color:var(--text-color);background-image:linear-gradient(to bottom,var(--grey-2-a0) 0,var(--grey-2) 80%);z-index:1}.code-container .show-btn::after{content:"";display:block;width:100%;height:1rem;background:var(--grey-2)}.code-container .show-btn .ic{margin-top:1rem}.code-container .show-btn.open{background:0 0;bottom:.5rem}.code-container .show-btn.open::after{display:none}.code-container .deletion{background:var(--color-pink-light)}.code-container .addition{background:var(--color-cyan-light)}.code-container .meta{color:var(--color-yellow);-moz-user-select:none;-webkit-user-select:none;user-select:none}.code-container .comment,.code-container .punctuation{color:var(--grey-5)}.code-container .attribute,.code-container .css .class,.code-container .css .id,.code-container .css .pseudo,.code-container .function,.code-container .html .doctype,.code-container .name,.code-container .regexp,.code-container .ruby .constant,.code-container .tag,.code-container .variable,.code-container .xml .doctype,.code-container .xml .pi,.code-container .xml .tag .title{color:var(--color-red)}.code-container .built_in,.code-container .builtin-name,.code-container .command,.code-container .constant,.code-container .literal,.code-container .number,.code-container .params,.code-container .preprocessor{color:var(--color-orange)}.code-container .css .rules .attribute,.code-container .formula,.code-container .header,.code-container .inheritance,.code-container .ruby .class .title,.code-container .ruby .symbol,.code-container .special,.code-container .string,.code-container .symbol,.code-container .value,.code-container .xml .cdata{color:var(--color-green)}.code-container .css .hexcolor,.code-container .namespace,.code-container .title{color:var(--color-aqua)}.code-container .class-name,.code-container .coffeescript .title,.code-container .javascript .title,.code-container .perl .sub,.code-container .python .decorator,.code-container .python .title,.code-container .ruby .function .title,.code-container .ruby .title .keyword{color:var(--color-blue)}.code-container .javascript .function,.code-container .keyword{color:var(--color-purple)}.md li{position:relative;margin:.2rem 0}.md li:before{transition:all .2s ease-in-out 0s}.md li p{margin:0}.md dl,.md ol,.md ul{margin:.5em 0 .5em;padding:.1em .2em .1em 1.4em}.md dl dt::before,.md ul>li::before{content:"";position:absolute;width:.4em;height:.4em;background:var(--primary-color);border-radius:50%;top:.85em;left:-1em}.md dl dt:hover::before,.md ul>li:hover::before{background:var(--color-pink)}.md ol:not([start]){counter-reset:counter}.md ol>li::before{counter-increment:counter;content:counter(counter);position:absolute;width:1.4em;height:1.4em;border-radius:50%;text-align:center;font-size:.8em;line-height:1.4;top:.5em;left:-1.8em;background:var(--primary-color);color:var(--grey-1);cursor:pointer}.md ol>li:hover::before{color:var(--grey-1);background:var(--color-pink)}.md dl dt{position:relative}.md dl dd{padding-left:.9375em}.md ul li.nodot::before{width:auto;height:auto;background:0 0!important;border:none!important;position:relative;top:0!important;left:0!important}.md ul li.task-list-item::before{width:auto;height:auto;background:0 0!important;border:none!important;position:relative;top:0!important;left:0!important}.md ul li.task-list-item input[type=checkbox]{display:none}.md ul li.task-list-item input[type=checkbox]+label::before{font-family:ic;font-weight:400;color:var(--grey-4);display:inline-block;margin-right:.625em}.md ul li.task-list-item input[type=checkbox]:checked+label::before{color:var(--note-text,var(--primary-color))}.md li ul>li::before{background:var(--grey-1);border:1px solid var(--primary-color)}.md dd ol>li::before,.md li ol>li::before{content:counter(counter) ".";background:0 0;color:var(--primary-color);font-size:1em;line-height:1;width:auto}.md dd ol>li:hover::before,.md li ol>li:hover::before{background:0 0;color:var(--color-pink)}.md ol>li.quiz.show.true::before,.md ol>li.quiz>ul.options li.right::before{color:var(--color-green)}.md ol>li.quiz.show.true::after,.md ol>li.quiz>ul.options li.right::after{color:var(--color-green);animation:fadeIn .5s}.md ol>li.quiz.show.false::before,.md ol>li.quiz>ul.options li.wrong::before{color:var(--color-red)}.md ol>li.quiz.show.false::after,.md ol>li.quiz>ul.options li.wrong::after{color:var(--color-red);animation:fadeIn .5s}.md ol>li.quiz{margin:.625rem 0}.md ol>li.quiz::before{counter-increment:counter;content:counter(counter) ".";position:absolute;color:var(--primary-color);text-align:right;width:2rem;left:-2.2rem;height:auto;background:0 0!important;border:none!important;font-size:1em}.md ol>li.quiz>p{margin:0;cursor:pointer}.md ol>li.quiz>p:first-child::before{content:"[" attr(data-type) "]";font-size:.75em;color:var(--grey-4);margin:auto .3125rem}.md ol>li.quiz.mistake>p:first-child::before{color:var(--primary-color)}.md ol>li.quiz .gap{display:inline-block;min-width:2.5rem;text-align:center;padding:0 .625rem;text-indent:-624.9375rem}.md ol>li.quiz .gap::after{display:block;content:"";background:var(--text-color);width:calc(100% + 1.25rem);height:.0625rem;margin-bottom:-.0625rem;margin-left:-.625rem}.md ol>li.quiz.show.fill .gap{text-indent:0}.md ol>li.quiz::after,.md ol>li.quiz::before{transition:all .2s ease-in-out 0s}.md ol>li.quiz blockquote{display:none}.md ol>li.quiz blockquote .mistake{display:block;border:.1875rem dashed var(--grey-4);padding:.625rem 1.25rem;background:var(--grey-3);margin:.3125rem}.md ol>li.quiz blockquote .mistake::before{display:block;content:"[" attr(data-type) "]";font-size:.8125em;color:var(--grey-4)}.md ol>li.quiz .note{display:none;padding:0;margin:0;background:0 0;border:none;border-radius:inherit;color:unset;font-size:inherit;filter:unset}.md ol>li.quiz.show .note,.md ol>li.quiz.show blockquote{display:block}.md ol>li.quiz ul.options{padding-inline-start:0.625rem}.md ol>li.quiz ul.options li::before{color:var(--color-grey);margin-right:.625rem;transition:all .2s ease-in-out 0s;width:auto;height:auto;background:0 0!important;border:none!important;position:relative;top:0!important;left:0!important}.md ol>li.quiz ul.options li:first-child::before{content:"A."}.md ol>li.quiz ul.options li:nth-child(2)::before{content:"B."}.md ol>li.quiz ul.options li:nth-child(3)::before{content:"C."}.md ol>li.quiz ul.options li:nth-child(4)::before{content:"D."}.md ol>li.quiz ul.options li:nth-child(5)::before{content:"E."}.md ol>li.quiz>ul.options li{cursor:pointer}.md ol>li.quiz>ul.options li::after{font-family:ic;font-weight:400;position:absolute;left:-1.5625rem;top:0}.md ol>li.quiz.show.false::before,.md ol>li.quiz.show.true::before{visibility:hidden}.md ol>li.quiz.show.false::after,.md ol>li.quiz.show.true::after{font-family:ic;font-weight:400;position:absolute;left:-1.2rem;top:.25rem}.links{display:flex;flex-wrap:wrap;font-size:.9rem}.links .item{position:relative;display:inline-flex;justify-self:center;align-items:center;line-height:1.5;width:calc(50% - 2rem);padding:.5rem 1rem;margin:1rem;transition:all .2s ease-in-out 0s;border-radius:.5rem;border:.0625rem solid var(--grey-2);box-shadow:0 .625rem 1.875rem -.9375rem var(--box-bg-shadow);--bg-color:var(--block-color, #666)}.links .item .image{display:block;width:4rem;height:4rem;border-radius:.9375rem;background:var(--grey-1) center no-repeat;background-size:contain;flex-shrink:0;border:none}.links .item .info{padding-left:1rem}.links .item .title{margin:.5rem 0;font-family:Mulish,'Noto Serif JP','Noto Serif SC',Mulish,-apple-system,"PingFang SC","Microsoft YaHei",sans-serif;font-weight:700;color:var(--bg-color);border:none}.links .item .title .ic{display:none}.links .item .desc{font-size:.75em;margin:.5rem 0}.links .item:hover{color:var(--header-text-color);background-color:var(--bg-color);box-shadow:0 .125rem 1.25rem var(--bg-color);border-color:var(--bg-color)}.links .item:hover .image{mix-blend-mode:normal;background-color:var(--bg-color)}.links .item:hover .title{color:var(--header-text-color)}@media (max-width:767px){.links{font-size:1em}.links .item{width:calc(100% - 2rem)}}.md .note{border-radius:.1875rem;margin:1rem 0;padding:1rem;position:relative;background:var(--note-bg,var(--grey-2));color:var(--grey-6);border-left:.25rem solid var(--note-border,var(--grey-4));font-size:.875em;padding-left:2.5rem;--primary-color:var(--note-text)}.md .note::before{position:absolute;left:.5rem;top:calc(50% - 1.5rem);font-family:ic;font-weight:400;font-size:1.5rem;color:var(--note-text,var(--grey-6))}.md .note.no-icon{padding-left:1rem}.md .note.no-icon::before{content:none}::-webkit-details-marker{display:none}details>summary:first-of-type{list-style-type:none}details summary{outline:0;cursor:pointer;padding:.625rem;background:var(--note-bg,none);border-radius:.25rem;transition:all .4s ease}details summary::before{font-family:ic;font-weight:400;color:var(--grey-4);margin-right:.5rem}details>div{padding:.625rem}details.danger,details.info,details.primary,details.success,details.warning{margin:0 0 .8rem}details.danger summary,details.info summary,details.primary summary,details.success summary,details.warning summary{color:var(--note-text)}details.danger summary::before,details.info summary::before,details.primary summary::before,details.success summary::before,details.warning summary::before{color:var(--note-text)}details[open]{background:var(--note-bg,none);border-radius:.25rem;color:var(--grey-6)}details[open] summary{background:var(--note-hover,var(--grey-2));border-radius:.25rem}details[open] summary::before{color:var(--primary-color)}details[open]>div{margin:0 0 .8rem}details[open].danger,details[open].info,details[open].primary,details[open].success,details[open].warning{color:var(--note-text)}details[open].danger>summary,details[open].info>summary,details[open].primary>summary,details[open].success>summary,details[open].warning>summary{color:#fff;border-radius:.25rem .25rem 0 0}details[open].danger>summary::before,details[open].info>summary::before,details[open].primary>summary::before,details[open].success>summary::before,details[open].warning>summary::before{color:#fff}span.label{display:inline;border-radius:.3rem;border:.0625rem solid;padding:.2rem .3rem;font-family:Inconsolata,consolas,Menlo,-apple-system,"PingFang SC","Microsoft YaHei";font-size:1em;background:var(--note-bg,var(--grey-2));border-color:var(--note-border,var(--grey-4));color:var(--note-text,var(--grey-6))}.tabs{display:block;position:relative}.tabs .nav{overflow:hidden;border-bottom:.0625rem solid var(--grey-1-a7);height:2.6875rem}.tabs .nav ul{display:flex;padding:0;white-space:nowrap;overflow-x:auto}.tabs .nav li{position:relative;cursor:pointer;border:none;display:inline-block;padding:.3125rem 1.25rem;margin:0}.tabs .nav li::before{content:"";position:absolute;left:50%;right:50%;top:auto;bottom:0;transition:all .2s ease-in-out;width:auto;height:auto;background:0 0;border-radius:0;border-bottom:.125rem solid transparent}.tabs .nav li.active::before{left:0;right:0;border-bottom-color:var(--note-hover,var(--primary-color))}.tabs .tab{display:none}.tabs .tab.active{display:block}.tabs .show-btn{position:absolute;cursor:pointer;right:0;bottom:0;width:2.875rem;height:2.875rem;text-align:center;color:var(--grey-4);z-index:1}.tabs .show-btn::before{font-family:ic;font-weight:400}.md .tabs{margin:0 0 2rem;border-radius:.5rem;border:.0625rem solid var(--grey-2);box-shadow:0 .625rem 1.875rem -.9375rem var(--box-bg-shadow)}.md .tabs .tab{padding:1.25rem}.tabs .nav{overflow:visible}.tabs .nav li.active{font-weight:700}.md .note h2,.md .note h3,.md .note h4,.md .note h5,.md .note h6,.md details h2,.md details h3,.md details h4,.md details h5,.md details h6{margin-top:.1875rem;border-bottom:initial;margin-bottom:0;padding-top:0}.md .note p,.md details p{line-height:1.8}.md .note a:not(.btn),.md details a:not(.btn){color:var(--note-hover);border-bottom:.0625rem dashed var(--note-text)}.md .note a:not(.btn):hover,.md details a:not(.btn):hover{color:var(--note-text);border-bottom-color:var(--note-text)}.md .note ul li::before,.md details ul li::before{background:var(--note-text)}.md .note ol li:before,.md details ol li:before{background:var(--note-text,var(--primary-color))!important;width:.8rem;height:.8rem;font-size:.45rem;line-height:.8rem;top:.4rem;left:-1.2rem}.media-container{position:relative;border-radius:.5rem;border:.0625rem solid var(--grey-2);box-shadow:0 .625rem 1.875rem -.9375rem var(--box-bg-shadow);overflow:hidden;margin-bottom:2rem}.player-info{color:var(--text-color);font-size:1em;min-width:16.25rem}.player-info .tabs{border:none;box-shadow:none}.player-info .tabs .show-btn{display:none}.player-info .tabs .tab{padding:0}.player-info .controller{font-family:ic;font-weight:400;cursor:pointer;font-size:1.25em;display:flex;justify-content:space-around;align-items:center;text-align:center}.player-info .controller .btn{color:var(--grey-6);width:18%}.player-info .controller .btn:hover{color:var(--color-pink)}.player-info .play-pause{font-size:130%}.player-info .volume{position:relative}.player-info .volume .bar{position:absolute;height:.25rem;background:var(--color-pink-a3);bottom:0;left:0;transition:all .2s ease-out 0s}.player-info .volume.off .bar{height:0}.player-info .playlist{border-top:.125rem solid var(--grey-9-a1)}.player-info ol{font-size:.8125em;padding:.3125rem 0;margin:.625rem 0 0;height:12.5rem;overflow-x:scroll;counter-reset:counter;position:relative}.player-info ol::-webkit-scrollbar{width:.1875rem;height:.1875rem}.player-info ol>li{display:flex;padding:.3125rem .9375rem .3125rem 1.5625rem;cursor:pointer;transition:all .2s ease-in-out 0s;height:2rem;overflow:hidden}.player-info ol>li.error{opacity:.5;-webkit-text-decoration-line:line-through;text-decoration-line:line-through}.player-info ol>li::before{height:auto;background:0 0!important;border:none!important;position:relative;top:0!important;left:0!important;font-size:inherit;line-height:inherit;margin-left:-1.25rem;width:1.875rem;counter-increment:counter;content:counter(counter);text-align:right;padding-right:.3125rem;color:var(--grey-5)}.player-info ol>li .info{display:block;width:100%}.player-info ol>li .info span:nth-child(2){float:right;margin-left:.625rem;color:var(--grey-5)}.player-info ol>li.current{color:var(--primary-color);position:relative}.player-info ol>li.current::before{font-family:ic;font-weight:400;color:currentColor}.player-info ol>li.current .progress .bar{position:absolute;height:100%;background:var(--color-pink-a3);top:0;left:0;transition:all .2s ease-out 0s}.player-info ol>li.current .progress::before{content:attr(data-ptime) " / " attr(data-dtime);color:var(--grey-5);position:absolute;right:0;padding:0 .3rem}.player-info ol>li.current .progress.seeking::before{color:currentColor}.player-info ol>li.current .info{padding-right:5rem;-webkit-user-select:none;-moz-user-select:none;user-select:none}.player-info ol>li.current .info span:nth-child(2){display:none}.player-info ol>li:hover{background-color:var(--grey-1-a7)}.player-info ol>li:hover::before{color:var(--primary-color)}.video{border-radius:0 0 .5rem .5rem}.audio .preview{display:flex;align-items:center;padding:1.25rem 1.25rem 0}@media (max-width:767px){.audio .preview{padding:1.25rem .625rem 0;flex-direction:column}}.audio .preview .info{display:flex;flex-direction:column;margin-left:.625rem;padding:.625rem;text-overflow:ellipsis;overflow:hidden;width:100%}@media (max-width:767px){.audio .preview .info{margin-left:0}}.audio .preview .info h4{margin:0;text-overflow:ellipsis;max-height:3rem;padding:0;overflow:hidden}.audio .preview .info span{font-size:.75em}.audio .preview .lrc{max-height:4rem;margin-top:.3125rem;font-size:.75em;position:relative;text-align:center;overflow:hidden}.audio .preview .lrc .inner{width:100%;transition:all .5s ease-out}.audio .preview .lrc p{font-size:.75rem;color:var(--grey-6);line-height:1rem!important;height:1rem!important;padding:0!important;margin:0!important;transition:all .5s ease-out;opacity:.3;overflow:hidden}.audio .preview .lrc p.current{opacity:1;overflow:visible;height:auto!important;min-height:1rem}.audio .cover{position:relative;flex-shrink:0;cursor:pointer}.audio .cover::before{content:"";position:absolute;z-index:1;background:url("../images/play_needle.png") no-repeat center/contain;width:3.4375rem;height:5.1875rem;top:-1.5625rem;left:-1.5625rem;transform:rotateZ(-60deg);animation:rotate-needle-pause .5s 1 normal linear forwards;transform-origin:0.625rem 0.625rem}.audio .cover .disc{animation:rotate 20s linear infinite;animation-play-state:paused;padding:1.5rem;max-height:144px}.audio .cover .disc::after{content:"";position:absolute;background:url("../images/play_disc.png") no-repeat center/contain;z-index:1;width:100%;height:100%;top:0;left:0}.audio .cover img{-o-object-fit:contain;object-fit:contain;width:6rem;height:6rem;border-radius:50%}.audio.playing .cover::before{animation:rotate-needle-resume .5s 1 normal linear forwards}.audio.playing .cover .disc{animation-play-state:running} /* rebuild by hrmmi */ \ No newline at end of file diff --git a/css/comment.css b/css/comment.css index 3f04d115..437174ba 100644 --- a/css/comment.css +++ b/css/comment.css @@ -1,3 +1,3 @@ -/* build time:Fri Jun 28 2024 18:11:52 GMT+0800 (中国标准时间)*/ +/* build time:Mon Jul 01 2024 11:40:16 GMT+0800 (中国标准时间)*/ .v.wrap{animation:none;opacity:0}.v .lozaded{animation:none}.v input,.v textarea{border:none;outline:0;background:0 0;font-size:.75em;transition:all .25s ease}.v a{color:var(--color-aqua)}.v a:hover{color:var(--primary-color)}.v .text-center{text-align:center}.v .text-right{text-align:right}.v .float-right{float:right!important}.v .pd5{padding:5px}.v .pd10{padding:10px}.v .emoji,.v .vemoji{max-height:1.5rem;display:inline;vertical-align:middle}.v .submitting{margin:1em 0}.v .form{border-radius:.5rem;border:.0625rem solid var(--grey-2);box-shadow:0 .625rem 1.875rem -.9375rem var(--box-bg-shadow);margin-bottom:.625rem;position:relative;padding:.625rem}.v .form>.inner .auth-section{display:flex;padding:.3em .6em}@media (max-width:567px){.v .form>.inner{flex-wrap:wrap}}.v .form>.inner .input{flex:1 1 27%;width:27%}@media (max-width:567px){.v .form>.inner .input{flex:1 1 100%;padding-right:0;width:100%}}.v .form>.inner .input input{padding:.625rem .3125rem;width:100%;border-bottom:.0625rem dashed var(--grey-4)}.v .form>.inner .input input:focus{border-bottom-color:var(--primary-color)}.v .form>.inner .textarea{padding:.3em .6em}.v .form>.inner .veditor{width:100%;min-height:8.75em;font-size:.875em;line-height:1.75;resize:vertical}.v .form>.inner .btn-group{padding:.625rem 0;display:flex}.v .form>.inner .btn-group .i-markdown{font-size:.75em}.v .form>.inner .btn-group .left{width:30%}.v .form>.inner .btn-group .right{width:70%;text-align:right}.v .form>.inner .vextra-area{margin:.4em 0 .4em .5em}.v .form>.inner .preview-box{padding:.4375rem 1.25rem;font-size:.8125em;border-radius:.5rem;border:.0625rem solid var(--grey-2);box-shadow:0 .625rem 1.875rem -.9375rem var(--box-bg-shadow)}.v .form>.inner .trigger-section{display:none}.v .form .smile-body img{display:inline-block;width:auto;height:3rem;cursor:pointer;padding:.5rem;border-radius:.25rem;transition:box-shadow .2s ease-out,background .2s ease-out}.v .form .smile-body img:hover,.v .form .smile-body li.active img{background:var(--grey-2);box-shadow:1px 1px 2px var(--grey-1),-1px -1px 2px var(--grey-4),inset 8px 8px 16px var(--grey-1),inset -8px -8px 16px var(--grey-3)}.v .form .smile-body .smile-icons{border-radius:.5rem;margin:.3rem 0;min-height:10rem;max-height:10rem;overflow:auto;background:var(--grey-2);padding:.5rem}.v .form .smile-body .smile-bar li{display:inline-block;margin-right:.25rem}.v .form .smile-body .smile-bar img{width:2.8rem;height:2.8rem;background:var(--grey-2);-o-object-fit:contain;object-fit:contain}.v .form .btn{cursor:pointer;display:inline-block;margin:.5rem;overflow:hidden;vertical-align:middle}.v .form .btn.actived .ic{color:var(--primary-color)}.v .form .btn .ic{color:var(--grey-6);transition:all .25s ease}.v .form .btn .ic:hover{color:var(--color-blue)}.v .form .cancel-reply{position:absolute;right:0;top:0;z-index:1}.v .form .vsubmit{font-size:.875em}.v .msg{position:absolute;background:var(--grey-1-a5);width:100%;height:100%;left:0;top:0}.v .msg .alert{padding:3em 0 0 0}.v .msg .alert .text{color:var(--grey-8);padding:15px}@media (max-width:767px){.v .msg .alert{padding:8em 0}.v .msg .alert .text{padding:10px}}.v .info{padding:.3125rem;font-weight:600;font-size:1.25em}.v .info .col{display:inline-block;padding:0 20px;position:relative;background:0 0;z-index:2}.v .info .col .count{font-size:1.375rem;font-weight:inherit}.v .next .more{cursor:pointer;margin:1em 0}.v .list{width:100%}.v .item{word-break:break-all;padding-top:1.25em}.v .item .avatar{width:3.125em;height:3.125em;float:left;border-radius:50%;margin-right:.7525em;border:.0625rem solid var(--grey-3);padding:.125em}.v .item:hover .avatar{animation:.8s ease-out 0s 1 normal both running shake}.v .item .main{overflow:hidden;padding-bottom:.5em;border-bottom:.0625rem dashed var(--grey-3)}.v .item .head,.v .item .meta{color:var(--grey-5);line-height:1.8}.v .item .name{margin-right:.875rem}.v .item .system,.v .item .tag,.v .item .time{display:inline-block;font-size:.75em}@media (max-width:567px){.v .item .system span{display:none}}.v .item .tag{border-radius:.125rem;color:var(--grey-1);padding:0 .3125rem;margin-right:.5rem;background-color:var(--color-grey)}.v .item .tag.master{background-color:var(--color-orange)}.v .item .tag.friend{background-color:var(--color-aqua)}.v .item .tag.investor{background-color:var(--color-pink)}.v .item .at{color:var(--primary-color)}.v .item .at:hover{color:var(--color-aqua)}.v .item .meta{line-height:1}.v .item .meta .at{float:right}.v .item .content{margin-bottom:.75em;padding:.625em}.v .item .content>.inner{font-size:.875em;overflow-wrap:break-word;word-wrap:break-word;word-break:break-all;line-height:2;position:relative}.v .item .content>.inner.expand{cursor:pointer;max-height:8em;overflow:hidden}.v .item .content>.inner.expand::before{display:block;content:"";position:absolute;width:100%;left:0;top:0;bottom:3.15em;background:linear-gradient(180deg,var(--grey-1-a0),var(--grey-1));z-index:999}.v .item .content>.inner.expand::after{display:block;content:attr(data-expand);text-align:center;position:absolute;width:100%;height:3.15em;line-height:3.15em;left:0;bottom:0;z-index:999;background:var(--grey-1)}.v .item .content .form{margin-top:1rem}.v .item:last-child .main{border-bottom:none}.v .item .children{padding-left:1em;border-left:.0625rem dashed var(--grey-3)}.v .item .children .avatar{width:2.225em;height:2.225em}.v .empty{padding:20px;text-align:center}.v .spinner{margin:10px auto;width:50px;height:30px;text-align:center;font-size:10px}.v .spinner>div{background-color:var(--primary-color);height:100%;width:6px;margin-right:3px;display:inline-block;animation:sk-stretchdelay 1.2s infinite ease-in-out}.v .spinner .r2{animation-delay:-1.1s}.v .spinner .r3{animation-delay:-1s}.v .spinner .r4{animation-delay:-.9s}.v .spinner .r5{animation-delay:-.8s}@keyframes sk-stretchdelay{0%,100%,40%{transform:scaleY(.4)}20%{transform:scaleY(1)}}.hljs-comment,.hljs-quote{color:var(--grey-5);font-style:italic}.hljs-keyword,.hljs-meta,.hljs-selector-tag,.hljs-subst{color:var(--grey-7);font-weight:700}.hljs-literal,.hljs-number,.hljs-tag .hljs-attr,.hljs-template-variable,.hljs-variable{color:var(--color-green)}.hljs-doctag,.hljs-string{color:var(--color-red)}.hljs-section,.hljs-selector-id,.hljs-title{color:var(--color-pink);font-weight:700}.hljs-subst{font-weight:400}.hljs-class .hljs-title,.hljs-type{color:var(--color-grey);font-weight:700}.hljs-attribute,.hljs-name,.hljs-tag{color:var(--color-orange);font-weight:400}.hljs-link,.hljs-regexp{color:var(--color-green)}.hljs-bullet,.hljs-symbol{color:var(--color-purple)}.hljs-built_in,.hljs-builtin-name{color:var(--color-aqua)}.hljs-deletion{background:#fdd}.hljs-addition{background:#dfd}.hljs-emphasis{font-style:italic}.hljs-strong{font-weight:700} /* rebuild by hrmmi */ \ No newline at end of file diff --git a/css/mermaid.css b/css/mermaid.css index 52aa062f..e904b646 100644 --- a/css/mermaid.css +++ b/css/mermaid.css @@ -1,3 +1,3 @@ -/* build time:Fri Jun 28 2024 18:11:52 GMT+0800 (中国标准时间)*/ +/* build time:Mon Jul 01 2024 11:40:16 GMT+0800 (中国标准时间)*/ pre.graphviz,pre.mermaid{background:0 0;padding:0;border:none}pre>svg{display:block;margin:0 auto}pre.graphviz{white-space:normal;max-width:calc(100% - 3.125rem)}pre.graphviz svg>g>polygon{fill:var(--grey-1)}.mermaid.graph svg{max-height:18.75rem}.mermaid .label{font-family:var(--mermaid-font-family);color:var(--grey-7);font-size:.875em}.mermaid .label text{fill:var(--text-color)}.mermaid .node circle,.mermaid .node ellipse,.mermaid .node path,.mermaid .node polygon,.mermaid .node rect{fill:var(--grey-1);stroke:var(--text-color);stroke-width:.0625rem}.mermaid .node .label{text-align:center}.mermaid .node.clickable{cursor:pointer}.mermaid .arrowheadPath{fill:var(--text-color)}.mermaid .edgePath .path{stroke:var(--text-color)}.mermaid .flowchart-link{stroke:var(--text-color);fill:none}.mermaid .edgeLabel{background-color:var(--grey-2);text-align:center}.mermaid .edgeLabel rect{opacity:.8;fill:var(--grey-2)!important}.mermaid .cluster rect{fill:var(--grey-2);stroke:var(--text-color);stroke-width:.0625rem}.mermaid .cluster text{fill:var(--text-color)}.mermaid div.mermaidTooltip{position:absolute;text-align:center;max-width:12.5rem;padding:.125rem;font-family:var(--mermaid-font-family);font-size:.75rem;background:var(--grey-1);border:.0625rem solid var(--text-color);border-radius:.125rem;pointer-events:none;z-index:99}.mermaid .actor{stroke:var(--text-color);fill:var(--grey-1)}.mermaid text.actor{fill:var(--text-color);stroke:none}.mermaid .actor-line{stroke:var(--text-color)}.mermaid .messageLine0{stroke-width:1.5;stroke-dasharray:'2 2';stroke:var(--text-color)}.mermaid .messageLine1{stroke-width:1.5;stroke-dasharray:'2 2';stroke:var(--text-color)}.mermaid #arrowhead{fill:var(--text-color)}.mermaid .sequenceNumber{fill:#fff}.mermaid #sequencenumber{fill:var(--text-color)}.mermaid #crosshead path{fill:var(--text-color)!important;stroke:var(--text-color)!important}.mermaid .messageText{fill:var(--text-color);stroke:none}.mermaid .labelBox{stroke:var(--text-color);fill:var(--grey-3)}.mermaid .labelText{fill:var(--text-color);stroke:none}.mermaid .loopText{fill:var(--text-color);stroke:none}.mermaid .loopLine{stroke-width:2;stroke-dasharray:'2 2';stroke:var(--text-color)}.mermaid .note{stroke:var(--grey-4);fill:var(--grey-3)}.mermaid .noteText{fill:#000;stroke:none;font-family:var(--mermaid-font-family);font-size:.875rem}.mermaid .activation0{fill:#f4f4f4;stroke:#666}.mermaid .activation1{fill:#f4f4f4;stroke:#666}.mermaid .activation2{fill:#f4f4f4;stroke:#666}.mermaid .mermaid-main-font{font-family:var(--mermaid-font-family)}.mermaid .section{stroke:none;opacity:.2}.mermaid .section0{fill:var(--grey-1)}.mermaid .section2{fill:var(--grey-1)}.mermaid .section1,.mermaid .section3{fill:var(--grey-0);opacity:.2}.mermaid .sectionTitle0{fill:var(--text-color)}.mermaid .sectionTitle1{fill:var(--text-color)}.mermaid .sectionTitle2{fill:var(--text-color)}.mermaid .sectionTitle3{fill:var(--text-color)}.mermaid .sectionTitle{text-anchor:start;font-size:.6875rem;text-height:0.875rem;font-family:var(--mermaid-font-family)}.mermaid .grid .tick{stroke:var(--text-color);opacity:.8;shape-rendering:crispEdges}.mermaid .grid .tick text{font-family:var(--mermaid-font-family)}.mermaid .grid path{stroke-width:0}.mermaid .today{fill:none;stroke:var(--primary-color);stroke-width:.125rem}.mermaid .task{stroke-width:2}.mermaid .taskText{text-anchor:middle;font-family:var(--mermaid-font-family)}.mermaid .taskText:not([font-size]){font-size:.6875rem}.mermaid .taskTextOutsideRight{fill:var(--text-color);text-anchor:start;font-size:.6875rem;font-family:var(--mermaid-font-family)}.mermaid .taskTextOutsideLeft{fill:var(--text-color);text-anchor:end;font-size:.6875rem}.mermaid .task.clickable{cursor:pointer}.mermaid .taskText.clickable{cursor:pointer;fill:#003163!important;font-weight:700}.mermaid .taskTextOutsideLeft.clickable{cursor:pointer;fill:#003163!important;font-weight:700}.mermaid .taskTextOutsideRight.clickable{cursor:pointer;fill:#003163!important;font-weight:700}.mermaid .taskText0,.mermaid .taskText1,.mermaid .taskText2,.mermaid .taskText3{fill:var(--grey-1)}.mermaid .task0,.mermaid .task1,.mermaid .task2,.mermaid .task3{fill:var(--text-color);stroke:var(--text-color)}.mermaid .taskTextOutside0,.mermaid .taskTextOutside2{fill:var(--color-blue)}.mermaid .taskTextOutside1,.mermaid .taskTextOutside3{fill:var(--color-purple)}.mermaid .active0,.mermaid .active1,.mermaid .active2,.mermaid .active3{fill:var(--grey-1);stroke:var(--text-color)}.mermaid .activeText0,.mermaid .activeText1,.mermaid .activeText2,.mermaid .activeText3{fill:var(--text-color)!important}.mermaid .done0,.mermaid .done1,.mermaid .done2,.mermaid .done3{stroke:var(--color-green);fill:var(--grey-1);stroke-width:2}.mermaid .doneText0,.mermaid .doneText1,.mermaid .doneText2,.mermaid .doneText3{fill:var(--text-color)!important}.mermaid .crit0,.mermaid .crit1,.mermaid .crit2,.mermaid .crit3{stroke:var(--primary-color);fill:var(--primary-color);stroke-width:2}.mermaid .activeCrit0,.mermaid .activeCrit1,.mermaid .activeCrit2,.mermaid .activeCrit3{stroke:var(--primary-color);fill:var(--grey-1);stroke-width:2}.mermaid .doneCrit0,.mermaid .doneCrit1,.mermaid .doneCrit2,.mermaid .doneCrit3{stroke:var(--text-color);fill:var(--text-color);stroke-width:2;cursor:pointer;shape-rendering:crispEdges}.mermaid .milestone{transform:rotate(45deg) scale(.8,.8)}.mermaid .milestoneText{font-style:italic}.mermaid .doneCritText0,.mermaid .doneCritText1,.mermaid .doneCritText2,.mermaid .doneCritText3{fill:var(--color-green)!important}.mermaid .activeCritText0,.mermaid .activeCritText1,.mermaid .activeCritText2,.mermaid .activeCritText3{fill:var(--text-color)!important}.mermaid .titleText{text-anchor:middle;font-size:1.125rem;fill:var(--text-color);font-family:var(--mermaid-font-family)}.mermaid g.classGroup text{fill:var(--text-color);stroke:none;font-family:var(--mermaid-font-family);font-size:.625rem}.mermaid g.classGroup text .title{font-weight:bolder}.mermaid g.clickable{cursor:pointer}.mermaid g.classGroup rect{fill:var(--grey-1);stroke:var(--text-color)}.mermaid g.classGroup line{stroke:var(--text-color);stroke-width:1}.mermaid .classLabel .box{stroke:none;stroke-width:0;fill:var(--grey-1);opacity:.5}.mermaid .classLabel .label{fill:var(--text-color);font-size:.625rem}.mermaid .relation{stroke:var(--text-color);stroke-width:1;fill:none}.mermaid .dashed-line{stroke-dasharray:3}@mixin composition{fill:var(--text-color);stroke:var(--text-color);stroke-width:1}@include composition;@include composition;@mixin aggregation{fill:var(--grey-1);stroke:var(--text-color);stroke-width:1}@include aggregation;@include aggregation;@include composition;@include composition;@include composition;@include composition;.mermaid .branch-label,.mermaid .commit-id,.mermaid .commit-msg{fill:#d3d3d3;color:#d3d3d3;font-family:var(--mermaid-font-family)}.mermaid .pieTitleText{text-anchor:middle;font-size:1.5625rem;fill:$taskTextDarkColor;font-family:var(--mermaid-font-family)}.mermaid .slice{font-family:var(--mermaid-font-family)}.mermaid g.stateGroup text{fill:var(--text-color);stroke:none;font-size:.625rem;font-family:var(--mermaid-font-family)}.mermaid g.stateGroup text{fill:var(--text-color);stroke:none;font-size:.625rem}.mermaid g.stateGroup .state-title{font-weight:bolder;fill:#000}.mermaid g.stateGroup rect{fill:var(--grey-1);stroke:var(--text-color)}.mermaid g.stateGroup line{stroke:var(--text-color);stroke-width:1}.mermaid .transition{stroke:var(--text-color);stroke-width:1;fill:none}.mermaid .stateGroup .composit{fill:#fff;border-bottom:.0625rem}.mermaid .stateGroup .alt-composit{fill:#e0e0e0;border-bottom:.0625rem}.mermaid .state-note{stroke:var(--grey-4);fill:var(--grey-3)}.mermaid .state-note text{fill:#000;stroke:none;font-size:.625rem}.mermaid .stateLabel .box{stroke:none;stroke-width:0;fill:var(--grey-1);opacity:.5}.mermaid .stateLabel text{fill:#000;font-size:.625rem;font-weight:700;font-family:var(--mermaid-font-family)}.mermaid .node circle.state-start{fill:#000;stroke:#000}.mermaid .node circle.state-end{fill:#000;stroke:#fff;stroke-width:1.5}.mermaid #statediagram-barbEnd{fill:var(--text-color)}.mermaid .statediagram-cluster rect{fill:var(--grey-1);stroke:var(--text-color);stroke-width:.0625rem}.mermaid .statediagram-cluster rect.outer{rx:0.3125rem;ry:0.3125rem}.mermaid .statediagram-state .divider{stroke:var(--text-color)}.mermaid .statediagram-state .title-state{rx:0.3125rem;ry:0.3125rem}.mermaid .statediagram-cluster.statediagram-cluster .inner{fill:#fff}.mermaid .statediagram-cluster.statediagram-cluster-alt .inner{fill:#e0e0e0}.mermaid .statediagram-cluster .inner{rx:0;ry:0}.mermaid .statediagram-state rect.basic{rx:0.3125rem;ry:0.3125rem}.mermaid .statediagram-state rect.divider{stroke-dasharray:10,10;fill:#efefef}.mermaid .note-edge{stroke-dasharray:5}.mermaid .statediagram-note rect{stroke:var(--grey-4);fill:var(--grey-3);stroke-width:.0625rem;rx:0;ry:0}:root{--mermaid-font-family:"trebuchet ms",verdana,arial} /* rebuild by hrmmi */ \ No newline at end of file diff --git a/feed.json b/feed.json index 6a026088..6bc0ec76 100644 --- a/feed.json +++ b/feed.json @@ -148,20 +148,6 @@ "ACGN" ] }, - { - "id": "https://fuuzen.github.io/math/discrete-math/graph/match/", - "url": "https://fuuzen.github.io/math/discrete-math/graph/match/", - "title": "匹配", - "date_published": "2024-05-23T16:00:00.000Z", - "content_html": "

    # 匹配

    \n

    # 匹配基础概念

    \n

    匹配 MMM —— 如果 MMM 是图 GGG 的边子集 (不含环),且 MMM 中的任意两条边没有共同顶点,则称 MMMGGG 的一个匹配,或对集,或边独立集。

    \n

    如果 GGG 中顶点是 GGG 的匹配 MMM 中某条边的端点,称它为 MMM 饱和点,否则为 MMM 非饱和点

    \n

    最大匹配 MMM (无向图定义) —— 如果 MMM 是图 GGG 的包含边数最多的匹配,称 MMMGGG 的一个最大匹配。

    \n

    特别是,若最大匹配饱和了 GGG 的所有顶点,称它为 GGG 的一个完美匹配

    \n

    MMM 交错路 —— 如果 MMM 是图 GGG 的匹配, GGG 中一条由 MMM 中的边和非 MMM 中的边交错形成的路,称为 GGG 中的一条 MMM 交错路。

    \n

    特别地,若 MMM 交错路的起点与终点都是 MMM 非饱和点,称这种 MMM 交错路为 MMM 可扩路

    \n

    # 贝尔热定理

    \n
    \n

    Berge

    \n
    \n

    GGG 的匹配 MMM 是最大匹配,当且仅当 GGG 不包含 MMM 可扩路。

    \n

    # HALL 定理

    \n
    \n

    完美匹配的充要条件,将 XXX 视为需求,YYY 视为供给,可以理解为供给必须满足需求。

    \n
    \n

    G=(X,Y)G=(X,Y)G=(X,Y) 是偶图,则 GGG 存在饱和 XXX 每个顶点的完美匹配的充要条件是:

    \n

    SX,  N(S)S\\forall S\\subseteq X,\\ \\ |N(S)|\\geq|S|\nSX,  N(S)S

    \n

    其中 N(S)N(S)N(S) 表示 SSS 的邻集(图 GGG 中所有与 SSS 相邻接顶点的集合),N(S)N(S)N(S) 一定是 YYY 的子集。

    \n
    \n

    G=(X,Y)G=(X,Y)G=(X,Y) 存在饱和 XXX 每个顶点的匹配也常说成存在由 XXXYYY 的匹配 。

    \n
    \n

    ⭐️推论:若 GGGk(k>0)k(k\\gt0)k(k>0) 正则偶图,则 GGG 存在完美匹配。

    \n

    # 典例

    \n

    # 例 1

    \n

    证明每个 kkk 方体都是 kkk 正则偶图。

    \n

    事实上,由 kkk 方体的构造: kkk 方体有 2k2k2k 个顶点,每个顶点可以用长度为 kkk 的二进制码来表示,两个顶点连线当且仅当代表两个顶点的二进制码只有一位坐标不同。

    \n

    如果我们划分 kkk 方体的 2k2k2k 个顶点,把坐标之和为偶数的顶点归入 XXX,否则归入 YYY 。显然,XXX 中顶点互不邻接,YYY 中顶点也如此。所以 kkk 方体是偶图。

    \n

    又不难知道 kkk 方体的每个顶点度数为 kkk , 所以 kkk 方体是 kkk 正则偶图。

    \n

    由推论: kkk 方体存在完美匹配。

    \n

    直接在 k 方体中找出完美匹配。

    \n

    kkk 方体顶点二进制码为 x1,x2,,xkx_1,x_2,\\cdots,x_kx1,x2,,xk , 我们取 x1,x2,,xk1,0x_1,x_2,\\cdots,x_{k-1},0x1,x2,,xk1,0 , 和

    \n

    (x1,x2,,xk1,1)(x-1,x_2,\\cdots,x_{k-1},1)(x1,x2,,xk1,1) 之间的全体边所成之集为 MMM

    \n

    显然,MMM 中的边均不相邻接,所以作成 kkk 方体的匹配,又容易知道:M=2k1|M|=2^{k-1}M=2k1,所以 MMM 是完美匹配。

    \n

    # 例 2

    \n

    𝐾2𝑛𝐾_{2𝑛}K2n𝐾𝑛,𝑛𝐾_{𝑛,𝑛}Kn,n 中不同的完美匹配的个数

    \n

    𝐾2𝑛𝐾_{2𝑛}K2n 的任意一个顶点有 2𝑛12𝑛-12n1 种不同的方法被匹配。所以 𝐾2𝑛𝐾_{2𝑛}K2n 的不同完美匹配个数等于 (2𝑛1)𝐾2𝑛2(2𝑛-1)𝐾_{2𝑛-2}(2n1)K2n2 , 如此推下去,可以归纳出 𝐾2𝑛𝐾_{2𝑛}K2n 的不同完美匹配个数为:(2𝑛1)!!(2𝑛-1)!!(2n1)!!

    \n

    同样的推导方法可归纳出 𝐾𝑛,𝑛𝐾_{𝑛,𝑛}Kn,n 的不同完美匹配个数为:𝑛!𝑛!n!

    \n

    # 例 3

    \n

    证明树至多存在一个完美匹配。

    \n

    证明:若不然,设 𝑀1𝑀_1M1𝑀2𝑀_2M2 是树 TTT 的两个不同的完美匹配,那么 𝑀1Δ𝑀2𝑀_1\\Delta𝑀_2≠\\varnothingM1ΔM2=, 容易知道:𝑇[𝑀1Δ𝑀2]𝑇[𝑀_1\\Delta 𝑀_2]T[M1ΔM2] 每个非空部分顶点度数为 2 ,即它存在圈,于是推出 TTT 中有圈,矛盾。

    \n

    # 图的点覆盖

    \n

    GGG 的一个顶点子集 KKK 称为 GGG 的一个点覆盖,如果 GGG 的每条边都至少有一个端点在 KKK 中。GGG 的一个包含点数最少的点覆盖称为 GGG 的最小点覆盖,其包含的点数称为 GGG 的覆盖数,记为 α(G)\\alpha(G)α(G)

    \n

    ⭐️设 MMMGGG 的匹配,KKKGGG 的覆盖,若 M=K|M|=|K|M=K , 则 MMM 是最大匹配,而 KKK 是最小覆盖。

    \n

    # 哥尼定理

    \n
    \n

    偶图的点覆盖与偶图匹配间的关系

    \n
    \n

    在偶图中,最大匹配的边数等于最小覆盖的顶点数。

    \n

    # 托特定理

    \n

    GGG 有完美匹配当且仅当对 VVV 的任意非空真子集 SSS , 有:

    \n

    o(GS)So(G-S)\\leq|S|\no(GS)S

    \n

    o(GS)o(G-S)o(GS) 表示奇分支数目(有奇数个顶点的分支)

    \n

    # 彼得森定理

    \n
    \n

    哥尼定理的推论,是完美匹配的充分条件而不是必要条件⚠️

    \n
    \n

    没有割边的 3 正则图存在完美匹配

    \n

    # 匈牙利算法

    \n

    可以解决问题:

    \n
      \n
    • 在无权偶图中寻找完美匹配
    • \n
    • 在有权偶图中寻找最优匹配\n
        \n
      • 在有权偶图中寻找最小匹配(权值和最小)
      • \n
      • 在有权偶图中寻找最大匹配(权值和最大,通过所有权值取反转化为最小匹配问题)。
      • \n
      \n
    • \n
    \n

    G=(X,Y)G=(X,Y)G=(X,Y),匈牙利算法需要满足约束:X=Y|X|=|Y|X=Y

    \n
    \n

    基本思想

    \n

    从任意初始匹配 M0M_0M0 出发,通过寻求一条 M0M_0M0 可扩路 PPP ,令 M1=M0ΔE(P)M_1=M_0\\Delta E(P)M1=M0ΔE(P) ,得到比 M0M_0M0 更大的匹配 M1M_1M1

    \n

    1965 年,Edmonds 首先提出:用扎根于 MMM 非饱和点 uuuMMM 交错树的生长来求 MMM 可扩路。

    \n
    \n

    # 无权偶图完美匹配

    \n

    # M 交错树

    \n

    定义 设 G=(X,Y)G=(X,Y)G=(X,Y)MMMGGG 的匹配,uuuMMM 非饱和点。称树 HHHGGG 的扎根于点 uuuMMM 交错树,如果:

    \n
      \n
    • uV(H)u\\in V(H)uV(H)
    • \n
    • 对任意 vV(H)v\\in V(H)vV(H)(u,v)(u,v)(u,v) 路是 MMM 交错路;
    • \n
    \n

    扎根于 MMM 非饱和点 uuuMMM 交错树 HHH 有两种情形:

    \n
      \n
    • 情形 1: 除点 uuu 外,HHH 中所有点为 MMM 饱和点,且在 MMM 上配对;
    • \n
    • 情形 2: HHH 包含除 uuu 外的 MMM 非饱和点。
    • \n
    \n

    # 匈牙利算法

    \n

    MMM 是初始匹配。HHH 是扎根于 MMM 非饱和点 uuu 的交错树。令:S=V(H)XS=V(H)\\cap XS=V(H)XT=V(H)YT=V(H)\\cap YT=V(H)Y

    \n
      \n
    • (a)\n
        \n
      • MMM 饱和 XXX 所有顶点,停止;
      • \n
      • 否则,设 uuuXXXMMM 非饱和顶点,置 S={u}S=\\{u\\}S={u}T=T=\\varnothingT=
      • \n
      \n
    • \n
    • (b)\n
        \n
      • N(S)=TN(S)=TN(S)=T,则 GGG 中不存在完美匹配;
      • \n
      • 否则设 yN(S)Ty\\in N(S)-TyN(S)T
      • \n
      \n
    • \n
    • (c)\n
        \n
      • yyyMMM 饱和点,且 yzMyz\\in MyzM,置 S=S{z}S=S\\cup\\{z\\}S=S{z}T=T{y}T=T\\cup \\{y\\}T=T{y},转(b);
      • \n
      • 否则,设 PPPMMM 可扩路,置 M=MΔE(P)M=M\\Delta E(P)M=MΔE(P),转(a)
      • \n
      \n
    • \n
    \n

    # 有权图最优匹配

    \n

    # 可行顶点标号

    \n

    G=(X,Y)G=(X,Y)G=(X,Y) ,若对 xX,yY\\forall x\\in X,\\forall y\\in YxX,yY,有:

    \n

    l(x)+l(y)w(xy)l(x)+l(y)\\geq w(xy)\nl(x)+l(y)w(xy)

    \n

    则称 lll 是赋权完全偶图 GGG可行顶点标号

    \n

    一个简单常用的可行顶点标号:

    \n

    {l(x)=maxyYw(xy),xXl(y)=0,yY\\begin{cases}\nl(x)=\\max_{y\\in Y}w(xy),&x\\in X\\\\\nl(y)=0,&y\\in Y\n\\end{cases}\n{l(x)=maxyYw(xy),l(y)=0,xXyY

    \n

    # 相等子图

    \n

    lll 是赋权完全偶图 G=(X,Y)G=(X,Y)G=(X,Y) 的可行顶点标号,令:

    \n

    El={xyE(G)l(x)+l(y)=w(xy)}E_l=\\{xy\\in E(G)|l(x)+l(y)=w(xy)\\}\nEl={xyE(G)l(x)+l(y)=w(xy)}

    \n

    Gl=G[El]G_l=G[E_l]Gl=G[El]GGG 的对应于 lll相等子图

    \n

    # Kuhn 算法

    \n
    \n

    Kuhn 采用顶点标号修改策略,找到了求最优匹配好算法如下

    \n

    ⚠️ 求的是权值和最大的匹配⚠️

    \n
    \n

    给一初始顶点标号 lll ,在相等子图 GlG_lGl 中任选一个匹配 MMM

    \n

    (1)

    \n
      \n
    • XXXMMM 饱和的,则 MMM 是最优匹配;
    • \n
    • 否则,令 uuu 是一个 MMM 非饱和点,置 S={u}S=\\{u\\}S={u}T=T=\\varnothingT=
    • \n
    \n

    (2)

    \n
      \n
    • TNGl(S)T\\sub N_{G_l}(S)TNGl(S),转(3);
    • \n
    • 否则,计算:
    • \n
    \n

    αl=minxS,yT{l(x)+l(y)w(xy)}\\alpha_l=\\min_{x\\in S,y\\notin T}\\{l(x)+l(y)-w(xy)\\}\nαl=xS,y/Tmin{l(x)+l(y)w(xy)}

    \n

    修改 lll 如下:

    \n

    l={l(v)αl,vSl(v)+αl,vTl(v),othersl=\\begin{cases}\nl(v)-\\alpha_l,&v\\in S\\\\\nl(v)+\\alpha_l,&v\\in T\\\\\nl(v),&\\text{others}\\\\\n\\end{cases}\nl=l(v)αl,l(v)+αl,l(v),vSvTothers

    \n

    给出新的可行顶点标号,在新标号下重新开始。

    \n

    (3)在 NGl(S)\\TN_{G_l}(S)\\backslash TNGl(S)\\T 中选择点 yyy

    \n
      \n
    • yyyMMM 饱和的,yzMyz\\in MyzM,则置 S=S{z}S=S\\cup\\{z\\}S=S{z}T=T{y}T=T\\cup\\{y\\}T=T{y},转(2);
    • \n
    • 否则,设 PPPGlG_lGlMMM 可扩路,置 M=MΔE(P)M=M\\Delta E(P)M=MΔE(P),转(1)
    • \n
    \n
    \n

    该算法把上面原来的匈牙利算法用于其中,主要是用来判定和求完美匹配。

    \n
    \n", - "tags": [ - "数学", - "离散数学", - "图论", - "离散数学", - "图论" - ] - }, { "id": "https://fuuzen.github.io/cs/os/ysosv2/lab5/", "url": "https://fuuzen.github.io/cs/os/ysosv2/lab5/", @@ -177,6 +163,20 @@ "大作业" ] }, + { + "id": "https://fuuzen.github.io/math/discrete-math/graph/match/", + "url": "https://fuuzen.github.io/math/discrete-math/graph/match/", + "title": "匹配", + "date_published": "2024-05-23T16:00:00.000Z", + "content_html": "

    # 匹配

    \n

    # 匹配基础概念

    \n

    匹配 MMM —— 如果 MMM 是图 GGG 的边子集 (不含环),且 MMM 中的任意两条边没有共同顶点,则称 MMMGGG 的一个匹配,或对集,或边独立集。

    \n

    如果 GGG 中顶点是 GGG 的匹配 MMM 中某条边的端点,称它为 MMM 饱和点,否则为 MMM 非饱和点

    \n

    最大匹配 MMM (无向图定义) —— 如果 MMM 是图 GGG 的包含边数最多的匹配,称 MMMGGG 的一个最大匹配。

    \n

    特别是,若最大匹配饱和了 GGG 的所有顶点,称它为 GGG 的一个完美匹配

    \n

    MMM 交错路 —— 如果 MMM 是图 GGG 的匹配, GGG 中一条由 MMM 中的边和非 MMM 中的边交错形成的路,称为 GGG 中的一条 MMM 交错路。

    \n

    特别地,若 MMM 交错路的起点与终点都是 MMM 非饱和点,称这种 MMM 交错路为 MMM 可扩路

    \n

    # 贝尔热定理

    \n
    \n

    Berge

    \n
    \n

    GGG 的匹配 MMM 是最大匹配,当且仅当 GGG 不包含 MMM 可扩路。

    \n

    # HALL 定理

    \n
    \n

    完美匹配的充要条件,将 XXX 视为需求,YYY 视为供给,可以理解为供给必须满足需求。

    \n
    \n

    G=(X,Y)G=(X,Y)G=(X,Y) 是偶图,则 GGG 存在饱和 XXX 每个顶点的完美匹配的充要条件是:

    \n

    SX,  N(S)S\\forall S\\subseteq X,\\ \\ |N(S)|\\geq|S|\nSX,  N(S)S

    \n

    其中 N(S)N(S)N(S) 表示 SSS 的邻集(图 GGG 中所有与 SSS 相邻接顶点的集合),N(S)N(S)N(S) 一定是 YYY 的子集。

    \n
    \n

    G=(X,Y)G=(X,Y)G=(X,Y) 存在饱和 XXX 每个顶点的匹配也常说成存在由 XXXYYY 的匹配 。

    \n
    \n

    ⭐️推论:若 GGGk(k>0)k(k\\gt0)k(k>0) 正则偶图,则 GGG 存在完美匹配。

    \n

    # 典例

    \n

    # 例 1

    \n

    证明每个 kkk 方体都是 kkk 正则偶图。

    \n

    事实上,由 kkk 方体的构造: kkk 方体有 2k2k2k 个顶点,每个顶点可以用长度为 kkk 的二进制码来表示,两个顶点连线当且仅当代表两个顶点的二进制码只有一位坐标不同。

    \n

    如果我们划分 kkk 方体的 2k2k2k 个顶点,把坐标之和为偶数的顶点归入 XXX,否则归入 YYY 。显然,XXX 中顶点互不邻接,YYY 中顶点也如此。所以 kkk 方体是偶图。

    \n

    又不难知道 kkk 方体的每个顶点度数为 kkk , 所以 kkk 方体是 kkk 正则偶图。

    \n

    由推论: kkk 方体存在完美匹配。

    \n

    直接在 k 方体中找出完美匹配。

    \n

    kkk 方体顶点二进制码为 x1,x2,,xkx_1,x_2,\\cdots,x_kx1,x2,,xk , 我们取 x1,x2,,xk1,0x_1,x_2,\\cdots,x_{k-1},0x1,x2,,xk1,0 , 和

    \n

    (x1,x2,,xk1,1)(x-1,x_2,\\cdots,x_{k-1},1)(x1,x2,,xk1,1) 之间的全体边所成之集为 MMM

    \n

    显然,MMM 中的边均不相邻接,所以作成 kkk 方体的匹配,又容易知道:M=2k1|M|=2^{k-1}M=2k1,所以 MMM 是完美匹配。

    \n

    # 例 2

    \n

    𝐾2𝑛𝐾_{2𝑛}K2n𝐾𝑛,𝑛𝐾_{𝑛,𝑛}Kn,n 中不同的完美匹配的个数

    \n

    𝐾2𝑛𝐾_{2𝑛}K2n 的任意一个顶点有 2𝑛12𝑛-12n1 种不同的方法被匹配。所以 𝐾2𝑛𝐾_{2𝑛}K2n 的不同完美匹配个数等于 (2𝑛1)𝐾2𝑛2(2𝑛-1)𝐾_{2𝑛-2}(2n1)K2n2 , 如此推下去,可以归纳出 𝐾2𝑛𝐾_{2𝑛}K2n 的不同完美匹配个数为:(2𝑛1)!!(2𝑛-1)!!(2n1)!!

    \n

    同样的推导方法可归纳出 𝐾𝑛,𝑛𝐾_{𝑛,𝑛}Kn,n 的不同完美匹配个数为:𝑛!𝑛!n!

    \n

    # 例 3

    \n

    证明树至多存在一个完美匹配。

    \n

    证明:若不然,设 𝑀1𝑀_1M1𝑀2𝑀_2M2 是树 TTT 的两个不同的完美匹配,那么 𝑀1Δ𝑀2𝑀_1\\Delta𝑀_2≠\\varnothingM1ΔM2=, 容易知道:𝑇[𝑀1Δ𝑀2]𝑇[𝑀_1\\Delta 𝑀_2]T[M1ΔM2] 每个非空部分顶点度数为 2 ,即它存在圈,于是推出 TTT 中有圈,矛盾。

    \n

    # 图的点覆盖

    \n

    GGG 的一个顶点子集 KKK 称为 GGG 的一个点覆盖,如果 GGG 的每条边都至少有一个端点在 KKK 中。GGG 的一个包含点数最少的点覆盖称为 GGG 的最小点覆盖,其包含的点数称为 GGG 的覆盖数,记为 α(G)\\alpha(G)α(G)

    \n

    ⭐️设 MMMGGG 的匹配,KKKGGG 的覆盖,若 M=K|M|=|K|M=K , 则 MMM 是最大匹配,而 KKK 是最小覆盖。

    \n

    # 哥尼定理

    \n
    \n

    偶图的点覆盖与偶图匹配间的关系

    \n
    \n

    在偶图中,最大匹配的边数等于最小覆盖的顶点数。

    \n

    # 托特定理

    \n

    GGG 有完美匹配当且仅当对 VVV 的任意非空真子集 SSS , 有:

    \n

    o(GS)So(G-S)\\leq|S|\no(GS)S

    \n

    o(GS)o(G-S)o(GS) 表示奇分支数目(有奇数个顶点的分支)

    \n

    # 彼得森定理

    \n
    \n

    哥尼定理的推论,是完美匹配的充分条件而不是必要条件⚠️

    \n
    \n

    没有割边的 3 正则图存在完美匹配

    \n

    # 匈牙利算法

    \n

    可以解决问题:

    \n
      \n
    • 在无权偶图中寻找完美匹配
    • \n
    • 在有权偶图中寻找最优匹配\n
        \n
      • 在有权偶图中寻找最小匹配(权值和最小)
      • \n
      • 在有权偶图中寻找最大匹配(权值和最大,通过所有权值取反转化为最小匹配问题)。
      • \n
      \n
    • \n
    \n

    G=(X,Y)G=(X,Y)G=(X,Y),匈牙利算法需要满足约束:X=Y|X|=|Y|X=Y

    \n
    \n

    基本思想

    \n

    从任意初始匹配 M0M_0M0 出发,通过寻求一条 M0M_0M0 可扩路 PPP ,令 M1=M0ΔE(P)M_1=M_0\\Delta E(P)M1=M0ΔE(P) ,得到比 M0M_0M0 更大的匹配 M1M_1M1

    \n

    1965 年,Edmonds 首先提出:用扎根于 MMM 非饱和点 uuuMMM 交错树的生长来求 MMM 可扩路。

    \n
    \n

    # 无权偶图完美匹配

    \n

    # M 交错树

    \n

    定义 设 G=(X,Y)G=(X,Y)G=(X,Y)MMMGGG 的匹配,uuuMMM 非饱和点。称树 HHHGGG 的扎根于点 uuuMMM 交错树,如果:

    \n
      \n
    • uV(H)u\\in V(H)uV(H)
    • \n
    • 对任意 vV(H)v\\in V(H)vV(H)(u,v)(u,v)(u,v) 路是 MMM 交错路;
    • \n
    \n

    扎根于 MMM 非饱和点 uuuMMM 交错树 HHH 有两种情形:

    \n
      \n
    • 情形 1: 除点 uuu 外,HHH 中所有点为 MMM 饱和点,且在 MMM 上配对;
    • \n
    • 情形 2: HHH 包含除 uuu 外的 MMM 非饱和点。
    • \n
    \n

    # 匈牙利算法

    \n

    MMM 是初始匹配。HHH 是扎根于 MMM 非饱和点 uuu 的交错树。令:S=V(H)XS=V(H)\\cap XS=V(H)XT=V(H)YT=V(H)\\cap YT=V(H)Y

    \n
      \n
    • (a)\n
        \n
      • MMM 饱和 XXX 所有顶点,停止;
      • \n
      • 否则,设 uuuXXXMMM 非饱和顶点,置 S={u}S=\\{u\\}S={u}T=T=\\varnothingT=
      • \n
      \n
    • \n
    • (b)\n
        \n
      • N(S)=TN(S)=TN(S)=T,则 GGG 中不存在完美匹配;
      • \n
      • 否则设 yN(S)Ty\\in N(S)-TyN(S)T
      • \n
      \n
    • \n
    • (c)\n
        \n
      • yyyMMM 饱和点,且 yzMyz\\in MyzM,置 S=S{z}S=S\\cup\\{z\\}S=S{z}T=T{y}T=T\\cup \\{y\\}T=T{y},转(b);
      • \n
      • 否则,设 PPPMMM 可扩路,置 M=MΔE(P)M=M\\Delta E(P)M=MΔE(P),转(a)
      • \n
      \n
    • \n
    \n

    # 有权图最优匹配

    \n

    # 可行顶点标号

    \n

    G=(X,Y)G=(X,Y)G=(X,Y) ,若对 xX,yY\\forall x\\in X,\\forall y\\in YxX,yY,有:

    \n

    l(x)+l(y)w(xy)l(x)+l(y)\\geq w(xy)\nl(x)+l(y)w(xy)

    \n

    则称 lll 是赋权完全偶图 GGG可行顶点标号

    \n

    一个简单常用的可行顶点标号:

    \n

    {l(x)=maxyYw(xy),xXl(y)=0,yY\\begin{cases}\nl(x)=\\max_{y\\in Y}w(xy),&x\\in X\\\\\nl(y)=0,&y\\in Y\n\\end{cases}\n{l(x)=maxyYw(xy),l(y)=0,xXyY

    \n

    # 相等子图

    \n

    lll 是赋权完全偶图 G=(X,Y)G=(X,Y)G=(X,Y) 的可行顶点标号,令:

    \n

    El={xyE(G)l(x)+l(y)=w(xy)}E_l=\\{xy\\in E(G)|l(x)+l(y)=w(xy)\\}\nEl={xyE(G)l(x)+l(y)=w(xy)}

    \n

    Gl=G[El]G_l=G[E_l]Gl=G[El]GGG 的对应于 lll相等子图

    \n

    # Kuhn 算法

    \n
    \n

    Kuhn 采用顶点标号修改策略,找到了求最优匹配好算法如下

    \n

    ⚠️ 求的是权值和最大的匹配⚠️

    \n
    \n

    给一初始顶点标号 lll ,在相等子图 GlG_lGl 中任选一个匹配 MMM

    \n

    (1)

    \n
      \n
    • XXXMMM 饱和的,则 MMM 是最优匹配;
    • \n
    • 否则,令 uuu 是一个 MMM 非饱和点,置 S={u}S=\\{u\\}S={u}T=T=\\varnothingT=
    • \n
    \n

    (2)

    \n
      \n
    • TNGl(S)T\\sub N_{G_l}(S)TNGl(S),转(3);
    • \n
    • 否则,计算:
    • \n
    \n

    αl=minxS,yT{l(x)+l(y)w(xy)}\\alpha_l=\\min_{x\\in S,y\\notin T}\\{l(x)+l(y)-w(xy)\\}\nαl=xS,y/Tmin{l(x)+l(y)w(xy)}

    \n

    修改 lll 如下:

    \n

    l={l(v)αl,vSl(v)+αl,vTl(v),othersl=\\begin{cases}\nl(v)-\\alpha_l,&v\\in S\\\\\nl(v)+\\alpha_l,&v\\in T\\\\\nl(v),&\\text{others}\\\\\n\\end{cases}\nl=l(v)αl,l(v)+αl,l(v),vSvTothers

    \n

    给出新的可行顶点标号,在新标号下重新开始。

    \n

    (3)在 NGl(S)\\TN_{G_l}(S)\\backslash TNGl(S)\\T 中选择点 yyy

    \n
      \n
    • yyyMMM 饱和的,yzMyz\\in MyzM,则置 S=S{z}S=S\\cup\\{z\\}S=S{z}T=T{y}T=T\\cup\\{y\\}T=T{y},转(2);
    • \n
    • 否则,设 PPPGlG_lGlMMM 可扩路,置 M=MΔE(P)M=M\\Delta E(P)M=MΔE(P),转(1)
    • \n
    \n
    \n

    该算法把上面原来的匈牙利算法用于其中,主要是用来判定和求完美匹配。

    \n
    \n", + "tags": [ + "数学", + "离散数学", + "图论", + "离散数学", + "图论" + ] + }, { "id": "https://fuuzen.github.io/ie/de/memory-storage/", "url": "https://fuuzen.github.io/ie/de/memory-storage/", diff --git a/friends/index.html b/friends/index.html index c5fda748..3468f694 100644 --- a/friends/index.html +++ b/friends/index.html @@ -1 +1 @@ -友情链接 | Balloon Party = 風船のパーティー = fuusen no party

    # 本站信息

    # 小伙伴们

    Views times
    \ No newline at end of file +友情链接 | Balloon Party = 風船のパーティー = fuusen no party
    \ No newline at end of file diff --git a/ie/de/bool-algebra/index.html b/ie/de/bool-algebra/index.html index 1f4fea6d..58591410 100644 --- a/ie/de/bool-algebra/index.html +++ b/ie/de/bool-algebra/index.html @@ -1 +1 @@ -逻辑代数 - 数字逻辑电路 - 电子信息 | Balloon Party = 風船のパーティー = fuusen no party

    # 逻辑代数中的基本运算

    # 三种基本运算

    三种基本运算

    # 真值表

    表征逻辑事件输入(条件)和输出(结果)之间全部可能状态的表格

    与运算:

    ABY
    000
    010
    100
    111

    或运算:

    ABY
    000
    011
    101
    111

    非运算:

    AY
    01
    10

    # 其他复合逻辑运算

    # 与非

    “先与再非”

    与非

    # 或非

    “先或再非”

    或非

    # 与或非

    “先与再或再非”

    与或非

    # 异或 XOR

    不同时输出 1,相同时输出 0

    AB=AB+ABA\oplus B=\overline{A}B+A\overline{B}

    XOR

    ⭐️异或运算满足的性质

    • 交换律
    • 结合律
    • 对于任何数 x,都有 xx=0x\oplus x=0x0=xx\oplus 0=xx1=xx\oplus 1=\overline{x}
    • 自反性 ABB=A0=AA\oplus B\oplus B=A\oplus0=A;(经典 coding 小技巧)

    # 同或 XNOR

    同或 —— 相同时输出 1,不同时输出 0

    同或

    # 逻辑代数的公式和定理

    # 基本公式

    逻辑代数的基本公式1

    逻辑代数的基本公式2

    # 常考公式 7/17 分配律

    A(B+C)=AB+ACA+BC=(A+B)(A+C)A(B +C) = AB + AC\\ A + BC = (A +B)(A +C)

    用途:将 “混杂” 在一起的变量分离,将逻辑函数表达式化成 “与或” 形式和 “或与” 形式;在后面逻辑函数的 “最小项之和” 和 “最大项之积” 表示时非常有用

    # 常考公式 8/18 反演律(德・摩根定律)

    (AB)=A+B(A+B)=AB(AB)' = A' + B' (A + B) ' = A'B'

    记忆方法:“与的非等于非的或,或的非等于非的与”;

    门电路的角度上,就是:

    • 负或门 = 与非门
    • 负与门 = 或非门

    与在概率论课程中学习的德摩根定律一致,即 “交的反等于反的并,并的反等于反的交”

    用途:实现与运算和或运算之间的转换,在后面对函数式进行特定形式的变换时非常有用

    # 基本定理

    # 代入定理

    在任何一个包含 A 的逻辑等式中,若以另外一个逻辑式代入式中 A 的位置,则等式依然成立;

    # 反演定理

    对任一逻辑式 Y ,在求解它的反时(即已知 Y 求解 Y’),只需要在保证运算优先次序不变的原则下,将其中所有的 \cdot 换成 ++++ 换成 \cdot ,原变量换成反变量,反变量换成原变量(不属于单个变量上的反号不动),1 变成 0,0 变成 1,得到的结果就是 Y’;

    # 对偶定理

    若两逻辑表达式相等,则它们的对偶式也相等;

    所谓的对偶式指的是在保持运算优先顺序不变的原则下,将表达式中所有的 \cdot 换成 ++++ 换成 \cdot ,0 换成 1,1 换成 0;

    注意与反演定理的区别!变量不取反!

    主要应用于证明题当中,证明两个形式较复杂的逻辑式相等,可以尝试:

    • 两边同时取对偶式,证明它们的对偶式相等
    • 对等号某一侧表达式先取对偶式,对对偶式进行化简之后,再取对偶就可以得到等号另一侧表达式

    # 逻辑函数及其表示方法

    以逻辑变量为输入,运算结果为输出,则输入变量值确定以后,输出的取值也随之而定,因此输入 — 输出之间是一种函数关系,称作逻辑函数;

    在我们学习的二值逻辑 (布尔逻辑) 中,输入变量和输出变量只有 0 和 1 两种状态;

    在求解实际应用问题中的逻辑函数关系时,首先必须做的事情是将具体事物抽象为逻辑变量并规定其不同状态(0 或 1),这是很重要的解题习惯,类似于在概率论答题时首先要记不同的事件为 A、B……;这里只研究现有的抽象后的逻辑函数式;

    # 不同表示方法

    # 真值表

    遍历输入变量所有取值下的输出值,列成表格

    # 逻辑函数式

    将输出写成输入变量的与、或、非等逻辑运算的组合式

    # 逻辑图

    逻辑函数式用图形符号表示

    # 波形图

    输入变量所有取值可能与对应输出的时域波形

    # 卡诺图

    # 计算机软件中的描述方式

    例如 Verilog 语言

    # 不同表示方法间的转换

    # 真值表 -> 逻辑函数式 (重点)

    (逻辑函数的最小项之和形式)

    1. 找出真值表中使逻辑函数输出 Y = 1 的输入变量取值的组合即使 Y = 1 的行;
    2. 使 Y = 1 的每组输入变量取值的组合(即每一行)对应一个乘积项,取值为 1 写入原变量,取值为 0 写入反变量;
    3. 将各行对应的乘积项相加,即最终结果为 “与或” 形式;

    # 逻辑函数式 <-> 逻辑图

    # 逻辑函数式 -> 逻辑图

    用图形符号代替逻辑函数式中的逻辑运算符;

    # 逻辑图 -> 逻辑函数式

    从输入到输出逐级写出每个图形符号对应的逻辑运算式;

    # 真值表 <-> 波形图

    # 真值表 -> 波形图

    将所有输入变量与对应输出变量的取值依次排列,以时间为横轴;

    # 波形图 -> 真值表

    在波形图中找出每个时间段里输入变量与输出变量的取值列成表格( 对于 NN 个输入变量要对应 2N2^N 个组合即 2N2^N 行)

    # 标准 / 规范 形式

    # DNF/SOP

    别名:

    • 最小项标准式
    • 析取范式
    • 积之和范式
    • 与或范式
    • Disjunctive Normal Form (DNF)
    • Sum of Product (SOP)

    任何一个逻辑函数式都可以化为若干个最小项之和的形式

    # 最小项 m 的定义:

    m 是乘积项,包含 n 个因子,n 个变量均以原变量或反变量的形式在 m 中出现一次;

    # 最小项的特点

    • 全体最小项之和为 1
    • 各个最小项之间是互斥的,即任何两个最小项之积为 0 ;换句话说,对于任何一个特定的输入变量取值,有且仅有一个最小项为 1;
    • 两个相邻的最小项之和可以合并,消去一对因子,只留下公共因子;

    这里的 “相邻” 不是按照自然二进制数编号后的相邻,而是指逻辑上的相邻,规定仅一个变量不同的最小项彼此称为相邻;

    例如 ABCA' B' CABCA'BC'ABCA'BCm1m_1m2m_2m3m_3m1m_1m2m_2 不是相邻,m1m_1m3m_3 才是相邻;m1m_1m3m_3 可以合并消去因子 B+BB+B'

    # 转化为 DNF

    事实上最小项就是在列写 n 个输入变量的逻辑函数的真值表时每一行对应的输入变量的乘积项,取值为 1 写入原变量,取值为 0 写入反变量;使最小项取值为 1 的变量取值,按照 n 位自然二进制数排列后对应的十进制数就是这个最小项的编号

    # CNF/POS

    别名:

    • 最大项标准式
    • 合取范式
    • 和之积范式
    • 或与范式
    • conjunctive normal form (CNF)
    • Product of Sum (POS)

    任何一个逻辑函数式都可以化为若干个最大项之积的形式

    # 最大项 M 的定义

    M 是相加项,包含 n 个因子,n 个变量均以原变量和反变量的形式在 M 中出现一次;可以把最大项看作是对每一个最小项应用反演定理,因此,最大项的编号原则是反的,即原变量记为 0,反变量记为 1,n 位自然二进制数排列后对应的十进制数为编号;

    例如 A+B+C′ 的编号是 001 即 M1 ;

    (记忆方法:对于与运算即乘积项,得 1 “容易”,即得 1 时输入变量组唯一确定;对于或运算即相加项,得 0 “容易”,即得 0 时输入变量组唯一确定;)

    # 最大项的特点

    • 全体最大项之积为 0 ;
    • 任何两个最大项之和为 1 ;换句话说,对于任何一个特定的输入变量取值,有且仅有一个最小项为 0;
    • 两个相邻的最大项之积保留公共变量

    这里的相邻和前面最小项的相邻一致,都是指逻辑上的相邻,即只有一个变量不同。

    例如 A+B+CA+B+C'A+B+CA+B'+C' 为相邻,即 M1M_1M3M_3 相邻,两者的乘积项即相与的结果是 A+CA+C'

    # 转化为 CNF

    当给定的逻辑函数式不是标准的最大项之积的形式时,一般不断地利用 AA=0A\cdot A'=0 补全缺少的因子,并结合分配律 A+BC=(A+B)(A+C)A + BC = (A +B)(A +C) 将与或形式转化为或与形式;

    # 最小项与最大项的关系

    根据最大项和最小项的定义以及德・摩根定理,易推导得到:

    Mi=miM_i=m'_i

    如果将逻辑函数写成最小项之和的形式:

    iR1mi\sum_{i\in R_1} m_i

    再将其写成最大项之积的形式:

    iR2Mi\prod_{i\in R_2} M_i

    则两者的下标编号集合 R1R_1R2R_2 是互补的。

    所以很多时候不需要直接展开求解最大项之积,先求解最小项之和更方便

    # 逻辑函数的化简

    # 逻辑函数的化简的目标

    逻辑函数的最小项之和虽然是通用的标准形式,但不一定就是最简的;在实际数字逻辑电路设计时,需要考虑在逻辑功能相同的条件下使用尽可能简单的电路,即电子器件数量足够少;

    事实上,没有绝对的最简,由于我们习惯性将逻辑函数写成 最小项之和 即 与或形式 ,因此对于 与或形式 的逻辑函数,包含的乘积项最少,且每个乘积项里的因子也不能再减少时,称其为最简形式即最简与或形式;其他形式的最简同理;

    常考最简式包括:

    • 最简与或式
    • 最简或与式
    • 最简与非 — 与非式
    • 最简或非 — 或非式
    • 最简与或非式

    实际的逻辑电路设计可能会受到电子器件种类与数量的限制,因此可能要进行变换;直接用卡诺图化简得到的最简与或式并不一定是设计时需要的逻辑函数形式,所以还会需要其他最简形式,例如:

    • 只能用或非门实现:与非 — 与非式
    • 只能用或非门实现:或非 — 或非式

    # 公式化简法

    • 使用前面学习的基本公式和常用公式
    • 没有固定的步骤,
    • 难以确定结果正确与否
    • 不通用但有时很快速好用

    对公式化简法,要求掌握但不必深入研究,熟悉教材上的例题和课后题即可;

    一个可能会用到公式化简法的情况是,当逻辑函数含有变量数很多时,即很难直接采用卡诺图去进行化简时,一般题目的设计会使解题时经常需要用 AB+AC+BCD=AB+ACAB + A'C + BCD = AB + A'C 这个公式来消去多余的变量,接着再用卡诺图去进行化简

    # ⭐️卡诺图化简法 (重点)

    # 卡诺图 (Karnaugh)

    2n2^n 个小方块分别代表 n 变量的所有最小项,并将它们排列成矩阵,而且使几何位置相邻的两个最小项在逻辑上也是相邻的(只有一个变量不同),就得到表示 n 变量全部最小项的卡诺图;

    # 卡诺图的绘制

    横纵轴不是按照自然二进制数排列,按照格雷码排列,目的是为了使逻辑上相邻对应几何上相邻,保证相邻的方块只有一个变量不同

    卡诺图的绘制

    记忆方法:

    • 增加一个变量看作是折纸一样翻折
    • 轴对称位置上的最小项也满足相邻
    • 把卡诺图看作是闭合的图形 (横纵轴都是循环的)

    教材上的卡诺图并不是唯一的,只要能够保证逻辑对称性几何对称性一致即可,最好去理解卡诺图绘制的过程

    无关项可以用 x 表示,表示可圈起来也可不圈起来

    # 卡诺图化简法的本质

    将最小项之和用图形描述,将逻辑上的相邻表现为几何图形的相邻,通过合并相邻的几何图形,合并后对应仅保留公共因子的最小项,最终结果就是最简与或式

    # 卡诺图化简逻辑函数的步骤

    # 最简与或式

    若只给出逻辑函数则从步骤 1 开始,若给出卡诺图则直接从步骤 2 开始

    1. 将逻辑函数写成最小项之和的形式
    2. 根据最小项之和绘制对应的卡诺图,逻辑函数式中含有的最小项的对应位置填 1,其余的位置填 0
    3. 合并相邻的最小项,对应最终结果为 最简与或式
    • 相邻表示逻辑相邻,轴对称位置上的方块也可合并
    • 合并时,每个方块可以重复使用,因为 A+A=AA+A=A
    • 合并的原则是乘积项的数目最少,每个乘积项的因子最少,即画的圈要尽量大(消去尽量多的不同的因子),画圈的数目尽量少
    • 结果不唯一,但乘积项的数目和每个乘积项的因子个数一致(这也是检查答案是否正确的依据)

    # 最简或与式

    1. 对原逻辑函数 YY 取反得到 YY'
    2. YY' 采用上述卡诺图求最简与或式步骤得到 YY' 的 最简与或式
    3. YY' 的 最简与或式 取反变换得到的 或与式 就是原逻辑函数 YY 的 最简或与式

    或者直接:

    1. 采用上述卡诺图求最简与或式步骤,但是合并为 0 的方块而不是 1 ,得到 YY' 的最简与或式
    2. YY' 的 最简与或式 取反变换得到的 或与式 就是原逻辑函数 YY 的 最简或与式

    # 最简与或非形式

    与最简或与式实际上是等价的,按照求最简或与式步骤即可,只是最后一步取反直接加上取反符号不需要再变换为或与式

    # 最简与非 — 与非式

    将得到的最简与或式取两次反,即逆向利用德・摩根定理

    # 最简或非 — 或非形式

    将与或非形式中的每个乘积项再利用德・摩根定理,即变为或非形式

    # 含有无关项的卡诺图化简

    一般第二章的题目会直接给出无关项,例如:

    • $ (...) + (...) +…… = 0 $
    • d()=0\sum d(…)= 0 (无关项 集合用 dd 表示)

    后面在基于实际应用设计逻辑电路时应自行根据实际情况来分析确定无关项

    1. 根据题意确定无关项
    2. 绘制卡诺图时,无关项的方块用 × 表示
    3. 化简时根据需要,将无关项填入 1 或 0 ,原则依然是是使化简后的项数最少,每项的因子最少,即画的圈最大,圈的个数最少
    Edited on Views times
    \ No newline at end of file +逻辑代数 - 数字逻辑电路 - 电子信息 | Balloon Party = 風船のパーティー = fuusen no party

    # 逻辑代数中的基本运算

    # 三种基本运算

    三种基本运算

    # 真值表

    表征逻辑事件输入(条件)和输出(结果)之间全部可能状态的表格

    与运算:

    ABY
    000
    010
    100
    111

    或运算:

    ABY
    000
    011
    101
    111

    非运算:

    AY
    01
    10

    # 其他复合逻辑运算

    # 与非

    “先与再非”

    与非

    # 或非

    “先或再非”

    或非

    # 与或非

    “先与再或再非”

    与或非

    # 异或 XOR

    不同时输出 1,相同时输出 0

    AB=AB+ABA\oplus B=\overline{A}B+A\overline{B}

    XOR

    ⭐️异或运算满足的性质

    • 交换律
    • 结合律
    • 对于任何数 x,都有 xx=0x\oplus x=0x0=xx\oplus 0=xx1=xx\oplus 1=\overline{x}
    • 自反性 ABB=A0=AA\oplus B\oplus B=A\oplus0=A;(经典 coding 小技巧)

    # 同或 XNOR

    同或 —— 相同时输出 1,不同时输出 0

    同或

    # 逻辑代数的公式和定理

    # 基本公式

    逻辑代数的基本公式1

    逻辑代数的基本公式2

    # 常考公式 7/17 分配律

    A(B+C)=AB+ACA+BC=(A+B)(A+C)A(B +C) = AB + AC\\ A + BC = (A +B)(A +C)

    用途:将 “混杂” 在一起的变量分离,将逻辑函数表达式化成 “与或” 形式和 “或与” 形式;在后面逻辑函数的 “最小项之和” 和 “最大项之积” 表示时非常有用

    # 常考公式 8/18 反演律(德・摩根定律)

    (AB)=A+B(A+B)=AB(AB)' = A' + B' (A + B) ' = A'B'

    记忆方法:“与的非等于非的或,或的非等于非的与”;

    门电路的角度上,就是:

    • 负或门 = 与非门
    • 负与门 = 或非门

    与在概率论课程中学习的德摩根定律一致,即 “交的反等于反的并,并的反等于反的交”

    用途:实现与运算和或运算之间的转换,在后面对函数式进行特定形式的变换时非常有用

    # 基本定理

    # 代入定理

    在任何一个包含 A 的逻辑等式中,若以另外一个逻辑式代入式中 A 的位置,则等式依然成立;

    # 反演定理

    对任一逻辑式 Y ,在求解它的反时(即已知 Y 求解 Y’),只需要在保证运算优先次序不变的原则下,将其中所有的 \cdot 换成 ++++ 换成 \cdot ,原变量换成反变量,反变量换成原变量(不属于单个变量上的反号不动),1 变成 0,0 变成 1,得到的结果就是 Y’;

    # 对偶定理

    若两逻辑表达式相等,则它们的对偶式也相等;

    所谓的对偶式指的是在保持运算优先顺序不变的原则下,将表达式中所有的 \cdot 换成 ++++ 换成 \cdot ,0 换成 1,1 换成 0;

    注意与反演定理的区别!变量不取反!

    主要应用于证明题当中,证明两个形式较复杂的逻辑式相等,可以尝试:

    • 两边同时取对偶式,证明它们的对偶式相等
    • 对等号某一侧表达式先取对偶式,对对偶式进行化简之后,再取对偶就可以得到等号另一侧表达式

    # 逻辑函数及其表示方法

    以逻辑变量为输入,运算结果为输出,则输入变量值确定以后,输出的取值也随之而定,因此输入 — 输出之间是一种函数关系,称作逻辑函数;

    在我们学习的二值逻辑 (布尔逻辑) 中,输入变量和输出变量只有 0 和 1 两种状态;

    在求解实际应用问题中的逻辑函数关系时,首先必须做的事情是将具体事物抽象为逻辑变量并规定其不同状态(0 或 1),这是很重要的解题习惯,类似于在概率论答题时首先要记不同的事件为 A、B……;这里只研究现有的抽象后的逻辑函数式;

    # 不同表示方法

    # 真值表

    遍历输入变量所有取值下的输出值,列成表格

    # 逻辑函数式

    将输出写成输入变量的与、或、非等逻辑运算的组合式

    # 逻辑图

    逻辑函数式用图形符号表示

    # 波形图

    输入变量所有取值可能与对应输出的时域波形

    # 卡诺图

    # 计算机软件中的描述方式

    例如 Verilog 语言

    # 不同表示方法间的转换

    # 真值表 -> 逻辑函数式 (重点)

    (逻辑函数的最小项之和形式)

    1. 找出真值表中使逻辑函数输出 Y = 1 的输入变量取值的组合即使 Y = 1 的行;
    2. 使 Y = 1 的每组输入变量取值的组合(即每一行)对应一个乘积项,取值为 1 写入原变量,取值为 0 写入反变量;
    3. 将各行对应的乘积项相加,即最终结果为 “与或” 形式;

    # 逻辑函数式 <-> 逻辑图

    # 逻辑函数式 -> 逻辑图

    用图形符号代替逻辑函数式中的逻辑运算符;

    # 逻辑图 -> 逻辑函数式

    从输入到输出逐级写出每个图形符号对应的逻辑运算式;

    # 真值表 <-> 波形图

    # 真值表 -> 波形图

    将所有输入变量与对应输出变量的取值依次排列,以时间为横轴;

    # 波形图 -> 真值表

    在波形图中找出每个时间段里输入变量与输出变量的取值列成表格( 对于 NN 个输入变量要对应 2N2^N 个组合即 2N2^N 行)

    # 标准 / 规范 形式

    # DNF/SOP

    别名:

    • 最小项标准式
    • 析取范式
    • 积之和范式
    • 与或范式
    • Disjunctive Normal Form (DNF)
    • Sum of Product (SOP)

    任何一个逻辑函数式都可以化为若干个最小项之和的形式

    # 最小项 m 的定义:

    m 是乘积项,包含 n 个因子,n 个变量均以原变量或反变量的形式在 m 中出现一次;

    # 最小项的特点

    • 全体最小项之和为 1
    • 各个最小项之间是互斥的,即任何两个最小项之积为 0 ;换句话说,对于任何一个特定的输入变量取值,有且仅有一个最小项为 1;
    • 两个相邻的最小项之和可以合并,消去一对因子,只留下公共因子;

    这里的 “相邻” 不是按照自然二进制数编号后的相邻,而是指逻辑上的相邻,规定仅一个变量不同的最小项彼此称为相邻;

    例如 ABCA' B' CABCA'BC'ABCA'BCm1m_1m2m_2m3m_3m1m_1m2m_2 不是相邻,m1m_1m3m_3 才是相邻;m1m_1m3m_3 可以合并消去因子 B+BB+B'

    # 转化为 DNF

    事实上最小项就是在列写 n 个输入变量的逻辑函数的真值表时每一行对应的输入变量的乘积项,取值为 1 写入原变量,取值为 0 写入反变量;使最小项取值为 1 的变量取值,按照 n 位自然二进制数排列后对应的十进制数就是这个最小项的编号

    # CNF/POS

    别名:

    • 最大项标准式
    • 合取范式
    • 和之积范式
    • 或与范式
    • conjunctive normal form (CNF)
    • Product of Sum (POS)

    任何一个逻辑函数式都可以化为若干个最大项之积的形式

    # 最大项 M 的定义

    M 是相加项,包含 n 个因子,n 个变量均以原变量和反变量的形式在 M 中出现一次;可以把最大项看作是对每一个最小项应用反演定理,因此,最大项的编号原则是反的,即原变量记为 0,反变量记为 1,n 位自然二进制数排列后对应的十进制数为编号;

    例如 A+B+C′ 的编号是 001 即 M1 ;

    (记忆方法:对于与运算即乘积项,得 1 “容易”,即得 1 时输入变量组唯一确定;对于或运算即相加项,得 0 “容易”,即得 0 时输入变量组唯一确定;)

    # 最大项的特点

    • 全体最大项之积为 0 ;
    • 任何两个最大项之和为 1 ;换句话说,对于任何一个特定的输入变量取值,有且仅有一个最小项为 0;
    • 两个相邻的最大项之积保留公共变量

    这里的相邻和前面最小项的相邻一致,都是指逻辑上的相邻,即只有一个变量不同。

    例如 A+B+CA+B+C'A+B+CA+B'+C' 为相邻,即 M1M_1M3M_3 相邻,两者的乘积项即相与的结果是 A+CA+C'

    # 转化为 CNF

    当给定的逻辑函数式不是标准的最大项之积的形式时,一般不断地利用 AA=0A\cdot A'=0 补全缺少的因子,并结合分配律 A+BC=(A+B)(A+C)A + BC = (A +B)(A +C) 将与或形式转化为或与形式;

    # 最小项与最大项的关系

    根据最大项和最小项的定义以及德・摩根定理,易推导得到:

    Mi=miM_i=m'_i

    如果将逻辑函数写成最小项之和的形式:

    iR1mi\sum_{i\in R_1} m_i

    再将其写成最大项之积的形式:

    iR2Mi\prod_{i\in R_2} M_i

    则两者的下标编号集合 R1R_1R2R_2 是互补的。

    所以很多时候不需要直接展开求解最大项之积,先求解最小项之和更方便

    # 逻辑函数的化简

    # 逻辑函数的化简的目标

    逻辑函数的最小项之和虽然是通用的标准形式,但不一定就是最简的;在实际数字逻辑电路设计时,需要考虑在逻辑功能相同的条件下使用尽可能简单的电路,即电子器件数量足够少;

    事实上,没有绝对的最简,由于我们习惯性将逻辑函数写成 最小项之和 即 与或形式 ,因此对于 与或形式 的逻辑函数,包含的乘积项最少,且每个乘积项里的因子也不能再减少时,称其为最简形式即最简与或形式;其他形式的最简同理;

    常考最简式包括:

    • 最简与或式
    • 最简或与式
    • 最简与非 — 与非式
    • 最简或非 — 或非式
    • 最简与或非式

    实际的逻辑电路设计可能会受到电子器件种类与数量的限制,因此可能要进行变换;直接用卡诺图化简得到的最简与或式并不一定是设计时需要的逻辑函数形式,所以还会需要其他最简形式,例如:

    • 只能用或非门实现:与非 — 与非式
    • 只能用或非门实现:或非 — 或非式

    # 公式化简法

    • 使用前面学习的基本公式和常用公式
    • 没有固定的步骤,
    • 难以确定结果正确与否
    • 不通用但有时很快速好用

    对公式化简法,要求掌握但不必深入研究,熟悉教材上的例题和课后题即可;

    一个可能会用到公式化简法的情况是,当逻辑函数含有变量数很多时,即很难直接采用卡诺图去进行化简时,一般题目的设计会使解题时经常需要用 AB+AC+BCD=AB+ACAB + A'C + BCD = AB + A'C 这个公式来消去多余的变量,接着再用卡诺图去进行化简

    # ⭐️卡诺图化简法 (重点)

    # 卡诺图 (Karnaugh)

    2n2^n 个小方块分别代表 n 变量的所有最小项,并将它们排列成矩阵,而且使几何位置相邻的两个最小项在逻辑上也是相邻的(只有一个变量不同),就得到表示 n 变量全部最小项的卡诺图;

    # 卡诺图的绘制

    横纵轴不是按照自然二进制数排列,按照格雷码排列,目的是为了使逻辑上相邻对应几何上相邻,保证相邻的方块只有一个变量不同

    卡诺图的绘制

    记忆方法:

    • 增加一个变量看作是折纸一样翻折
    • 轴对称位置上的最小项也满足相邻
    • 把卡诺图看作是闭合的图形 (横纵轴都是循环的)

    教材上的卡诺图并不是唯一的,只要能够保证逻辑对称性几何对称性一致即可,最好去理解卡诺图绘制的过程

    无关项可以用 x 表示,表示可圈起来也可不圈起来

    # 卡诺图化简法的本质

    将最小项之和用图形描述,将逻辑上的相邻表现为几何图形的相邻,通过合并相邻的几何图形,合并后对应仅保留公共因子的最小项,最终结果就是最简与或式

    # 卡诺图化简逻辑函数的步骤

    # 最简与或式

    若只给出逻辑函数则从步骤 1 开始,若给出卡诺图则直接从步骤 2 开始

    1. 将逻辑函数写成最小项之和的形式
    2. 根据最小项之和绘制对应的卡诺图,逻辑函数式中含有的最小项的对应位置填 1,其余的位置填 0
    3. 合并相邻的最小项,对应最终结果为 最简与或式
    • 相邻表示逻辑相邻,轴对称位置上的方块也可合并
    • 合并时,每个方块可以重复使用,因为 A+A=AA+A=A
    • 合并的原则是乘积项的数目最少,每个乘积项的因子最少,即画的圈要尽量大(消去尽量多的不同的因子),画圈的数目尽量少
    • 结果不唯一,但乘积项的数目和每个乘积项的因子个数一致(这也是检查答案是否正确的依据)

    # 最简或与式

    1. 对原逻辑函数 YY 取反得到 YY'
    2. YY' 采用上述卡诺图求最简与或式步骤得到 YY' 的 最简与或式
    3. YY' 的 最简与或式 取反变换得到的 或与式 就是原逻辑函数 YY 的 最简或与式

    或者直接:

    1. 采用上述卡诺图求最简与或式步骤,但是合并为 0 的方块而不是 1 ,得到 YY' 的最简与或式
    2. YY' 的 最简与或式 取反变换得到的 或与式 就是原逻辑函数 YY 的 最简或与式

    # 最简与或非形式

    与最简或与式实际上是等价的,按照求最简或与式步骤即可,只是最后一步取反直接加上取反符号不需要再变换为或与式

    # 最简与非 — 与非式

    将得到的最简与或式取两次反,即逆向利用德・摩根定理

    # 最简或非 — 或非形式

    将与或非形式中的每个乘积项再利用德・摩根定理,即变为或非形式

    # 含有无关项的卡诺图化简

    一般第二章的题目会直接给出无关项,例如:

    • $ (...) + (...) +…… = 0 $
    • d()=0\sum d(…)= 0 (无关项 集合用 dd 表示)

    后面在基于实际应用设计逻辑电路时应自行根据实际情况来分析确定无关项

    1. 根据题意确定无关项
    2. 绘制卡诺图时,无关项的方块用 × 表示
    3. 化简时根据需要,将无关项填入 1 或 0 ,原则依然是是使化简后的项数最少,每项的因子最少,即画的圈最大,圈的个数最少
    Edited on Views times
    \ No newline at end of file diff --git a/ie/de/coding/index.html b/ie/de/coding/index.html index c96309ef..d1ce505f 100644 --- a/ie/de/coding/index.html +++ b/ie/de/coding/index.html @@ -1 +1 @@ -码制 - 数字逻辑电路 - 电子信息 | Balloon Party = 風船のパーティー = fuusen no party

    # 编码的概念

    广义的编码指的就是将信息从一种形式转换为另一种规范的形式;

    对于数字电路,就是将现实世界中的各种物理信息转换为离散的电压序列,来表示、处理和分析信息;

    事实上,进位计数制也是一种编码,用不同进制数来表示 “量”;

    数制码制分开介绍,首先单独讲解数制是因为数制的一个重要的特点是其编码后能够参与算术运算;而一般来说码制是对非数值信息的表示;

    # 二进制编码

    # 无符号

    只能表示 0 和正数

    # 有符号

    (有符号)二进制数:在前面增加一个符号位,0 表示正数,1 表示负数;引入符号位后,一个二进制数的最高位是符号位,符号位的 0 / 1 不代表大小,只是代表正负

    有符号二进制数的三种编码表示:

    • 原码 ——Sign Magnitude Representation

      • 正数和 0:普通的二进制数表示
      • 负数:绝对值的二进制数表示加上负号
    • 反码 ——One's Complement

      • 正数和 0: = 原码
      • 负数:原码所有数字位取反
    • 补码 ——Two's Complement

      • 正数和 0: = 原码
      • 负数:= 反码 + 1

    nn 位补码 Bn1Bn2Bn3...B0B_{n-1} B_{n-2} B_{n-3}... B_{0} 统一表示:

    Bn1(2n1)+Bn22n2+Bn32n3+...+B020B_{n-1}\cdot(-2^{n-1})+B_{n-2}\cdot2^{n-2}+B_{n-3}\cdot2^{n-3}+...+B_{0}\cdot2^{0}

    nn 位原码 Bn1Bn2Bn3...B0B_{n-1} B_{n-2} B_{n-3}... B_{0} 统一表示:

    Bn12n1+Bn22n2+Bn32n3+...+B020B_{n-1}\cdot2^{n-1}+B_{n-2}\cdot2^{n-2}+B_{n-3}\cdot2^{n-3}+...+B_{0}\cdot2^{0}

    补码的补码即为原码

    反码的加法:若最高位产生进位,需要循环进位,即最低位 “进一位” 加一

    关于原码、反码、补码的易错点:

    • 反码取反是指对数字位取反,不改变符号位!
      • 例如 -7 即 1111 的反码是 1000,不是 0000 !
    • 补码是不对称的,例如补码为 1000 代表 - 8 ,而不代表 - 0 ! 即同样位数的二进制补码,负数会比正数 “多一个”;
    • 补码的 “加 1” 指的是在最低位加 1 的二进制加法运算,不是指数值 1,即如果含有小数部分,1 是加在整体的最右边;

    利用二进制补码运算求解带符号数的加法运算:

    1. 分析需要使用多少个数字位;(培养思维习惯!不能忽略!)
    2. 用带符号位的二进制数表示,求解每个二进制数的补码;
    3. 对补码进行加法运算,再对结果求补码,即为最终结果;

    注意最后一步,对补码进行加法运算时,数字位最高位产生的进位要进到符号位上去!(教材中描述的 “舍弃产生的进位” 指的是舍弃符号位产生的进位)

    # 无符号数:原码 -> 补码

    # 方法一:

    原码全部取反,再 +1

    # 方法二:

    从右向左找到第一个 1 ,这个 1 前面的所有位取反

    若没有 1 ,也就是 0,原码 = 补码

    # 负数:补码 -> 原码

    这里只讨论负数的转换,正数不需要转换

    # 方法一:

    补码 -1 ,再取反

    # 方法二:

    从右向左找到第一个 1 ,这个 1 前面的位取反(符号位也要变为 0

    若没有 1 ,也就是 0,原码 = 补码

    # 十进制数的二进制代码

    十进制代码用四位二进制数(最多可表示 24=162^4=16 个)来表示 0~9

    常见十进制代码

    8421、2421、5211 每个数字指的是每个位的权;

    # BCD/8421

    8421 码也叫 BCD 码

    余 3 码是在 BCD 码即 8421 码的基础上 +3 ,即用 3~12 对应的二进制数来表示 0~9;

    余 3 循环码是取格雷码中的 3~12;

    # 格雷码

    教材中的格雷码特指的是四位格雷码,即用四位代码来表示 0 ~ 15 共十六种状态;

    # 编码特点:

    • 每一位的状态变化都按一定的顺序循环:0110、00111100、0000111111110000、0000000011111111(四位 16 个,只循环半个周期)(其他位数的格雷码表依次类推,自右向左每一位状态循环中连续的 0 和 1 数目加倍)
    • 按表中顺序变化时,相邻的代码只有一位改变状态,这是格雷码最大的特点和优点;(应用:减少代码转换过程中的过渡噪声)

    # 四维格雷码表

    需要熟练掌握!重点是熟悉编码的规则和特点

    四维格雷码表

    # 二进制码与格雷码的映射

    • 最高位相同
    • 二进制码从左到右,将相邻 A B 两位相加(无进位),得到格雷码对应的 B 位

    # 格雷码应用

    旋转编码器(Rotate Encoder)

    # ASCII 码

    美国信息交换标准代码,用 7 位二进制代码(共 128 个)来表示信息,广泛应用于计算机和通信领域中;

    ascii

    在线查询:

    • 菜鸟教程
    • OSchina

    # 校验

    • 奇校验
      • 在二进制串的末尾增加一位 10 ,使得整个新二进制的 1 的个数为奇数
    • 偶校验
      • 在二进制串的末尾增加一位 10 ,使得整个新二进制的 1 的个数为偶数
    Edited on Views times
    \ No newline at end of file +码制 - 数字逻辑电路 - 电子信息 | Balloon Party = 風船のパーティー = fuusen no party

    # 编码的概念

    广义的编码指的就是将信息从一种形式转换为另一种规范的形式;

    对于数字电路,就是将现实世界中的各种物理信息转换为离散的电压序列,来表示、处理和分析信息;

    事实上,进位计数制也是一种编码,用不同进制数来表示 “量”;

    数制码制分开介绍,首先单独讲解数制是因为数制的一个重要的特点是其编码后能够参与算术运算;而一般来说码制是对非数值信息的表示;

    # 二进制编码

    # 无符号

    只能表示 0 和正数

    # 有符号

    (有符号)二进制数:在前面增加一个符号位,0 表示正数,1 表示负数;引入符号位后,一个二进制数的最高位是符号位,符号位的 0 / 1 不代表大小,只是代表正负

    有符号二进制数的三种编码表示:

    • 原码 ——Sign Magnitude Representation

      • 正数和 0:普通的二进制数表示
      • 负数:绝对值的二进制数表示加上负号
    • 反码 ——One's Complement

      • 正数和 0: = 原码
      • 负数:原码所有数字位取反
    • 补码 ——Two's Complement

      • 正数和 0: = 原码
      • 负数:= 反码 + 1

    nn 位补码 Bn1Bn2Bn3...B0B_{n-1} B_{n-2} B_{n-3}... B_{0} 统一表示:

    Bn1(2n1)+Bn22n2+Bn32n3+...+B020B_{n-1}\cdot(-2^{n-1})+B_{n-2}\cdot2^{n-2}+B_{n-3}\cdot2^{n-3}+...+B_{0}\cdot2^{0}

    nn 位原码 Bn1Bn2Bn3...B0B_{n-1} B_{n-2} B_{n-3}... B_{0} 统一表示:

    Bn12n1+Bn22n2+Bn32n3+...+B020B_{n-1}\cdot2^{n-1}+B_{n-2}\cdot2^{n-2}+B_{n-3}\cdot2^{n-3}+...+B_{0}\cdot2^{0}

    补码的补码即为原码

    反码的加法:若最高位产生进位,需要循环进位,即最低位 “进一位” 加一

    关于原码、反码、补码的易错点:

    • 反码取反是指对数字位取反,不改变符号位!
      • 例如 -7 即 1111 的反码是 1000,不是 0000 !
    • 补码是不对称的,例如补码为 1000 代表 - 8 ,而不代表 - 0 ! 即同样位数的二进制补码,负数会比正数 “多一个”;
    • 补码的 “加 1” 指的是在最低位加 1 的二进制加法运算,不是指数值 1,即如果含有小数部分,1 是加在整体的最右边;

    利用二进制补码运算求解带符号数的加法运算:

    1. 分析需要使用多少个数字位;(培养思维习惯!不能忽略!)
    2. 用带符号位的二进制数表示,求解每个二进制数的补码;
    3. 对补码进行加法运算,再对结果求补码,即为最终结果;

    注意最后一步,对补码进行加法运算时,数字位最高位产生的进位要进到符号位上去!(教材中描述的 “舍弃产生的进位” 指的是舍弃符号位产生的进位)

    # 无符号数:原码 -> 补码

    # 方法一:

    原码全部取反,再 +1

    # 方法二:

    从右向左找到第一个 1 ,这个 1 前面的所有位取反

    若没有 1 ,也就是 0,原码 = 补码

    # 负数:补码 -> 原码

    这里只讨论负数的转换,正数不需要转换

    # 方法一:

    补码 -1 ,再取反

    # 方法二:

    从右向左找到第一个 1 ,这个 1 前面的位取反(符号位也要变为 0

    若没有 1 ,也就是 0,原码 = 补码

    # 十进制数的二进制代码

    十进制代码用四位二进制数(最多可表示 24=162^4=16 个)来表示 0~9

    常见十进制代码

    8421、2421、5211 每个数字指的是每个位的权;

    # BCD/8421

    8421 码也叫 BCD 码

    余 3 码是在 BCD 码即 8421 码的基础上 +3 ,即用 3~12 对应的二进制数来表示 0~9;

    余 3 循环码是取格雷码中的 3~12;

    # 格雷码

    教材中的格雷码特指的是四位格雷码,即用四位代码来表示 0 ~ 15 共十六种状态;

    # 编码特点:

    • 每一位的状态变化都按一定的顺序循环:0110、00111100、0000111111110000、0000000011111111(四位 16 个,只循环半个周期)(其他位数的格雷码表依次类推,自右向左每一位状态循环中连续的 0 和 1 数目加倍)
    • 按表中顺序变化时,相邻的代码只有一位改变状态,这是格雷码最大的特点和优点;(应用:减少代码转换过程中的过渡噪声)

    # 四维格雷码表

    需要熟练掌握!重点是熟悉编码的规则和特点

    四维格雷码表

    # 二进制码与格雷码的映射

    • 最高位相同
    • 二进制码从左到右,将相邻 A B 两位相加(无进位),得到格雷码对应的 B 位

    # 格雷码应用

    旋转编码器(Rotate Encoder)

    # ASCII 码

    美国信息交换标准代码,用 7 位二进制代码(共 128 个)来表示信息,广泛应用于计算机和通信领域中;

    ascii

    在线查询:

    • 菜鸟教程
    • OSchina

    # 校验

    • 奇校验
      • 在二进制串的末尾增加一位 10 ,使得整个新二进制的 1 的个数为奇数
    • 偶校验
      • 在二进制串的末尾增加一位 10 ,使得整个新二进制的 1 的个数为偶数
    Edited on Views times
    \ No newline at end of file diff --git a/ie/de/combinatorial-logic/index.html b/ie/de/combinatorial-logic/index.html index 05426bff..f6ba27a2 100644 --- a/ie/de/combinatorial-logic/index.html +++ b/ie/de/combinatorial-logic/index.html @@ -1 +1 @@ -组合逻辑电路 - 数字逻辑电路 - 电子信息 | Balloon Party = 風船のパーティー = fuusen no party

    基本框架

    • 数字电路
      • 组合逻辑电路
        • 以门电路为基本单元
      • 时序逻辑电路
        • 以触发器为基本单元

    # 组合逻辑电路基础

    Programmable Logic Device : PLD —— 可编程逻辑器件

    # 特点

    功能上,任意时刻的输出仅仅取决于该时刻的输入,与电路原来的状态无关;

    结构上,组合逻辑电路不含有存储单元(记忆单元),只有从输入到输出的通路,没有反馈环路

    # 分析方法

    给定一个组合逻辑电路,分析其电路逻辑功能:

    1. 根据电路图,从输入到输出逐级分析,得到输出与输入的逻辑函数式;
    2. 对得到的逻辑函数式进行化简、形式变换或列出真值表,最后分析其逻辑功能;

    # 设计方法

    • Step 1:逻辑抽象;
      • 分析因果关系,确定输入 / 输出变量;
      • 定义逻辑状态的含意(赋值);
        • 输入和输出都有可能需要多个变量
        • 一个对象有 kk 种状态,则可能需要 n=log2kn=\lceil\log_2k\rceil 个信号来表示;若 2n>k2^n\gt k,多余的项为无关项中的约束项,可以不写在真值表中
      • 列出真值表;
    • Step 2:将真值表转换为逻辑函数式;
    • Step 3:选定器件类型;
      • 是否有器件类型限制,例如 “只能用与非门” 这类条件;
      • 是否有指定使用的中规模器件,例如 “使用数据选择器” 这类条件;
    • Step 4:根据所选器件,对逻辑式进行化简或形式变换;(或进行相应的描述(PLD))
      • 与非门 —— 与非 — 与非式
      • 或非门 —— 或非 — 或非式
    • Step 5:画出逻辑电路连接图

    # 加法器

    # 半加法器

    Basic Adders

    半加法器

    ABC_Σ\Sigma
    0000
    0101
    1001
    1110

    Cout=ABΣ=AB\begin{aligned} C_{out}&=AB\\ \Sigma&=A\oplus B \end{aligned}

    门电路实现:

    半加法器门电路实现

    # 全加法器

    Full-adder

    全加法器

    ABC_C_Σ\Sigma
    00000
    00101
    01001
    01110
    10001
    10110
    11010
    11111

    Cout=AB+(AB)CinΣ=ABCin\begin{aligned} C_{out}&=AB+(A\oplus B)C_{in}\\ \Sigma&=A\oplus B\oplus C_{in} \end{aligned}

    输出写为这个形式,与半加法器的输出对比,全加法器可以用 2 个半加法器级联和 1 个或门实现

    门电路实现全加法器:

    门电路实现全加法器

    半加法器实现全加法器:

    半加法器实现全加法器

    # 二进制加法器

    Parallel Binary Adders

    全加法器级联实现,利用了模块化设计的思想

    模块化思想,或者说集成化思想,是人们不断积累的设计经验。有很多场合下的逻辑问题是具有共性的,为了更方便地设计电路,通常可以将一些具有特定功能、解决某类问题的电路封装打包即集成化,组成一些常用的组合逻辑电路模块(中规模集成器件);在设计大规模的集成电路时,就可以调用这些典型的电路模块而避免使用复杂且数量巨大的分立的门电路

    # 74LS283

    4-bit parallel adder, 4 位二进制加法器

    4位并行加法器

    2 个 74LS283 级联可以得到 8 位二进制加法器(8-bit parallel adder):

    全加法器级联实现4位并行加法器

    4 个全加法器级联实现,但是这样上一级的 CoutC_{out} 到下一级 CinC_{in} 会存在延迟,级联越多,这样的串行部分产生的延迟越多

    全加法器中,令:

    Cg=ABCp=A+B\begin{aligned} C_{g}&=AB\\ C_{p}&=A+B \end{aligned}

    则有:

    Cout=Cg+CpCinC_{out}=C_{g}+C_{p}C_{in}

    look-ahead-adder

    # 比较器

    comparator

    简单的 1、2 位比较器直接利用异或门可以实现。

    位数比较多时,比如 4 位比较器,从高位开始比较,若出现不一样则剩余低位可以不必比较,直接输出结果(大于、小于)否则直到所有位比较完,并且都相等,则输出 “相等”

    # 74LS85

    4 位比较器

    74LS85

    • 8 位数据输入,按从 3 到 0 的高位到低位顺序比较
    • 3 位级联输入,当本次比较结果为相等时输出级联输入的结果
    • 3 位比较结果输出,高电平有效

    注意 3 个级联输入: A<BA<BA=BA=BA>BA>B 。 这些输入允许几个比较器级联起来,用来比较大于 4 位数字的数。

    • 低位比较器的 A<BA<BA=BA =BA>BA>B 输出与对应的下一个较高位比较器的级联输入相连接;
    • 最低位比较器在 A=BA=B 输入端又必须接高电平,在 A<BA<BA>BA>B 输入端又必须接低电平;

    # 扩展 8 位比较器

    2 个 74HC85 扩展为 8 位比较器:

    74LS85扩展8位比较器

    # 扩展 16 位比较器

    5 个 74LS85 扩展为 16 位比较器:

    74LS85扩展16位比较器

    # 译码器 (n:m)

    decoder

    # 74HC154(4:16)

    4 位译码器,4 线 - 16 线译码器,也是 4 线 - 16 线解多路复用器

    • 4 位输入
    • 2 位片选,当且仅当 2 位都为低位时才正常工作
    • 16 位输出,低电平有效

    74HC154

    # 扩展

    74HC54 扩展实现 5 位输入译码器:

    74HC54实现5位输入译码器

    # 74HC42

    BCD - 十进制译码器,把每 一个 BCD 码 (8421 码) 转换成 10 个十进制数中的一个数所表示的输出

    • 4 位输入,A0,A1,A2,A3A_0,A_1,A_2,A_3 分别表示 1 2 4 8
    • 10 位输出, 低电平有效

    # 74LS47

    BCD-7 段译码器 / 驱动器,它把一个 BCD 输入进行译码,并驱动 7 段显示器

    • 4 位输入,A0,A1,A2,A3A_0,A_1,A_2,A_3 分别表示 1 2 4 8 ,高电平有效
    • 3 位控制位
      • LT\overline{LT}
      • RBI\overline{RBI}
      • BI/RNO\overline{BI/RNO}
    • 7 位输出,低电平有效

    74LS48 与它的区别在于输出的 7 位没有取非,它的输出是高电平有效

    # 试灯

    试灯,点亮所有灯管,用来检测是否有段已经烧坏

    # 灭零

    LTRBI=10\overline{LT}\ \overline{RBI}=10灭零。把数字前面的 0 熄灭称为头部灭零,把数字尾端的 0 熄灭称为尾部灭零。只是把不需要的零熄灭!

    # 头部灭零

    头部灭零

    # 尾部灭零

    尾部灭零

    # 熄灯

    LT=0\overline{LT}=0 时熄灯

    # 2-4 译码器

    # 3-8 译码器

    3 线 - 8 线译码器

    # ⭐️74LS138(3:8)

    • 3 位输入位为选择位
    • 8 位输出位
    • 3 位片选位

    # 5-32 译码器

    可以用 1 个 74LS139 和 4 个 74LS138 实现

    # 编码器 (n:m)

    数字电路中的编码器是在已经实现了非电信号到电信号的转换、模拟信号到数字信号的转换之后对信息进行编码,即此时输入的信号是一组 高、低电平 (1 和 0 );

    编码器的功能就是根据输入的每一个高、低电平来输出其对应的二进制代码

    输入 nn电平信号,至少需要 log2n\lceil\log_2n\rceil 个输出端口,即 log2n\lceil\log_2n\rceil 位二进制编码

    # 普通编码器

    普通编码器规定了任意时刻有且只有一个有效输入信号,作为默认给定的约束条件

    # 十进制 - BCD 编码器

    十进制-BCD编码器

    # 8/3 普通编码器

    普通8:3编码器

    真值表:(不包含无关项)

    I0I_0I1I_1I2I_2I3I_3I4I_4I5I_5I6I_6I7I_7Y2Y_2Y1Y_1Y0Y_0
    1
    11
    11
    111
    11
    111
    111
    1111

    因为任一时刻只能有一个输入为 1,因此只需要将输入为 1 的变量保留,其余的变量无关(相当于在卡诺图中 x 的方块填入 1 合并消去无关变量)化简为最简与或式:

    Y2=I4+I5+I6+I7Y1=I2+I3+I6+I7Y0=I1+I3+I5+I7Y_2=I_4+I_5+I_6+I_7\\ Y_1=I_2+I_3+I_6+I_7\\ Y_0=I_1+I_3+I_5+I_7

    从而可画出逻辑电路图:

    普通8:3编码器逻辑电路图

    此设计存在的问题:

    • 不能有多个信号同时输入
    • 也不能没有信号输入(因为根据我们的约束条件,00000000 也属于无关项);

    因此我们以上的设计只是一个思路的训练,最终的设计结果并不能直接应用到实际当中

    # 优先编码器

    • 允许多个信号同时输入
    • 编码时存在优先顺序

    # 8/3 优先编码器

    规定:

    • 高电平输入有效,即输入 1 表示对此输入信号编码,输出的 Y2Y1Y0Y_2Y_1Y_0 即为 0 ~ 7 对应的三位二进制代码 000~111;
    • 二进制数大的优先级高,即 I7I_7 优先级最高,I0I_0 最低;

    真值表:(不包含无关项)

    I0I_0I1I_1I2I_2I3I_3I4I_4I5I_5I6I_6I7I_7Y2Y_2Y1Y_1Y0Y_0
    xxxxxxx1111
    xxxxxx111
    xxxxx111
    xxxx11
    xxx111
    xx11
    x11
    1

    此真值表不完整!当前只有 2812^8–1 行,缺少了一行 00000000 !

    这时存在的问题是,输入全为 0 时输出为 I0I_0 表示的状态,无法表示输入全为 0 的状态。在实际应用的 8/3 优先编码器中,还附加了其他的输入输出端口,以便于片选和拓展。

    附加了其他的输入输出端口进行片选和拓展的最主要目还是厂家希望产品应用更广泛,同种型号芯片可以实现不同功能

    # 75LS148(8:3)

    • 8 位输入,低电平有效
    • 1 位片选位 EI\overline{EI} ,低电平有效,正常工作
    • 3 位输出编码,低电平有效⚠️
    • 2 位输出有效位
      • EO\overline{EO}EI\overline{EI} 有效且所有输入都无效时为 0
      • GS\overline{GS}EI\overline{EI} 无效且任何一个输入有效时为 0

    功能表:

    EIEI'I0I_0'I1I_1'I2I_2'I3I_3'I4I_4'I5I_5'I6I_6'I7I_7'Y2Y_2'Y1Y_1'Y0Y_0'EOEO'GSGS'
    1xxxxxxxx11111
    111111111111
    xxxxxxx1
    xxxxxx111
    xxxxx1111
    xxxx111111
    xxx111111
    xx11111111
    x111111111
    11111111111

    # 扩展

    2 个 74LS148 实现 16 线 - 4 线编码器:

    74LS148实现16线-4线编码器

    # 多路复用器 (n:1)

    (Mux)多路复用器(Multiplexer)或数据选择器(Data Selector),能够实现并行数据到串行数据的转化。

    包括:

    • 数据选择位(低电平有效)
    • 数据输入位
    • 数据输出位

    此时数据输入输出位不必再关心是否低电平有效

    # 4 路复用器

    • 2 位数据选择位 S0,S1\overline{S_0},\overline{S_1} ,低电平有效
    • 4 位数据输入位 D0,D1,D2,D3D_0,D_1,D_2,D_3
    • 1 位数据输出位 YY

    Y=S1S0D0+S1S0D1+S1S0D0+S1S0D1Y=\overline{S_1S_0}D_0+\overline{S_1}S_0D_1+S_1\overline{S_0}D_0+S_1S_0D_1

    4位多路复用器

    # 8 路复用器

    Y=S3S2S1S0D0+S3S2S1S0D1+S3S2S1S0D2+S3S2S1S0D3\begin{aligned} Y&=\overline{S_3S_2S_1S_0}D_0+\overline{S_3S_2S_1}S_0D_1+\overline{S_3S_2}S_1\overline{S_0}D_2+\overline{S_3S_2S_1S_0}D_3 \end{aligned}

    # 74LS151

    # 扩展 16 路复用器

    2 个 74LS151 扩展实现 16 路复用器

    74LS151扩展实现16路复用器

    # 74LS153

    双 4 选 1 数据选择器,也就是 2 组 4 选 1 共用同一个选择输入和同一个片选输入。

    # 74LS157

    四 2 选 1 数据选择器,也就是 4 组 2 选 1 共用同一个选择输入和同一个片选输入。

    • 1 位片选输入,电平有效
    • 1 位选择输入
    • 8 位数据输入
    • 4 位数据输出

    # 7 段显示多路复用器

    把 BCD 数字分配到一个 7 段显示器上的一个简单方法

    # ⭐️扩展

    2 个 4 选 1 扩展为 8 选 1

    1 个 4 选 1 + 门电路扩展为 5 选 1

    5 个 4 选 1 扩展为 16 选 1 (⭐️树状扩展)

    # ⭐️实现门电路

    # 解多路复用器 (1:n)

    解多路分配器(Demultiplexer)或数据分配器()

    数据选择器的逆

    # 1 线 - 4 线多路分配器

    1线-4线多路分配器

    1线-4线多路分配器波形图

    # 74HC154

    4 位译码器 74HC154 可以作为 4 线 - 16 线解多路复用器

    74HC154作4线-16线解多路复用器

    # 传输时延与竞争冒险

    • 毛刺 glitch
    • 竞争 race
    • 冒险 hazard

    时延导致竞争的存在,竞争可能产生毛刺,竞争导致了逻辑误判,就是冒险

    # 消除冒险的方法

    # 滤波电容

    # 冗余项

    \ No newline at end of file +组合逻辑电路 - 数字逻辑电路 - 电子信息 | Balloon Party = 風船のパーティー = fuusen no party

    基本框架

    • 数字电路
      • 组合逻辑电路
        • 以门电路为基本单元
      • 时序逻辑电路
        • 以触发器为基本单元

    # 组合逻辑电路基础

    Programmable Logic Device : PLD —— 可编程逻辑器件

    # 特点

    功能上,任意时刻的输出仅仅取决于该时刻的输入,与电路原来的状态无关;

    结构上,组合逻辑电路不含有存储单元(记忆单元),只有从输入到输出的通路,没有反馈环路

    # 分析方法

    给定一个组合逻辑电路,分析其电路逻辑功能:

    1. 根据电路图,从输入到输出逐级分析,得到输出与输入的逻辑函数式;
    2. 对得到的逻辑函数式进行化简、形式变换或列出真值表,最后分析其逻辑功能;

    # 设计方法

    • Step 1:逻辑抽象;
      • 分析因果关系,确定输入 / 输出变量;
      • 定义逻辑状态的含意(赋值);
        • 输入和输出都有可能需要多个变量
        • 一个对象有 kk 种状态,则可能需要 n=log2kn=\lceil\log_2k\rceil 个信号来表示;若 2n>k2^n\gt k,多余的项为无关项中的约束项,可以不写在真值表中
      • 列出真值表;
    • Step 2:将真值表转换为逻辑函数式;
    • Step 3:选定器件类型;
      • 是否有器件类型限制,例如 “只能用与非门” 这类条件;
      • 是否有指定使用的中规模器件,例如 “使用数据选择器” 这类条件;
    • Step 4:根据所选器件,对逻辑式进行化简或形式变换;(或进行相应的描述(PLD))
      • 与非门 —— 与非 — 与非式
      • 或非门 —— 或非 — 或非式
    • Step 5:画出逻辑电路连接图

    # 加法器

    # 半加法器

    Basic Adders

    半加法器

    ABC_Σ\Sigma
    0000
    0101
    1001
    1110

    Cout=ABΣ=AB\begin{aligned} C_{out}&=AB\\ \Sigma&=A\oplus B \end{aligned}

    门电路实现:

    半加法器门电路实现

    # 全加法器

    Full-adder

    全加法器

    ABC_C_Σ\Sigma
    00000
    00101
    01001
    01110
    10001
    10110
    11010
    11111

    Cout=AB+(AB)CinΣ=ABCin\begin{aligned} C_{out}&=AB+(A\oplus B)C_{in}\\ \Sigma&=A\oplus B\oplus C_{in} \end{aligned}

    输出写为这个形式,与半加法器的输出对比,全加法器可以用 2 个半加法器级联和 1 个或门实现

    门电路实现全加法器:

    门电路实现全加法器

    半加法器实现全加法器:

    半加法器实现全加法器

    # 二进制加法器

    Parallel Binary Adders

    全加法器级联实现,利用了模块化设计的思想

    模块化思想,或者说集成化思想,是人们不断积累的设计经验。有很多场合下的逻辑问题是具有共性的,为了更方便地设计电路,通常可以将一些具有特定功能、解决某类问题的电路封装打包即集成化,组成一些常用的组合逻辑电路模块(中规模集成器件);在设计大规模的集成电路时,就可以调用这些典型的电路模块而避免使用复杂且数量巨大的分立的门电路

    # 74LS283

    4-bit parallel adder, 4 位二进制加法器

    4位并行加法器

    2 个 74LS283 级联可以得到 8 位二进制加法器(8-bit parallel adder):

    全加法器级联实现4位并行加法器

    4 个全加法器级联实现,但是这样上一级的 CoutC_{out} 到下一级 CinC_{in} 会存在延迟,级联越多,这样的串行部分产生的延迟越多

    全加法器中,令:

    Cg=ABCp=A+B\begin{aligned} C_{g}&=AB\\ C_{p}&=A+B \end{aligned}

    则有:

    Cout=Cg+CpCinC_{out}=C_{g}+C_{p}C_{in}

    look-ahead-adder

    # 比较器

    comparator

    简单的 1、2 位比较器直接利用异或门可以实现。

    位数比较多时,比如 4 位比较器,从高位开始比较,若出现不一样则剩余低位可以不必比较,直接输出结果(大于、小于)否则直到所有位比较完,并且都相等,则输出 “相等”

    # 74LS85

    4 位比较器

    74LS85

    • 8 位数据输入,按从 3 到 0 的高位到低位顺序比较
    • 3 位级联输入,当本次比较结果为相等时输出级联输入的结果
    • 3 位比较结果输出,高电平有效

    注意 3 个级联输入: A<BA<BA=BA=BA>BA>B 。 这些输入允许几个比较器级联起来,用来比较大于 4 位数字的数。

    • 低位比较器的 A<BA<BA=BA =BA>BA>B 输出与对应的下一个较高位比较器的级联输入相连接;
    • 最低位比较器在 A=BA=B 输入端又必须接高电平,在 A<BA<BA>BA>B 输入端又必须接低电平;

    # 扩展 8 位比较器

    2 个 74HC85 扩展为 8 位比较器:

    74LS85扩展8位比较器

    # 扩展 16 位比较器

    5 个 74LS85 扩展为 16 位比较器:

    74LS85扩展16位比较器

    # 译码器 (n:m)

    decoder

    # 74HC154(4:16)

    4 位译码器,4 线 - 16 线译码器,也是 4 线 - 16 线解多路复用器

    • 4 位输入
    • 2 位片选,当且仅当 2 位都为低位时才正常工作
    • 16 位输出,低电平有效

    74HC154

    # 扩展

    74HC54 扩展实现 5 位输入译码器:

    74HC54实现5位输入译码器

    # 74HC42

    BCD - 十进制译码器,把每 一个 BCD 码 (8421 码) 转换成 10 个十进制数中的一个数所表示的输出

    • 4 位输入,A0,A1,A2,A3A_0,A_1,A_2,A_3 分别表示 1 2 4 8
    • 10 位输出, 低电平有效

    # 74LS47

    BCD-7 段译码器 / 驱动器,它把一个 BCD 输入进行译码,并驱动 7 段显示器

    • 4 位输入,A0,A1,A2,A3A_0,A_1,A_2,A_3 分别表示 1 2 4 8 ,高电平有效
    • 3 位控制位
      • LT\overline{LT}
      • RBI\overline{RBI}
      • BI/RNO\overline{BI/RNO}
    • 7 位输出,低电平有效

    74LS48 与它的区别在于输出的 7 位没有取非,它的输出是高电平有效

    # 试灯

    试灯,点亮所有灯管,用来检测是否有段已经烧坏

    # 灭零

    LTRBI=10\overline{LT}\ \overline{RBI}=10灭零。把数字前面的 0 熄灭称为头部灭零,把数字尾端的 0 熄灭称为尾部灭零。只是把不需要的零熄灭!

    # 头部灭零

    头部灭零

    # 尾部灭零

    尾部灭零

    # 熄灯

    LT=0\overline{LT}=0 时熄灯

    # 2-4 译码器

    # 3-8 译码器

    3 线 - 8 线译码器

    # ⭐️74LS138(3:8)

    • 3 位输入位为选择位
    • 8 位输出位
    • 3 位片选位

    # 5-32 译码器

    可以用 1 个 74LS139 和 4 个 74LS138 实现

    # 编码器 (n:m)

    数字电路中的编码器是在已经实现了非电信号到电信号的转换、模拟信号到数字信号的转换之后对信息进行编码,即此时输入的信号是一组 高、低电平 (1 和 0 );

    编码器的功能就是根据输入的每一个高、低电平来输出其对应的二进制代码

    输入 nn电平信号,至少需要 log2n\lceil\log_2n\rceil 个输出端口,即 log2n\lceil\log_2n\rceil 位二进制编码

    # 普通编码器

    普通编码器规定了任意时刻有且只有一个有效输入信号,作为默认给定的约束条件

    # 十进制 - BCD 编码器

    十进制-BCD编码器

    # 8/3 普通编码器

    普通8:3编码器

    真值表:(不包含无关项)

    I0I_0I1I_1I2I_2I3I_3I4I_4I5I_5I6I_6I7I_7Y2Y_2Y1Y_1Y0Y_0
    1
    11
    11
    111
    11
    111
    111
    1111

    因为任一时刻只能有一个输入为 1,因此只需要将输入为 1 的变量保留,其余的变量无关(相当于在卡诺图中 x 的方块填入 1 合并消去无关变量)化简为最简与或式:

    Y2=I4+I5+I6+I7Y1=I2+I3+I6+I7Y0=I1+I3+I5+I7Y_2=I_4+I_5+I_6+I_7\\ Y_1=I_2+I_3+I_6+I_7\\ Y_0=I_1+I_3+I_5+I_7

    从而可画出逻辑电路图:

    普通8:3编码器逻辑电路图

    此设计存在的问题:

    • 不能有多个信号同时输入
    • 也不能没有信号输入(因为根据我们的约束条件,00000000 也属于无关项);

    因此我们以上的设计只是一个思路的训练,最终的设计结果并不能直接应用到实际当中

    # 优先编码器

    • 允许多个信号同时输入
    • 编码时存在优先顺序

    # 8/3 优先编码器

    规定:

    • 高电平输入有效,即输入 1 表示对此输入信号编码,输出的 Y2Y1Y0Y_2Y_1Y_0 即为 0 ~ 7 对应的三位二进制代码 000~111;
    • 二进制数大的优先级高,即 I7I_7 优先级最高,I0I_0 最低;

    真值表:(不包含无关项)

    I0I_0I1I_1I2I_2I3I_3I4I_4I5I_5I6I_6I7I_7Y2Y_2Y1Y_1Y0Y_0
    xxxxxxx1111
    xxxxxx111
    xxxxx111
    xxxx11
    xxx111
    xx11
    x11
    1

    此真值表不完整!当前只有 2812^8–1 行,缺少了一行 00000000 !

    这时存在的问题是,输入全为 0 时输出为 I0I_0 表示的状态,无法表示输入全为 0 的状态。在实际应用的 8/3 优先编码器中,还附加了其他的输入输出端口,以便于片选和拓展。

    附加了其他的输入输出端口进行片选和拓展的最主要目还是厂家希望产品应用更广泛,同种型号芯片可以实现不同功能

    # 75LS148(8:3)

    • 8 位输入,低电平有效
    • 1 位片选位 EI\overline{EI} ,低电平有效,正常工作
    • 3 位输出编码,低电平有效⚠️
    • 2 位输出有效位
      • EO\overline{EO}EI\overline{EI} 有效且所有输入都无效时为 0
      • GS\overline{GS}EI\overline{EI} 无效且任何一个输入有效时为 0

    功能表:

    EIEI'I0I_0'I1I_1'I2I_2'I3I_3'I4I_4'I5I_5'I6I_6'I7I_7'Y2Y_2'Y1Y_1'Y0Y_0'EOEO'GSGS'
    1xxxxxxxx11111
    111111111111
    xxxxxxx1
    xxxxxx111
    xxxxx1111
    xxxx111111
    xxx111111
    xx11111111
    x111111111
    11111111111

    # 扩展

    2 个 74LS148 实现 16 线 - 4 线编码器:

    74LS148实现16线-4线编码器

    # 多路复用器 (n:1)

    (Mux)多路复用器(Multiplexer)或数据选择器(Data Selector),能够实现并行数据到串行数据的转化。

    包括:

    • 数据选择位(低电平有效)
    • 数据输入位
    • 数据输出位

    此时数据输入输出位不必再关心是否低电平有效

    # 4 路复用器

    • 2 位数据选择位 S0,S1\overline{S_0},\overline{S_1} ,低电平有效
    • 4 位数据输入位 D0,D1,D2,D3D_0,D_1,D_2,D_3
    • 1 位数据输出位 YY

    Y=S1S0D0+S1S0D1+S1S0D0+S1S0D1Y=\overline{S_1S_0}D_0+\overline{S_1}S_0D_1+S_1\overline{S_0}D_0+S_1S_0D_1

    4位多路复用器

    # 8 路复用器

    Y=S3S2S1S0D0+S3S2S1S0D1+S3S2S1S0D2+S3S2S1S0D3\begin{aligned} Y&=\overline{S_3S_2S_1S_0}D_0+\overline{S_3S_2S_1}S_0D_1+\overline{S_3S_2}S_1\overline{S_0}D_2+\overline{S_3S_2S_1S_0}D_3 \end{aligned}

    # 74LS151

    # 扩展 16 路复用器

    2 个 74LS151 扩展实现 16 路复用器

    74LS151扩展实现16路复用器

    # 74LS153

    双 4 选 1 数据选择器,也就是 2 组 4 选 1 共用同一个选择输入和同一个片选输入。

    # 74LS157

    四 2 选 1 数据选择器,也就是 4 组 2 选 1 共用同一个选择输入和同一个片选输入。

    • 1 位片选输入,电平有效
    • 1 位选择输入
    • 8 位数据输入
    • 4 位数据输出

    # 7 段显示多路复用器

    把 BCD 数字分配到一个 7 段显示器上的一个简单方法

    # ⭐️扩展

    2 个 4 选 1 扩展为 8 选 1

    1 个 4 选 1 + 门电路扩展为 5 选 1

    5 个 4 选 1 扩展为 16 选 1 (⭐️树状扩展)

    # ⭐️实现门电路

    # 解多路复用器 (1:n)

    解多路分配器(Demultiplexer)或数据分配器()

    数据选择器的逆

    # 1 线 - 4 线多路分配器

    1线-4线多路分配器

    1线-4线多路分配器波形图

    # 74HC154

    4 位译码器 74HC154 可以作为 4 线 - 16 线解多路复用器

    74HC154作4线-16线解多路复用器

    # 传输时延与竞争冒险

    • 毛刺 glitch
    • 竞争 race
    • 冒险 hazard

    时延导致竞争的存在,竞争可能产生毛刺,竞争导致了逻辑误判,就是冒险

    # 消除冒险的方法

    # 滤波电容

    # 冗余项

    \ No newline at end of file diff --git a/ie/de/digital-process/index.html b/ie/de/digital-process/index.html index 9ab80670..b6de5671 100644 --- a/ie/de/digital-process/index.html +++ b/ie/de/digital-process/index.html @@ -1 +1 @@ -数模转换 - 数字逻辑电路 - 电子信息 | Balloon Party = 風船のパーティー = fuusen no party

    # 采样定理

    # ADC

    # 运算放大器

    运算放大器是一种线性放大器。

    • 两个输入
      • 反相
      • 同相
    • 和一个输出

    特点:

    • 很高的电压增益
    • 很大的输入阻抗
    • 很低的输出阻抗

    运算放大器的 3 种使用方式:

    运算放大器

    用做反相放大器时,图中,虚地处电压为 0 V ,由于很大的输入阻抗没有电流流入运算放大器,输入电压处 VinV_{in} 所有电流流向输出电压处 VoutV_{out} ,可得:

    VoutVin=RfRi\frac{V_{out}}{V_{in}}=-\frac{R_f}{R_i}

    其中 RfR_f 是反馈电阻,RiR_i 是输入电阻。

    其 中 VoutVin\frac{V_{out}}{V_{in}}闭环电压增益 (闭环是指由电阻 R,提供的从输出到输入的反馈),负号表示反相。

    # 快速 ADC

    3 位快速 ADC:

    3位快速ADC

    每个比较器的参考电压都由电阻分压电路设定。每个比较器的输出都连接到优先权编码器的一个输入。编码器由 EN 输入上的一个脉冲使能,3 位码表示出现在编码器输出上的输入值。 二进制数由最高级别输入的高电平决定。 二进制数的序列表示 ADC 的输入,每次使能脉冲有效的期间,输入信号得到采样。转换的精度由 2 个因素确定:

    • 使能脉冲的频率
    • 二进制编码的位数

    # 双积分 ADC

    一个斜坡发生器 (积分器) 用来产生双积分特性。本质是在一个计数器进位的时间周期内,信号对电容充电,然后在恒定的电容放电速度下,在放电时间内计数器计数实现对这一信号电压大小的量化。

    双积分ADC原理

    这里可变斜率由 VinV_{in} 的大小决定,越大则可变斜率越大,电容充电速度越快。而电容放电的速度是固定的。

    双积分ADC

    由于需要充电、放电,所以 DAC 相对较慢,而且该方法要求采样保持时间足够长。

    # DAC

    # 二进制权值输入 DAC

    4位二进制权值输入DAC

    输入的高电平电压若为 5 V ,输出:

    4位二进制权值输入DAC输出

    # R/2R 阶梯形 DAC

    这里利用了戴维南定理等效电路

    该方法结构更加精简,容易实现。

    nn 输入的高电平电压若为 ViV_i ,则输出电压为从 0 到 ViV_i ,以 Vi2n\frac{V_i}{2^n} 为间隔,依次对应输入信号的 0 到 2n12^n-1

    R:2R阶梯形DAC

    Edited on Views times
    \ No newline at end of file +数模转换 - 数字逻辑电路 - 电子信息 | Balloon Party = 風船のパーティー = fuusen no party

    # 采样定理

    # ADC

    # 运算放大器

    运算放大器是一种线性放大器。

    • 两个输入
      • 反相
      • 同相
    • 和一个输出

    特点:

    • 很高的电压增益
    • 很大的输入阻抗
    • 很低的输出阻抗

    运算放大器的 3 种使用方式:

    运算放大器

    用做反相放大器时,图中,虚地处电压为 0 V ,由于很大的输入阻抗没有电流流入运算放大器,输入电压处 VinV_{in} 所有电流流向输出电压处 VoutV_{out} ,可得:

    VoutVin=RfRi\frac{V_{out}}{V_{in}}=-\frac{R_f}{R_i}

    其中 RfR_f 是反馈电阻,RiR_i 是输入电阻。

    其 中 VoutVin\frac{V_{out}}{V_{in}}闭环电压增益 (闭环是指由电阻 R,提供的从输出到输入的反馈),负号表示反相。

    # 快速 ADC

    3 位快速 ADC:

    3位快速ADC

    每个比较器的参考电压都由电阻分压电路设定。每个比较器的输出都连接到优先权编码器的一个输入。编码器由 EN 输入上的一个脉冲使能,3 位码表示出现在编码器输出上的输入值。 二进制数由最高级别输入的高电平决定。 二进制数的序列表示 ADC 的输入,每次使能脉冲有效的期间,输入信号得到采样。转换的精度由 2 个因素确定:

    • 使能脉冲的频率
    • 二进制编码的位数

    # 双积分 ADC

    一个斜坡发生器 (积分器) 用来产生双积分特性。本质是在一个计数器进位的时间周期内,信号对电容充电,然后在恒定的电容放电速度下,在放电时间内计数器计数实现对这一信号电压大小的量化。

    双积分ADC原理

    这里可变斜率由 VinV_{in} 的大小决定,越大则可变斜率越大,电容充电速度越快。而电容放电的速度是固定的。

    双积分ADC

    由于需要充电、放电,所以 DAC 相对较慢,而且该方法要求采样保持时间足够长。

    # DAC

    # 二进制权值输入 DAC

    4位二进制权值输入DAC

    输入的高电平电压若为 5 V ,输出:

    4位二进制权值输入DAC输出

    # R/2R 阶梯形 DAC

    这里利用了戴维南定理等效电路

    该方法结构更加精简,容易实现。

    nn 输入的高电平电压若为 ViV_i ,则输出电压为从 0 到 ViV_i ,以 Vi2n\frac{V_i}{2^n} 为间隔,依次对应输入信号的 0 到 2n12^n-1

    R:2R阶梯形DAC

    Edited on Views times
    \ No newline at end of file diff --git a/ie/de/gate-circuit/index.html b/ie/de/gate-circuit/index.html index b670e56a..2b650782 100644 --- a/ie/de/gate-circuit/index.html +++ b/ie/de/gate-circuit/index.html @@ -1 +1 @@ -门电路 - 数字逻辑电路 - 电子信息 | Balloon Party = 風船のパーティー = fuusen no party

    # 基本概念

    !!! 可能会在选择填空题中考察

    • 上拉 —— 使输出钳位至高电平

    • 下拉 —— 使输出钳位至低电平;

    • 灌电流 —— 输出为低电平时,负载电流从外部电路注入输出端口;

    • 拉电流 —— 输出为高电平时,负载电流从输出端口流出到外部电路;

    当门电路没有外部的有源输入信号,而是输入端口与地之间接入一个电阻时:

    • 关门电阻 —— 能够维持门电路输出高电平的电阻的最大值;
    • 开门电阻 —— 能够维持门电路输出低电平的电阻的最小值;

    (联想 TTL 反相器的输入端负载特性曲线)

    • 扇入系数 —— 门电路允许的输入端的个数;

    • 扇出系数 —— 门电路允许驱动的同类型负载门的个数;

    • 高阻态 —— 从输出端口往里看的等效电阻近似无穷大,门电路失去对输出电压的控制,输出端口处电平由外部电路决定,即门电路与后级电路无联系;

    • 传输延迟时间 —— 由于半导体器件寄生电容的存在,输出量的变化总是滞后于输入量的变化;

    • 输入端噪声容限 —— 能够维持输出电压在规定高低电平范围内的输入电压的波动范围;

    • 缓冲级 —— 多级集成门电路应用中为了保持输入输出特性一致性输入级和输出级各加一级反相器;

    # 基本逻辑元件

    元件分类

    # 与门 AND

    X=ABX=AB

    AND

    与非门 NAND 构造与门 AND :

    AND-from-NAND

    或非门 NOR 构造与门 AND :

    AND-from-NOR

    # 或门 OR

    X=A+BX=A+B

    OR

    与非门 NAND 构造或门 OR :

    OR-from-NAND

    或非门 NOR 构造或门 OR :

    OR-fron-NOR

    # 非门 Inverter

    也叫反相器

    Y=XY=\overline{X}

    Inverter

    与非门 NAND 构造非门 Inverter :

    Inverter-from-NAND

    或非门 NOR 构造非门 Inverter :

    Inverter-from-NOR

    # 与非门 NAND

    等价于 Negative-OR

    X=ABX=\overline{AB}

    NAND

    两个输入的与非门真值表:

    输入 1输入 2输出
    001
    011
    101
    110

    对于 n 个输入的与非门,当且仅当输入全部为 1 时,输出为 0 ,其余情况输出均为 1 ;或者说任何一个输入为 0 ,则输出为 1

    与非门可以实现与门、或门、非门、或非门

    NOR 门来实现 NAND 门:

    NAND-from-NOR

    # 或非门 NOR

    等价于 Negative-AND

    X=A+BX=\overline{A+B}

    NOR

    两个输入的或非门真值表:

    输入 1输入 2输出
    001
    010
    100
    110

    对于 n 个输入的或非门,当且仅当输入全部为 0 时,输出为 1 ,其余情况输出均为 0 ;或者说任何一个输入为 1 ,则输出为 0

    或非门可以实现与门、或门、非门、或非门

    NAND 门来实现 NOR 门:

    NOR-from-NAND

    # 异或门 XOR

    Exclusive-OR

    XOR (/ˌɛks ˈɔːr/, /ˌɛks ˈɔː/, /ˈksɔːr/ or /ˈksɔː/), EOR, EXOR, {\displaystyle {\dot {\vee }}}, {\displaystyle {\overline {\vee }}}, {\displaystyle {\underline {\vee }}}, , {\displaystyle \oplus }, {\displaystyle \nleftrightarrow }, {\displaystyle \not \equiv }.

    XOR

    两个输入的或非门真值表:

    输入 1输入 2输出
    000
    011
    101
    110

    对于 n 个输入的异或门,当且仅当输入不同时,输出为 1 ;输入相同时,输出为 0

    异或门可以作为半加法器,输出为进位

    # 同或门 XNOR

    Exclusive-NOR

    XNOR

    两个输入的或非门真值表:

    输入 1输入 2输出
    001
    010
    100
    111

    对于 n 个输入的同或门,当且仅当输入相同时,输出为 1 ;输入不同时,输出为 0

    # 门电路逻辑功能的分析与设计

    一个 CMOS 门电路的构成:

    • 上拉部分
      • 由 PMOS 管实现,导通对应输入低电平,输出高电平
    • 下拉部分
      • 由 NMOS 管实现,导通对应输入高电平,输出低电平

    逻辑功能上:

    • 与运算对应到电路中就是串联结构
    • 或运算对应到电路中就是并联结构

    根据前面的基本设计思想,当含有多个输入变量时,只需要:

    • 根据其逻辑输出为 0 时设计下拉部分电路

      • (或先根据其逻辑输出为 1 时设计上拉部分电路,输入变量取反),
    • 然后根据对称性绘制另一侧的上 / 下拉电路(串变为并,并变为串)

    • 组合在一起即可;

    # 门电路的输入特性与输出特性

    # 门电路的输入端负载特性

    # OD 门 / OC 门外接上拉电阻的计算分析

    Edited on Views times
    \ No newline at end of file +门电路 - 数字逻辑电路 - 电子信息 | Balloon Party = 風船のパーティー = fuusen no party

    # 基本概念

    !!! 可能会在选择填空题中考察

    • 上拉 —— 使输出钳位至高电平

    • 下拉 —— 使输出钳位至低电平;

    • 灌电流 —— 输出为低电平时,负载电流从外部电路注入输出端口;

    • 拉电流 —— 输出为高电平时,负载电流从输出端口流出到外部电路;

    当门电路没有外部的有源输入信号,而是输入端口与地之间接入一个电阻时:

    • 关门电阻 —— 能够维持门电路输出高电平的电阻的最大值;
    • 开门电阻 —— 能够维持门电路输出低电平的电阻的最小值;

    (联想 TTL 反相器的输入端负载特性曲线)

    • 扇入系数 —— 门电路允许的输入端的个数;

    • 扇出系数 —— 门电路允许驱动的同类型负载门的个数;

    • 高阻态 —— 从输出端口往里看的等效电阻近似无穷大,门电路失去对输出电压的控制,输出端口处电平由外部电路决定,即门电路与后级电路无联系;

    • 传输延迟时间 —— 由于半导体器件寄生电容的存在,输出量的变化总是滞后于输入量的变化;

    • 输入端噪声容限 —— 能够维持输出电压在规定高低电平范围内的输入电压的波动范围;

    • 缓冲级 —— 多级集成门电路应用中为了保持输入输出特性一致性输入级和输出级各加一级反相器;

    # 基本逻辑元件

    元件分类

    # 与门 AND

    X=ABX=AB

    AND

    与非门 NAND 构造与门 AND :

    AND-from-NAND

    或非门 NOR 构造与门 AND :

    AND-from-NOR

    # 或门 OR

    X=A+BX=A+B

    OR

    与非门 NAND 构造或门 OR :

    OR-from-NAND

    或非门 NOR 构造或门 OR :

    OR-fron-NOR

    # 非门 Inverter

    也叫反相器

    Y=XY=\overline{X}

    Inverter

    与非门 NAND 构造非门 Inverter :

    Inverter-from-NAND

    或非门 NOR 构造非门 Inverter :

    Inverter-from-NOR

    # 与非门 NAND

    等价于 Negative-OR

    X=ABX=\overline{AB}

    NAND

    两个输入的与非门真值表:

    输入 1输入 2输出
    001
    011
    101
    110

    对于 n 个输入的与非门,当且仅当输入全部为 1 时,输出为 0 ,其余情况输出均为 1 ;或者说任何一个输入为 0 ,则输出为 1

    与非门可以实现与门、或门、非门、或非门

    NOR 门来实现 NAND 门:

    NAND-from-NOR

    # 或非门 NOR

    等价于 Negative-AND

    X=A+BX=\overline{A+B}

    NOR

    两个输入的或非门真值表:

    输入 1输入 2输出
    001
    010
    100
    110

    对于 n 个输入的或非门,当且仅当输入全部为 0 时,输出为 1 ,其余情况输出均为 0 ;或者说任何一个输入为 1 ,则输出为 0

    或非门可以实现与门、或门、非门、或非门

    NAND 门来实现 NOR 门:

    NOR-from-NAND

    # 异或门 XOR

    Exclusive-OR

    XOR (/ˌɛks ˈɔːr/, /ˌɛks ˈɔː/, /ˈksɔːr/ or /ˈksɔː/), EOR, EXOR, {\displaystyle {\dot {\vee }}}, {\displaystyle {\overline {\vee }}}, {\displaystyle {\underline {\vee }}}, , {\displaystyle \oplus }, {\displaystyle \nleftrightarrow }, {\displaystyle \not \equiv }.

    XOR

    两个输入的或非门真值表:

    输入 1输入 2输出
    000
    011
    101
    110

    对于 n 个输入的异或门,当且仅当输入不同时,输出为 1 ;输入相同时,输出为 0

    异或门可以作为半加法器,输出为进位

    # 同或门 XNOR

    Exclusive-NOR

    XNOR

    两个输入的或非门真值表:

    输入 1输入 2输出
    001
    010
    100
    111

    对于 n 个输入的同或门,当且仅当输入相同时,输出为 1 ;输入不同时,输出为 0

    # 门电路逻辑功能的分析与设计

    一个 CMOS 门电路的构成:

    • 上拉部分
      • 由 PMOS 管实现,导通对应输入低电平,输出高电平
    • 下拉部分
      • 由 NMOS 管实现,导通对应输入高电平,输出低电平

    逻辑功能上:

    • 与运算对应到电路中就是串联结构
    • 或运算对应到电路中就是并联结构

    根据前面的基本设计思想,当含有多个输入变量时,只需要:

    • 根据其逻辑输出为 0 时设计下拉部分电路

      • (或先根据其逻辑输出为 1 时设计上拉部分电路,输入变量取反),
    • 然后根据对称性绘制另一侧的上 / 下拉电路(串变为并,并变为串)

    • 组合在一起即可;

    # 门电路的输入特性与输出特性

    # 门电路的输入端负载特性

    # OD 门 / OC 门外接上拉电阻的计算分析

    Edited on Views times
    \ No newline at end of file diff --git a/ie/de/integrated-circuit/index.html b/ie/de/integrated-circuit/index.html index a96713b9..8f7995d2 100644 --- a/ie/de/integrated-circuit/index.html +++ b/ie/de/integrated-circuit/index.html @@ -1 +1 @@ -集成电路 (CMOS&TTL) - 数字逻辑电路 - 电子信息 | Balloon Party = 風船のパーティー = fuusen no party

    # 逻辑电平规范与抗噪

    # 逻辑电平规范

    ⭐️保证输出的电平作为下一级输入是有效的

    # CMOS

    cmos输入输出逻辑电平

    # TTL

    ttl输入输出逻辑电平

    # 噪声容限

    ⭐️保证输出的电平作为下一级输入是有效的基础上,输出与输入的界限相差即为噪声容限

    电路抗噪度的一个衡量标准叫做噪声容限 (noisemargin),用伏特表示。对于一个给定的逻辑电路,有两个噪声容限值:

    • 噪声容限高电平 (VNHV_{NH}) (Noisemargin High)
    • 噪声容限低电平 (VNLV_{NL}) (Noisemargin Low)

    这两个参数由如下的公式定义:

    VNH=VOH(min)VIH(min)VNL=VIL(max)VOL(max)V_{NH}=V_{OH(\min)}-V_{IH(\min)}\\ V_{NL}=V_{IL(\max)}-V_{OL(\max)}

    有时看到噪声容限是用 VCCV_{CC} 的百分比来表示的

    噪声容限

    # 负载和扇出

    当一个逻辑门的输出连接到了一个或多个其他门的输人时,驱动门上就会有一个负载,对于一个给定的门,它可以驱动的负载门输人是有数量限制的。这个限制就称为这个门的扇出

    基本概念:

    • 灌电流 —— 输出为低电平时,负载电流从外部电路注入输出端口;
    • 拉电流 —— 输出为高电平时,负载电流从输出端口流出到外部电路;
    • 扇入系数 —— 门电路允许的输入端的个数;
    • 扇出系数 —— 门电路允许驱动的同类型负载门的个数;

    # CMOS

    当驱动门的输出是高电平时,负载门的输入电容通过驱动门的输出电阻充电。当驱动门电路的输出是低电平时,电容放电:

    cmos容性负载

    如果给驱动门的输出添加更多的负载门输人,那么总的电容将会增加,因为输入电容实际上是并行出现的。电容的增加会延长充电和放电时间,因此就会降低这个门电路的最大工作频率。因此,CMOS 门电路的扇出是和工作频率有关的。负载门输入越少,最大频率就越高。

    # TTL

    TTL 驱动门向处于高电平状态的负载门提供灌电流 (IIHI_{IH}),从处于低电平状态的负载门吸收拉电流 (IILI_{IL})。

    TTL 电路的功耗在工作频率的范围以内基本上是恒定功率。

    ttl负载

    负载元件越多,驱动门承载的灌电流或拉电流就越大,驱动门中内阻的压降就越大,要保证不能脱离标准逻辑电平,限制了扇出。

    # CMOS

    数字电路中只考虑增强型 MOS 管

    nmos-and-pmos

    # NMOS

    NMOSFET,N 型金氧半场效晶体管

    N 沟道增强型 MOS 管,箭头往里走,栅级无圆圈

    NMOS

    左侧是简化符号,右侧是标准符号

    状态栅级 G漏级 D源级 S (接地)
    导通101
    关闭010

    nmos

    # PMOS

    PMOSFET,P 型金氧半场效晶体管

    P 沟道增强型 MOS 管,箭头往外走,栅级有圆圈

    PMOS

    左侧是简化符号,右侧是标准符号

    状态栅级 G漏级 D源级 S (接电源)
    导通010
    关闭101

    pmos

    # CMOS

    由 NMOS 和 PMOS 两种管子组成的互补 MOS 电路,即 CMOS 电路。

    CMOS 电路元件不使用的管脚尽量不能悬空⚠️

    # 非门

    cmos非门

    # 与非门

    cmos与非门

    或非门和与非门之间存在对偶关系,将 NMOS 和 PMOS 交换、高电平和低电平交换,就能够相互转换

    # 或非门

    cmos或非门

    # 三态门

    三态门引入了高阻抗状态

    三态门的 3 个状态如下:

    三态门3个状态

    一个错误❌的三态 CMOS 反相器:

    三态CMOS反相器-错误

    Enable\overline{Enable} 和输入是对称设计的,所以输入为高电平时,输出也一样处于高阻抗状态。

    正确设计:

    cmos三态非门

    # 漏极开路门

    Open-drain,OD 门

    术语漏极开路的意思是输出晶体管的漏极端没有连接,必须从外部通过负载连接到 VDDV_{DD}

    实现大负载

    与非门示例:

    cmos漏极开路与非门

    # TTL

    TTL 电路中悬空的输入端将起到高电平的效果。

    TTL 电路元件不使用的管脚尽量不能悬空⚠️,避免噪声影响。

    # 双极型晶体管

    Bipoplar Junction Transistor (BJT) (三极管)

    开关效应原理图:

    双极型晶体管原理

    # TTL 门电路

    # 非门

    ttl非门

    工作原理如下:

    ttl非门原理

    # 与非门

    ttl与非门

    # 集电极开路门

    Open-Collector Gates,OC 门

    集电路开路

    驱动 LED 和灯泡:

    ttl集电极开路

    \ No newline at end of file +集成电路 (CMOS&TTL) - 数字逻辑电路 - 电子信息 | Balloon Party = 風船のパーティー = fuusen no party

    # 逻辑电平规范与抗噪

    # 逻辑电平规范

    ⭐️保证输出的电平作为下一级输入是有效的

    # CMOS

    cmos输入输出逻辑电平

    # TTL

    ttl输入输出逻辑电平

    # 噪声容限

    ⭐️保证输出的电平作为下一级输入是有效的基础上,输出与输入的界限相差即为噪声容限

    电路抗噪度的一个衡量标准叫做噪声容限 (noisemargin),用伏特表示。对于一个给定的逻辑电路,有两个噪声容限值:

    • 噪声容限高电平 (VNHV_{NH}) (Noisemargin High)
    • 噪声容限低电平 (VNLV_{NL}) (Noisemargin Low)

    这两个参数由如下的公式定义:

    VNH=VOH(min)VIH(min)VNL=VIL(max)VOL(max)V_{NH}=V_{OH(\min)}-V_{IH(\min)}\\ V_{NL}=V_{IL(\max)}-V_{OL(\max)}

    有时看到噪声容限是用 VCCV_{CC} 的百分比来表示的

    噪声容限

    # 负载和扇出

    当一个逻辑门的输出连接到了一个或多个其他门的输人时,驱动门上就会有一个负载,对于一个给定的门,它可以驱动的负载门输人是有数量限制的。这个限制就称为这个门的扇出

    基本概念:

    • 灌电流 —— 输出为低电平时,负载电流从外部电路注入输出端口;
    • 拉电流 —— 输出为高电平时,负载电流从输出端口流出到外部电路;
    • 扇入系数 —— 门电路允许的输入端的个数;
    • 扇出系数 —— 门电路允许驱动的同类型负载门的个数;

    # CMOS

    当驱动门的输出是高电平时,负载门的输入电容通过驱动门的输出电阻充电。当驱动门电路的输出是低电平时,电容放电:

    cmos容性负载

    如果给驱动门的输出添加更多的负载门输人,那么总的电容将会增加,因为输入电容实际上是并行出现的。电容的增加会延长充电和放电时间,因此就会降低这个门电路的最大工作频率。因此,CMOS 门电路的扇出是和工作频率有关的。负载门输入越少,最大频率就越高。

    # TTL

    TTL 驱动门向处于高电平状态的负载门提供灌电流 (IIHI_{IH}),从处于低电平状态的负载门吸收拉电流 (IILI_{IL})。

    TTL 电路的功耗在工作频率的范围以内基本上是恒定功率。

    ttl负载

    负载元件越多,驱动门承载的灌电流或拉电流就越大,驱动门中内阻的压降就越大,要保证不能脱离标准逻辑电平,限制了扇出。

    # CMOS

    数字电路中只考虑增强型 MOS 管

    nmos-and-pmos

    # NMOS

    NMOSFET,N 型金氧半场效晶体管

    N 沟道增强型 MOS 管,箭头往里走,栅级无圆圈

    NMOS

    左侧是简化符号,右侧是标准符号

    状态栅级 G漏级 D源级 S (接地)
    导通101
    关闭010

    nmos

    # PMOS

    PMOSFET,P 型金氧半场效晶体管

    P 沟道增强型 MOS 管,箭头往外走,栅级有圆圈

    PMOS

    左侧是简化符号,右侧是标准符号

    状态栅级 G漏级 D源级 S (接电源)
    导通010
    关闭101

    pmos

    # CMOS

    由 NMOS 和 PMOS 两种管子组成的互补 MOS 电路,即 CMOS 电路。

    CMOS 电路元件不使用的管脚尽量不能悬空⚠️

    # 非门

    cmos非门

    # 与非门

    cmos与非门

    或非门和与非门之间存在对偶关系,将 NMOS 和 PMOS 交换、高电平和低电平交换,就能够相互转换

    # 或非门

    cmos或非门

    # 三态门

    三态门引入了高阻抗状态

    三态门的 3 个状态如下:

    三态门3个状态

    一个错误❌的三态 CMOS 反相器:

    三态CMOS反相器-错误

    Enable\overline{Enable} 和输入是对称设计的,所以输入为高电平时,输出也一样处于高阻抗状态。

    正确设计:

    cmos三态非门

    # 漏极开路门

    Open-drain,OD 门

    术语漏极开路的意思是输出晶体管的漏极端没有连接,必须从外部通过负载连接到 VDDV_{DD}

    实现大负载

    与非门示例:

    cmos漏极开路与非门

    # TTL

    TTL 电路中悬空的输入端将起到高电平的效果。

    TTL 电路元件不使用的管脚尽量不能悬空⚠️,避免噪声影响。

    # 双极型晶体管

    Bipoplar Junction Transistor (BJT) (三极管)

    开关效应原理图:

    双极型晶体管原理

    # TTL 门电路

    # 非门

    ttl非门

    工作原理如下:

    ttl非门原理

    # 与非门

    ttl与非门

    # 集电极开路门

    Open-Collector Gates,OC 门

    集电路开路

    驱动 LED 和灯泡:

    ttl集电极开路

    \ No newline at end of file diff --git a/ie/de/memory-storage/index.html b/ie/de/memory-storage/index.html index 207b47b3..06dd8aab 100644 --- a/ie/de/memory-storage/index.html +++ b/ie/de/memory-storage/index.html @@ -1 +1 @@ -存储器 - 数字逻辑电路 - 电子信息 | Balloon Party = 風船のパーティー = fuusen no party

    两类主要的半导体存储器是 RAM 和 ROM。

    • RAM (随机访问存储器) 类型的存储器在等量时间内可以访问所有的地址,并且可以按照任何顺序来选择这些地址以执行读或者写操作。所有的 RAM 都具有读和写的能力。因为 RAM 在电源关闭时会丢失所存储的数据,所以它是易失性内存。 RAM 系列 RAM 的两种类别是静 RAM (SRAM) 和动态 RAM (DRAM) 。当直流电源移走后, SRAM 和 DRAM 都会丢失存储的数据,因此被归类为易失性存储器。
      • 静态 RAM 通常使用锁存器作为存储元件,因而只要加上直流电源,就可以永远地存储数据。
        • 异步 SRAM (ASRAM)
        • 同步 SRAM (SB SRAM),具有突发特性
      • 动态 RAM 使用电容作为存储元件,如果没有对电容再充电就不能长期存储数据,再充电则称为刷新过程。
        • 快速页模式 DRAM (FPMDRAM)
        • 扩充数据输出 DRAM (EDO DRAM)
        • 突发 EDO DRAM (BEDO DRAM)
        • 同步 DRAM (SDRAM)
    • ROM (只读存储器) 类型的存储器将永久或者半永久地保存数据。数据可以从 ROM 中读出,但是和 RAM 不同,它没有写操作。ROM 和 RAM 相似,也是随机存储器,但是术语 RAM 的 传统意思是随机读 / 写存储器。本章将会介绍几种类型的 RAM 和 ROM。因为 ROM 在电源关闭 后仍然保存着数据,所以它是永久性存储器
      • MROM (Mask ROM) —— 掩膜 ROM,出场时已经固定,不能更改,适合大量生产,简单,便宜,非易失性。
      • PROM —— 可一次性编程 ROM
      • EPROM —— 可擦除 PROM,也就是可多次编程
      • UV EPROM —— 紫外线版本的 EPROM
      • EEPROM —— 电版本的

    对于给定的物理空间和成本,DRAM 可以存储比 SRAM 多得多的数据,因为 DRAM 单元更加简单,并在给定芯片面积里,因此可以比 SRAM 填充更多的单元

    # RAM

    # SRAM

    SRAM 基本单元(是 D - 触发器 ):

    sram单元

    32k×832k\times8 SRAM

    32kx8sram

    逻辑组成如下:

    32kx8sram逻辑图

    读:

    32kx8sram读

    写:

    32kx8sram写

    # DRAM

    DRAM 的基本单元,使用电容,电容充电表示 1 高电位,放电表示 0 低电位。

    mosdram单元

    DRAM 的 4 种基本操作:

    mosdram单元基本操作

    1M×11M\times1 DRAM 逻辑图:

    1mx1dram逻辑图

    DRAM 的读写周期:

    dram读写周期

    # ROM

    ROM(Read Only Memory), 即只读存储器,由于其在掉电后数据不会丢失,因此在早期的时候,ROM 芯片中的内容一般都是在生产的时候被烧写进去的,然后终生具有这些固定的内容。不过,随着技术的不断发展和革新,现在很多型号的 ROM 都已经可以被多次修改了,但是大家仍然习惯对其沿用 “只读存储器” 的称号。

    # MOS ROM

    MOS ROM 基本单元:

    mosrom单元

    256x4rom

    256x4rom逻辑图

    # ROM 设计组合逻辑电路

    实际上是枚举所有的输入输出,也就是真值表,存储到 ROM 里。

    输入作为地址,通过地址译码器,输出相应地址的数据。

    • 有黑点⚫️表示存储 1 ;
    • 否则表示存储 0 ;

    给出逻辑表达式(函数)时,求出标准的极小值只和的形式,依据极小值的下标确定画⚫️的位置

    # PROM

    PROM(Programmable Read Only Memory), 即可编程只读存储器,不过它也只能被程序写一次,以后就只能保存和使用这些数据了,所以 PROM 也称为一次可编程存储器。

    # EPROM

    EPROM(Erasable Programmable Read Only Memory),即可擦除可编程只读存储器。它解决了 PROM 只能编程一次的问题,不过其存储内容的擦除必须借助紫外线的长时间照射才能完成,所以 EPROM 又叫紫外线擦除可编程只读存储器。

    # EEPROM

    EEPROM(Electrically Erasable Programmable Read Only Memory),即电可擦除可编程只读存储器。它克服了 EPROM 在擦除时较为麻烦的问题,可以直接通过计算机程序对其进行重复的擦写,其擦除、重写的原理是通过使用高于常规的操作电压实现的。对于 EEPROM 来说,最重要的参数指标就是寿命问题,也就是可擦写的次数。

    # 闪存

    闪存 Flash Memory 是 EEPROM 的一种特殊形式,也就是变种。

    EEPROM 的擦除和重写都是针对整个芯片而言的,但 Flash 的数据删除和重写则是以块(块的容量一般在 256KB~20MB 之间)为单位的,因此 Flash 的更新速度要远比 EEPROM 快。不过由于数据的删除和重写不是以单个字节为单位的,因此 Flash 仍不能达到 RAM 的要求。

    Flash 又分为 NOR 型和 NAND 型两个大类,其中 NOR 型具有独立的地址线和数据线,适合频繁的随机读、写场合,适合程序代码的存储和直接运行,但容量通常较小,价格也比较贵;而 NAND 型的地址线和数据线是公用的,因此其本身的操作速度和频率就会偏低,但是容量相对较大,且价格也要便宜,所以适合大量资料的存储。

    # 存储器扩展

    # 字长扩展

    共享数据线和控制线,并联数据线。

    2 个 64k×464k\times4 ROM 的字长扩展得到个 64k×864k\times8 ROM :

    64kx4rom字长扩展64kx8rom

    # 字容量扩展

    514kx4ram字容量扩展1mx4ram

    💡 利用译码器

    # 字长字扩展结合

    # 特殊存储器

    # FIFO

    FIFO

    FIFO 寄存器模块的应用,实现数据整型:

    FIFO应用

    # LIFO

    类似栈的效果,LIFO 寄存器结构:

    LIFO结构

    Edited on Views times
    \ No newline at end of file +存储器 - 数字逻辑电路 - 电子信息 | Balloon Party = 風船のパーティー = fuusen no party

    两类主要的半导体存储器是 RAM 和 ROM。

    • RAM (随机访问存储器) 类型的存储器在等量时间内可以访问所有的地址,并且可以按照任何顺序来选择这些地址以执行读或者写操作。所有的 RAM 都具有读和写的能力。因为 RAM 在电源关闭时会丢失所存储的数据,所以它是易失性内存。 RAM 系列 RAM 的两种类别是静 RAM (SRAM) 和动态 RAM (DRAM) 。当直流电源移走后, SRAM 和 DRAM 都会丢失存储的数据,因此被归类为易失性存储器。
      • 静态 RAM 通常使用锁存器作为存储元件,因而只要加上直流电源,就可以永远地存储数据。
        • 异步 SRAM (ASRAM)
        • 同步 SRAM (SB SRAM),具有突发特性
      • 动态 RAM 使用电容作为存储元件,如果没有对电容再充电就不能长期存储数据,再充电则称为刷新过程。
        • 快速页模式 DRAM (FPMDRAM)
        • 扩充数据输出 DRAM (EDO DRAM)
        • 突发 EDO DRAM (BEDO DRAM)
        • 同步 DRAM (SDRAM)
    • ROM (只读存储器) 类型的存储器将永久或者半永久地保存数据。数据可以从 ROM 中读出,但是和 RAM 不同,它没有写操作。ROM 和 RAM 相似,也是随机存储器,但是术语 RAM 的 传统意思是随机读 / 写存储器。本章将会介绍几种类型的 RAM 和 ROM。因为 ROM 在电源关闭 后仍然保存着数据,所以它是永久性存储器
      • MROM (Mask ROM) —— 掩膜 ROM,出场时已经固定,不能更改,适合大量生产,简单,便宜,非易失性。
      • PROM —— 可一次性编程 ROM
      • EPROM —— 可擦除 PROM,也就是可多次编程
      • UV EPROM —— 紫外线版本的 EPROM
      • EEPROM —— 电版本的

    对于给定的物理空间和成本,DRAM 可以存储比 SRAM 多得多的数据,因为 DRAM 单元更加简单,并在给定芯片面积里,因此可以比 SRAM 填充更多的单元

    # RAM

    # SRAM

    SRAM 基本单元(是 D - 触发器 ):

    sram单元

    32k×832k\times8 SRAM

    32kx8sram

    逻辑组成如下:

    32kx8sram逻辑图

    读:

    32kx8sram读

    写:

    32kx8sram写

    # DRAM

    DRAM 的基本单元,使用电容,电容充电表示 1 高电位,放电表示 0 低电位。

    mosdram单元

    DRAM 的 4 种基本操作:

    mosdram单元基本操作

    1M×11M\times1 DRAM 逻辑图:

    1mx1dram逻辑图

    DRAM 的读写周期:

    dram读写周期

    # ROM

    ROM(Read Only Memory), 即只读存储器,由于其在掉电后数据不会丢失,因此在早期的时候,ROM 芯片中的内容一般都是在生产的时候被烧写进去的,然后终生具有这些固定的内容。不过,随着技术的不断发展和革新,现在很多型号的 ROM 都已经可以被多次修改了,但是大家仍然习惯对其沿用 “只读存储器” 的称号。

    # MOS ROM

    MOS ROM 基本单元:

    mosrom单元

    256x4rom

    256x4rom逻辑图

    # ROM 设计组合逻辑电路

    实际上是枚举所有的输入输出,也就是真值表,存储到 ROM 里。

    输入作为地址,通过地址译码器,输出相应地址的数据。

    • 有黑点⚫️表示存储 1 ;
    • 否则表示存储 0 ;

    给出逻辑表达式(函数)时,求出标准的极小值只和的形式,依据极小值的下标确定画⚫️的位置

    # PROM

    PROM(Programmable Read Only Memory), 即可编程只读存储器,不过它也只能被程序写一次,以后就只能保存和使用这些数据了,所以 PROM 也称为一次可编程存储器。

    # EPROM

    EPROM(Erasable Programmable Read Only Memory),即可擦除可编程只读存储器。它解决了 PROM 只能编程一次的问题,不过其存储内容的擦除必须借助紫外线的长时间照射才能完成,所以 EPROM 又叫紫外线擦除可编程只读存储器。

    # EEPROM

    EEPROM(Electrically Erasable Programmable Read Only Memory),即电可擦除可编程只读存储器。它克服了 EPROM 在擦除时较为麻烦的问题,可以直接通过计算机程序对其进行重复的擦写,其擦除、重写的原理是通过使用高于常规的操作电压实现的。对于 EEPROM 来说,最重要的参数指标就是寿命问题,也就是可擦写的次数。

    # 闪存

    闪存 Flash Memory 是 EEPROM 的一种特殊形式,也就是变种。

    EEPROM 的擦除和重写都是针对整个芯片而言的,但 Flash 的数据删除和重写则是以块(块的容量一般在 256KB~20MB 之间)为单位的,因此 Flash 的更新速度要远比 EEPROM 快。不过由于数据的删除和重写不是以单个字节为单位的,因此 Flash 仍不能达到 RAM 的要求。

    Flash 又分为 NOR 型和 NAND 型两个大类,其中 NOR 型具有独立的地址线和数据线,适合频繁的随机读、写场合,适合程序代码的存储和直接运行,但容量通常较小,价格也比较贵;而 NAND 型的地址线和数据线是公用的,因此其本身的操作速度和频率就会偏低,但是容量相对较大,且价格也要便宜,所以适合大量资料的存储。

    # 存储器扩展

    # 字长扩展

    共享数据线和控制线,并联数据线。

    2 个 64k×464k\times4 ROM 的字长扩展得到个 64k×864k\times8 ROM :

    64kx4rom字长扩展64kx8rom

    # 字容量扩展

    514kx4ram字容量扩展1mx4ram

    💡 利用译码器

    # 字长字扩展结合

    # 特殊存储器

    # FIFO

    FIFO

    FIFO 寄存器模块的应用,实现数据整型:

    FIFO应用

    # LIFO

    类似栈的效果,LIFO 寄存器结构:

    LIFO结构

    Edited on Views times
    \ No newline at end of file diff --git a/ie/de/numerical-system/index.html b/ie/de/numerical-system/index.html index 5f5ba882..a54d8fe5 100644 --- a/ie/de/numerical-system/index.html +++ b/ie/de/numerical-system/index.html @@ -1 +1 @@ -数制 - 数字逻辑电路 - 电子信息 | Balloon Party = 風船のパーティー = fuusen no party

    # 数制

    # 数制的概念

    数制 (numerical system),又称 “计数制”,指用一组固定的符号和统一的规则来表示数值的方法;

    数制关注的内容:每一位的构成(使用什么符号即数码)和进位的规则;

    基本概念:

    • 基数 —— 使用的数码的个数,即每一位最多能用多少个符号来表示;基数是 N 则称为 N 进制, 即 “逢 N 进一”
    • 权 —— 每一位的 1 代表的大小;

    # 常用数制

    常用数制

    # 不同进制的转换

    # 其他进制 -> 十进制

    只需要按照 D=kiNiD=\sum k_i N^i 展开即可;

    • kik_i 为第 i 位的系数
    • NiN^i 为第 i 位的权
    • NN 为基数

    # 十进制 -> 二进制

    # 整数部分

    反复除 2 取余数,倒排(从右向左排列)(从低位到高位排列);

    # 小数部分

    反复乘 2 取整数部分,正排(从左向右排列)(从高位到低位排列);

    # 二进制 -> 十六进制

    # 整数部分

    从右向左(从低位到高位)每四个划为一组(不够的在左边补 0 ),

    # 小数部分

    从左向右(从高位到低位)每四个划为一组(不够的在右边补 0 );

    # 二进制 -> 八进制

    # 整数部分

    从右向左(从低位到高位)每三个划为一组(不够的在左边补 0 ),

    # 小数部分

    从左向右(从高位到低位)每三个划为一组(不够的在右边补 0 );

    # 十六进制 -> 二进制

    将每一位用对应的四位二进制数代替即可

    # 八进制 -> 二进制

    将每一位用对应的三位二进制数代替即可

    # 十进制 -> 八进制、十六进制

    先转换为二进制,再将二进制转换为八进制、十六进制

    # 八进制 <-> 十六进制

    先转换为二进制,再将二进制转换为目标进制

    # 二进制

    # 二进制算术运算

    特点:

    • 和十进制算数运算的规则基本相同,区别在于 “逢二进一”
    • 乘法和除法可以转化为移位操作与加减操作
    • 通过补码运算(即将引入的概念),可以将减法运算转换为加法运算;

    因此,数字电路普遍采用二进制进行算术运算

    Edited on Views times
    \ No newline at end of file +数制 - 数字逻辑电路 - 电子信息 | Balloon Party = 風船のパーティー = fuusen no party

    # 数制

    # 数制的概念

    数制 (numerical system),又称 “计数制”,指用一组固定的符号和统一的规则来表示数值的方法;

    数制关注的内容:每一位的构成(使用什么符号即数码)和进位的规则;

    基本概念:

    • 基数 —— 使用的数码的个数,即每一位最多能用多少个符号来表示;基数是 N 则称为 N 进制, 即 “逢 N 进一”
    • 权 —— 每一位的 1 代表的大小;

    # 常用数制

    常用数制

    # 不同进制的转换

    # 其他进制 -> 十进制

    只需要按照 D=kiNiD=\sum k_i N^i 展开即可;

    • kik_i 为第 i 位的系数
    • NiN^i 为第 i 位的权
    • NN 为基数

    # 十进制 -> 二进制

    # 整数部分

    反复除 2 取余数,倒排(从右向左排列)(从低位到高位排列);

    # 小数部分

    反复乘 2 取整数部分,正排(从左向右排列)(从高位到低位排列);

    # 二进制 -> 十六进制

    # 整数部分

    从右向左(从低位到高位)每四个划为一组(不够的在左边补 0 ),

    # 小数部分

    从左向右(从高位到低位)每四个划为一组(不够的在右边补 0 );

    # 二进制 -> 八进制

    # 整数部分

    从右向左(从低位到高位)每三个划为一组(不够的在左边补 0 ),

    # 小数部分

    从左向右(从高位到低位)每三个划为一组(不够的在右边补 0 );

    # 十六进制 -> 二进制

    将每一位用对应的四位二进制数代替即可

    # 八进制 -> 二进制

    将每一位用对应的三位二进制数代替即可

    # 十进制 -> 八进制、十六进制

    先转换为二进制,再将二进制转换为八进制、十六进制

    # 八进制 <-> 十六进制

    先转换为二进制,再将二进制转换为目标进制

    # 二进制

    # 二进制算术运算

    特点:

    • 和十进制算数运算的规则基本相同,区别在于 “逢二进一”
    • 乘法和除法可以转化为移位操作与加减操作
    • 通过补码运算(即将引入的概念),可以将减法运算转换为加法运算;

    因此,数字电路普遍采用二进制进行算术运算

    Edited on Views times
    \ No newline at end of file diff --git a/ie/de/sequential-circuit/index.html b/ie/de/sequential-circuit/index.html index 639567d6..c0774368 100644 --- a/ie/de/sequential-circuit/index.html +++ b/ie/de/sequential-circuit/index.html @@ -1 +1 @@ -时序逻辑电路 - 数字逻辑电路 - 电子信息 | Balloon Party = 風船のパーティー = fuusen no party

    # 基础元件

    # 555 Timer

    可能会考哦

    mono-stable one-shot (单稳态触发器)

    555-单稳态触发器

    astable multi-vibration oscillator (非稳态多谐振荡器)

    555-振荡器

    f=1.44(R1+2R2)C1DC=(R1+R2R1+2R2)100%f=\frac{1.44}{(R_1+2R_2)C_1}\\ DC=\big(\frac{R_1+R_2}{R_1+2R_2}\big)100\%

    # S-R 触发器

    Qn+1=S+RQnQ^{n+1}=S+\overline{R}Q^{n}

    # D 触发器

    Q=DQ'=D

    GATEGATE 高电平有效使能:

    Qn+1=GATED+GATEQnQ^{n+1}=GATE\cdot D+\overline{GATE}\cdot Q^{n}

    # ⭐️J-K 触发器

    Qn+1=JQn+QnKQ^{n+1}=J\overline{Q^{n}}+Q^{n}\overline{K}

    JK

    # 设计

    # 思路一

    1. State Diagram 状态转移图
      • 有多少状态决定使用多少触发器,nn 个状态应使用 log2n\lceil\log_2n\rceil
    2. Next-State Table 次态表(状态转移表)
    3. Flip-Flop Transition Table 触发器转换表,由触发器特性方程决定
    4. Karnaugh Maps 根据次态表、触发器转换表画出卡诺图
    5. Locgi Expressions for Flip-Flop Inputs 根据卡诺图化简得到逻辑表达式
    6. Counter Implementation 根据逻辑表达式设计逻辑电路

    # 思路二

    每一个输入也要视为一个状态位(类似于 QQ )来画卡诺图

    卡诺图每一个格子内容为:

    Q3n+1Q2n+1Q1n+1Q0n+1/CQ^{n+1}_3Q^{n+1}_2Q^{n+1}_1Q^{n+1}_0/C

    对于不可能的情况,也就是无关项,那么 5 个值统一为 X ,也就是:XXXX/XXXXX/X

    将内容拆开,则对应地画出 5 个卡诺图,然后分别化简,则能够得到:

    • 4 个状态 (转移) 方程
    • 1 个状态输出方程

    再根据选定的触发器的特性方程,则可以写出各个触发器的各个输入的驱动方程以及输出方程

    在一些芯片上,计数使能被简单地标记为 CTEN (或者一些诸如 G 之类的其他名称), 终端计数 (TC) 和一些 IC 计数器上的异步 (行波) 时钟输出 (RCO) 比较相似

    一些计数器:

    • 74HC161 十六进制计数器
    • 74HC190 十进制计数器
    \ No newline at end of file +时序逻辑电路 - 数字逻辑电路 - 电子信息 | Balloon Party = 風船のパーティー = fuusen no party

    # 基础元件

    # 555 Timer

    可能会考哦

    mono-stable one-shot (单稳态触发器)

    555-单稳态触发器

    astable multi-vibration oscillator (非稳态多谐振荡器)

    555-振荡器

    f=1.44(R1+2R2)C1DC=(R1+R2R1+2R2)100%f=\frac{1.44}{(R_1+2R_2)C_1}\\ DC=\big(\frac{R_1+R_2}{R_1+2R_2}\big)100\%

    # S-R 触发器

    Qn+1=S+RQnQ^{n+1}=S+\overline{R}Q^{n}

    # D 触发器

    Q=DQ'=D

    GATEGATE 高电平有效使能:

    Qn+1=GATED+GATEQnQ^{n+1}=GATE\cdot D+\overline{GATE}\cdot Q^{n}

    # ⭐️J-K 触发器

    Qn+1=JQn+QnKQ^{n+1}=J\overline{Q^{n}}+Q^{n}\overline{K}

    JK

    # 设计

    # 思路一

    1. State Diagram 状态转移图
      • 有多少状态决定使用多少触发器,nn 个状态应使用 log2n\lceil\log_2n\rceil
    2. Next-State Table 次态表(状态转移表)
    3. Flip-Flop Transition Table 触发器转换表,由触发器特性方程决定
    4. Karnaugh Maps 根据次态表、触发器转换表画出卡诺图
    5. Locgi Expressions for Flip-Flop Inputs 根据卡诺图化简得到逻辑表达式
    6. Counter Implementation 根据逻辑表达式设计逻辑电路

    # 思路二

    每一个输入也要视为一个状态位(类似于 QQ )来画卡诺图

    卡诺图每一个格子内容为:

    Q3n+1Q2n+1Q1n+1Q0n+1/CQ^{n+1}_3Q^{n+1}_2Q^{n+1}_1Q^{n+1}_0/C

    对于不可能的情况,也就是无关项,那么 5 个值统一为 X ,也就是:XXXX/XXXXX/X

    将内容拆开,则对应地画出 5 个卡诺图,然后分别化简,则能够得到:

    • 4 个状态 (转移) 方程
    • 1 个状态输出方程

    再根据选定的触发器的特性方程,则可以写出各个触发器的各个输入的驱动方程以及输出方程

    在一些芯片上,计数使能被简单地标记为 CTEN (或者一些诸如 G 之类的其他名称), 终端计数 (TC) 和一些 IC 计数器上的异步 (行波) 时钟输出 (RCO) 比较相似

    一些计数器:

    • 74HC161 十六进制计数器
    • 74HC190 十进制计数器
    \ No newline at end of file diff --git a/ie/de/shift-register/index.html b/ie/de/shift-register/index.html index 7f2bb222..84918461 100644 --- a/ie/de/shift-register/index.html +++ b/ie/de/shift-register/index.html @@ -1 +1 @@ -| Balloon Party = 風船のパーティー = fuusen no party

    # 串行输入 / 串行输出

    串行输入:串行输出

    # 串行输入 / 并行输出

    串行输入:并行输出

    # 并行输入 / 串行输出

    并行输入:串行输出

    # 并行输入 / 并行输出

    最简单

    # 双向移位寄存器

    4位双向位移

    # 74HC194

    通用 4 位双向移位寄存器

    74HC194

    74HC94波形图

    # 移位寄存器计数器

    # 约翰逊计数器

    约翰逊计数器

    约翰逊计数器波形图

    一般来说,约翰逊计 数器将会产生模 2n2n ,其中 nn 是计数器中级的个数

    # 环形计数器

    环形计数器

    环形计数器波形图

    # 移位寄存器应用

    # 精确延时控制

    串行输入 / 串行输出移位寄存器可以用来提供从输入到输出的时间延迟,此时间延迟是寄存器中数 nn 和时钟频率的函数

    从一个数字系统到另一个数字系统的串行数据传送,常常用来减少传输线中的导线数

    8位串并转换器

    当没有数据被传送时,串行数据线上就会有连续的 1 存在 ,0 位就是数据起始位,告诉串并转换器即将有 8 位数据序列到来:

    8位串并转换器数据串格式

    详细波形图解释工作流程如下:

    8位串并转换器波形图

    Edited on Views times
    \ No newline at end of file +| Balloon Party = 風船のパーティー = fuusen no party

    # 串行输入 / 串行输出

    串行输入:串行输出

    # 串行输入 / 并行输出

    串行输入:并行输出

    # 并行输入 / 串行输出

    并行输入:串行输出

    # 并行输入 / 并行输出

    最简单

    # 双向移位寄存器

    4位双向位移

    # 74HC194

    通用 4 位双向移位寄存器

    74HC194

    74HC94波形图

    # 移位寄存器计数器

    # 约翰逊计数器

    约翰逊计数器

    约翰逊计数器波形图

    一般来说,约翰逊计 数器将会产生模 2n2n ,其中 nn 是计数器中级的个数

    # 环形计数器

    环形计数器

    环形计数器波形图

    # 移位寄存器应用

    # 精确延时控制

    串行输入 / 串行输出移位寄存器可以用来提供从输入到输出的时间延迟,此时间延迟是寄存器中数 nn 和时钟频率的函数

    从一个数字系统到另一个数字系统的串行数据传送,常常用来减少传输线中的导线数

    8位串并转换器

    当没有数据被传送时,串行数据线上就会有连续的 1 存在 ,0 位就是数据起始位,告诉串并转换器即将有 8 位数据序列到来:

    8位串并转换器数据串格式

    详细波形图解释工作流程如下:

    8位串并转换器波形图

    Edited on Views times
    \ No newline at end of file diff --git a/ie/signals-and-systems/CTFT/index.html b/ie/signals-and-systems/CTFT/index.html index 667e3c91..9cbb5a97 100644 --- a/ie/signals-and-systems/CTFT/index.html +++ b/ie/signals-and-systems/CTFT/index.html @@ -1,4 +1,4 @@ -连续时间信号的傅里叶变换 - 信号与系统 - 电子信息 | Balloon Party = 風船のパーティー = fuusen no party

    # 公式表

    # 傅里叶变换性质

    傅里叶变换性质1傅里叶变换性质2

    # 基本傅里叶变换对

    基本傅里叶变换对表

    连续时间非周期信号傅里叶变换的推导基本思想

    • 非周期函数可以通过周期性延拓得到周期函数
    • 周期函数可以令基波周期趋于无穷大得到连续的非周期函数

    对于周期性矩形脉冲信号:

    ak=2T1TSa(kπ2T1T)Tak=2T1Sa(ωT1)\begin{aligned} a_k &= \frac{2T_1}{T}\text{Sa}(k\pi\frac{2T_1}{T})\\ Ta_k &= 2T_1\text{Sa}(\omega T_1) \end{aligned}

    其中:ω=kω0=2kπT\omega=k\omega_0=\frac{2k\pi}{T}

    两点之间的距离:

    Δω=(k+1)ω0kω0=ω0=2πT\Delta\omega=(k+1)\omega_0-k\omega_0=\omega_0=\frac{2\pi}{T}

    TT\to\infin 时有 ω0=2πT0\omega_0=\frac{2\pi}{T}\to 0 ,即两点之间距离趋于 0 ,函数包络线趋于连续函数。

    # 非周期信号

    # FT 公式推导

    连续时间非周期信号傅里叶变换的推导

    连续时间非周期信号傅里叶变换的推导

    周期函数 x~(t)\tilde{x}(t) 以 T 为周期,在 [T2,T2][-\frac{T}{2},\frac{T}{2}] 内与 x(t)x(t) 相等,它的傅里叶级数表示为:

    x~(t)=k=+akejkω0t\tilde{x}(t)=\sum_{k=-\infin}^{+\infin}a_k e^{jk\omega_0 t}

    波浪号表示周期性

    TT\to\infin 时,在 (,)(-\infin,\infin)x~(t)=x(t)\tilde{x}(t)=x(t)

    ak=1TT2T2x~(t)ejkω0tdt=1TT2T2x(t)ejkω0tdt=1Tx(t)ejkω0tdt\begin{aligned} a_k &= \frac{1}{T}{\int}_{-\frac{T}{2}}^{\frac{T}{2}} \tilde{x}(t) e^{-jk\omega_0 t}dt\\ &= \frac{1}{T}{\int}_{-\frac{T}{2}}^{\frac{T}{2}} x(t) e^{-jk\omega_0 t}dt\\ &= \frac{1}{T}{\int}_{-\infin}^{\infin} x(t) e^{-jk\omega_0 t}dt\\ \end{aligned}

    定义频谱:(傅里叶正变换)

    X(jω)=x(t)ejkω0tdtX(j\omega)={\int}_{-\infin}^{\infin} x(t) e^{-jk\omega_0 t}dt

    从而 aka_k 可以表示为:

    ak=1TX(jkω0)a_k=\frac{1}{T}X(jk\omega_0)

    因此,在 [T2,T2][-\frac{T}{2},\frac{T}{2}] 内有:

    x(t)=x~(t)=k=+akejkω0t=1Tk=+X(jkω0)ejkω0t=12πk=+X(jkω0)ejkω0tω0=12πk=+X(jω)ejωtΔω\begin{aligned} x(t)=\tilde{x}(t) &= \sum_{k=-\infin}^{+\infin} a_k e^{jk\omega_0 t}\\ &= \frac{1}{T} \sum_{k=-\infin}^{+\infin} X(jk\omega_0) e^{jk\omega_0 t}\\ &= \frac{1}{2\pi} \sum_{k=-\infin}^{+\infin} X(jk\omega_0) e^{jk\omega_0 t}\omega_0\\ &= \frac{1}{2\pi} \sum_{k=-\infin}^{+\infin} X(j\omega) e^{j\omega t}\Delta\omega\\ \end{aligned}

    TT\to\infin 时,Δω=ω00\Delta\omega=\omega_0\to0,即 Δωdω\Delta\omega\to d\omega 则得到(傅里叶反变换):

    x(t)=x~(t)=12π+X(jω)ejωtdω\begin{aligned} x(t)=\tilde{x}(t) &= \frac{1}{2\pi} \int_{-\infin}^{+\infin} X(j\omega) e^{j\omega t}d\omega\\ \end{aligned}

    从而得到傅里叶变换对

    这两个公式是 “一对” 的,要么同时可积,要么都不可积

    傅里叶正变换,分析公式,(ejωte^{j\omega t} 的线性组合)

    X(jω)=x(t)ejωtdtX(j\omega)={\int}_{-\infin}^{\infin} x(t) e^{-j\omega t}dt

    傅里叶反变换,综合公式,(获得 x(t)x(t) 的频谱)

    x(t)=12π+X(jω)ejωtdω\begin{aligned} x(t)=\frac{1}{2\pi} \int_{-\infin}^{+\infin} X(j\omega) e^{j\omega t}d\omega\\ \end{aligned}

    可以表示为:

    x(t)CTFTX(jω)F{x(t)}=X(jω)x(t)\stackrel{CTFT}{\longleftrightarrow}X(j\omega)\\\\ \mathscr{F}\{x(t)\}=X(j\omega)

    周期信号傅里叶级数的系数为其一个周期内信号傅里叶变换的等间隔采样:

    {x(t)X(jω)x~(t)akak=1TX(jkω0)\begin{cases} x(t)\lrarr X(j\omega)\\ \tilde{x}(t)\lrarr a_k \end{cases} \rArr a_k=\frac{1}{T}X(jk\omega_0)

    # 收敛性

    既然傅里叶变换是从周期信号的傅里叶级数引出的,傅里叶变换的收敛问题就应该和傅里叶级数的收敛相一致。所以也有对应的两组条件;同样地两组条件并不完全等价。它们都是傅里叶变换存在的充分条件。

    和周期信号的情况一样,当 x(t)x(t) 的傅里叶变换 X(jω)X(j\omega) 存在时,其反变换在间断点附近会产生 Gibbs 现象

    # 平方可积条件

    # Dirichlet 条件

    # 常见信号的 FT

    # 低通信号 (单侧)

    低通信号(单侧)1

    x(t)=eatu(t),a>0x(t)=e^{-at}u(t),\ \ \ a\gt0

    有正变换:

    X(jω)=0+eatejωtdt=1a+jωX(jω)=1a2+ω2ddωX(jω)=arctanωa\begin{aligned} X(j\omega) &=\int_{0}^{+\infin}e^{-at}e^{-j\omega t}dt=\frac{1}{a+j\omega}\\ |X(j\omega)| &=\frac{1}{\sqrt{a^2+\omega^2}}\\ \frac{d}{d\omega}|X(j\omega)| &=-\arctan\frac{\omega}{a} \end{aligned}

    # 公式表

    # 傅里叶变换性质

    傅里叶变换性质1傅里叶变换性质2

    # 基本傅里叶变换对

    基本傅里叶变换对表

    连续时间非周期信号傅里叶变换的推导基本思想

    • 非周期函数可以通过周期性延拓得到周期函数
    • 周期函数可以令基波周期趋于无穷大得到连续的非周期函数

    对于周期性矩形脉冲信号:

    ak=2T1TSa(kπ2T1T)Tak=2T1Sa(ωT1)\begin{aligned} a_k &= \frac{2T_1}{T}\text{Sa}(k\pi\frac{2T_1}{T})\\ Ta_k &= 2T_1\text{Sa}(\omega T_1) \end{aligned}

    其中:ω=kω0=2kπT\omega=k\omega_0=\frac{2k\pi}{T}

    两点之间的距离:

    Δω=(k+1)ω0kω0=ω0=2πT\Delta\omega=(k+1)\omega_0-k\omega_0=\omega_0=\frac{2\pi}{T}

    TT\to\infin 时有 ω0=2πT0\omega_0=\frac{2\pi}{T}\to 0 ,即两点之间距离趋于 0 ,函数包络线趋于连续函数。

    # 非周期信号

    # FT 公式推导

    连续时间非周期信号傅里叶变换的推导

    连续时间非周期信号傅里叶变换的推导

    周期函数 x~(t)\tilde{x}(t) 以 T 为周期,在 [T2,T2][-\frac{T}{2},\frac{T}{2}] 内与 x(t)x(t) 相等,它的傅里叶级数表示为:

    x~(t)=k=+akejkω0t\tilde{x}(t)=\sum_{k=-\infin}^{+\infin}a_k e^{jk\omega_0 t}

    波浪号表示周期性

    TT\to\infin 时,在 (,)(-\infin,\infin)x~(t)=x(t)\tilde{x}(t)=x(t)

    ak=1TT2T2x~(t)ejkω0tdt=1TT2T2x(t)ejkω0tdt=1Tx(t)ejkω0tdt\begin{aligned} a_k &= \frac{1}{T}{\int}_{-\frac{T}{2}}^{\frac{T}{2}} \tilde{x}(t) e^{-jk\omega_0 t}dt\\ &= \frac{1}{T}{\int}_{-\frac{T}{2}}^{\frac{T}{2}} x(t) e^{-jk\omega_0 t}dt\\ &= \frac{1}{T}{\int}_{-\infin}^{\infin} x(t) e^{-jk\omega_0 t}dt\\ \end{aligned}

    定义频谱:(傅里叶正变换)

    X(jω)=x(t)ejkω0tdtX(j\omega)={\int}_{-\infin}^{\infin} x(t) e^{-jk\omega_0 t}dt

    从而 aka_k 可以表示为:

    ak=1TX(jkω0)a_k=\frac{1}{T}X(jk\omega_0)

    因此,在 [T2,T2][-\frac{T}{2},\frac{T}{2}] 内有:

    x(t)=x~(t)=k=+akejkω0t=1Tk=+X(jkω0)ejkω0t=12πk=+X(jkω0)ejkω0tω0=12πk=+X(jω)ejωtΔω\begin{aligned} x(t)=\tilde{x}(t) &= \sum_{k=-\infin}^{+\infin} a_k e^{jk\omega_0 t}\\ &= \frac{1}{T} \sum_{k=-\infin}^{+\infin} X(jk\omega_0) e^{jk\omega_0 t}\\ &= \frac{1}{2\pi} \sum_{k=-\infin}^{+\infin} X(jk\omega_0) e^{jk\omega_0 t}\omega_0\\ &= \frac{1}{2\pi} \sum_{k=-\infin}^{+\infin} X(j\omega) e^{j\omega t}\Delta\omega\\ \end{aligned}

    TT\to\infin 时,Δω=ω00\Delta\omega=\omega_0\to0,即 Δωdω\Delta\omega\to d\omega 则得到(傅里叶反变换):

    x(t)=x~(t)=12π+X(jω)ejωtdω\begin{aligned} x(t)=\tilde{x}(t) &= \frac{1}{2\pi} \int_{-\infin}^{+\infin} X(j\omega) e^{j\omega t}d\omega\\ \end{aligned}

    从而得到傅里叶变换对

    这两个公式是 “一对” 的,要么同时可积,要么都不可积

    傅里叶正变换,分析公式,(ejωte^{j\omega t} 的线性组合)

    X(jω)=x(t)ejωtdtX(j\omega)={\int}_{-\infin}^{\infin} x(t) e^{-j\omega t}dt

    傅里叶反变换,综合公式,(获得 x(t)x(t) 的频谱)

    x(t)=12π+X(jω)ejωtdω\begin{aligned} x(t)=\frac{1}{2\pi} \int_{-\infin}^{+\infin} X(j\omega) e^{j\omega t}d\omega\\ \end{aligned}

    可以表示为:

    x(t)CTFTX(jω)F{x(t)}=X(jω)x(t)\stackrel{CTFT}{\longleftrightarrow}X(j\omega)\\\\ \mathscr{F}\{x(t)\}=X(j\omega)

    周期信号傅里叶级数的系数为其一个周期内信号傅里叶变换的等间隔采样:

    {x(t)X(jω)x~(t)akak=1TX(jkω0)\begin{cases} x(t)\lrarr X(j\omega)\\ \tilde{x}(t)\lrarr a_k \end{cases} \rArr a_k=\frac{1}{T}X(jk\omega_0)

    # 收敛性

    既然傅里叶变换是从周期信号的傅里叶级数引出的,傅里叶变换的收敛问题就应该和傅里叶级数的收敛相一致。所以也有对应的两组条件;同样地两组条件并不完全等价。它们都是傅里叶变换存在的充分条件。

    和周期信号的情况一样,当 x(t)x(t) 的傅里叶变换 X(jω)X(j\omega) 存在时,其反变换在间断点附近会产生 Gibbs 现象

    # 平方可积条件

    # Dirichlet 条件

    # 常见信号的 FT

    # 低通信号 (单侧)

    低通信号(单侧)1

    x(t)=eatu(t),a>0x(t)=e^{-at}u(t),\ \ \ a\gt0

    有正变换:

    X(jω)=0+eatejωtdt=1a+jωX(jω)=1a2+ω2ddωX(jω)=arctanωa\begin{aligned} X(j\omega) &=\int_{0}^{+\infin}e^{-at}e^{-j\omega t}dt=\frac{1}{a+j\omega}\\ |X(j\omega)| &=\frac{1}{\sqrt{a^2+\omega^2}}\\ \frac{d}{d\omega}|X(j\omega)| &=-\arctan\frac{\omega}{a} \end{aligned}

    低通信号(单侧)2

    低通信号(单侧)3

    # 低通信号 (双侧)

    低通信号(双侧)1

    x(t)=eat,a>0x(t)=e^{-a|t|},\ \ \ a\gt0

    有正变换:

    X(jω)=0eatejωtdt+0+eatejωtdt=1ajω+1a+jω=2aa2+ω2X(jω)=X(jω)\begin{aligned} X(j\omega) &=\int_{-\infin}^{0}e^{at}e^{-j\omega t}dt +\int_{0}^{+\infin}e^{-at}e^{-j\omega t}dt\\ &=\frac{1}{a-j\omega}+\frac{1}{a+j\omega}\\ &= \frac{2a}{a^2+\omega^2}\\ \\ |X(j\omega)| &= X(j\omega)\\ \end{aligned}

    低通信号(双侧)2

    # 冲激信号

    x(t)=δ(t)x(t)=\delta(t)

    有正变换:

    X(jω)=+δ(t)ejωtdt=1X(j\omega)=\int_{-\infin}^{+\infin}\delta(t)e^{-j\omega t}dt=1

    冲激信号正变换

    这表明 δ(t)\delta(t) 中包括了所有的频率成分,且所有频率分量的幅度、相位都相同。因此,系统的单位冲激响应 h(t)h(t) 才能完全描述一个 LTI 系统的特性,δ(t)\delta(t) 才在信号与系统分析中具有如此重要的意义。

    # 常数信号

    * 这个名字我乱起的

    x(t)=1x(t)=1

    因为:

    12π+2πδ(ω)ejωtdω=1\begin{aligned} \frac{1}{2\pi} \int_{-\infin}^{+\infin} 2\pi\ \delta(\omega)\ e^{j\omega t}d\omega = 1 \end{aligned}

    所以有反变换:

    X(jω)=2πδ(ω)\begin{aligned} X(j\omega) &= 2\pi\delta(\omega) \end{aligned}


    冲激信号和双侧阶跃信号 (我乱起名) 的对偶关系:

    δ(t)F1F2πδ(ω)\delta(t)\stackrel{F}{\longleftrightarrow}1\stackrel{F}{\longleftrightarrow}2\pi\delta(\omega)

    # 矩形信号

    矩形信号

    x(t)={1,t<T10,t>T1x(t)= \begin{cases} 1,\ \ \ |t|\lt T_1\\ 0,\ \ \ |t|\gt T_1 \end{cases}

    有正变换:

    X(jω)=T1T1ejωtdt=2sinωT1ω=2T1sinωT1ωT1=2T1Sa(ωT1)=2T1sinc(ωT1π)ak=X(jω)T=2T1TSa(kω0T1)=2T1TSa(2T1T)\begin{aligned} X(j\omega) &=\int_{-T_1}^{T_1} e^{-j\omega t}dt\\ &=\frac{2\sin{\omega T_1}}{\omega}\\ &=\frac{2T_1\sin{\omega T_1}}{\omega T_1}\\ &=2T_1\text{Sa}(\omega T_1) = 2T_1\text{sinc}(\frac{\omega T_1}{\pi})\\\\ a_k = \frac{X(j\omega)}{T} &= \frac{2T_1}{T}\text{Sa}(k\omega_0 T_1) = \frac{2T_1}{T}\text{Sa}(\frac{2T_1}{T}) \end{aligned}

    不同脉冲宽度对频谱的影响:

    不同脉冲宽度对矩形信号频谱的影响

    可见,信号宽度在时域和频域之间有一种相反的关系。

    # 采样信号

    X(jω)={1,ω<W0,ω>WX(j\omega)= \begin{cases} 1,\ \ \ |\omega|\lt W\\ 0,\ \ \ |\omega|\gt W \end{cases}

    有反变换:

    x(t)=12πW+Wejωtdω=sinWTπt=WπSa(Wt)=WπSa(Wtπ)\begin{aligned} x(t) &= \frac{1}{2\pi}\int_{-W}^{+W} e^{-j\omega t}d\omega\\ &=\frac{\sin{WT}}{\pi t}\\ &= \frac{W}{\pi}\text{Sa}(Wt) = \frac{W}{\pi}\text{Sa}(\frac{Wt}{\pi}) \end{aligned}

    采样信号

    可见 x(t)x(t) 是采样信号


    矩形信号和采样信号的对偶关系:(上矩形信号、下采样信号)

    矩形信号和采样信号对偶关系

    # 周期信号

    TakTa_k 无论在 TT 有限还是无限的时候都可以是有限的,所以我们可以将 TT 有限时的周期信号和 TT 无限时的非周期信号用傅里叶变换统一起来

    但是 aka_k 却不是,所以我们无法将周期信号和非周期信号都用傅里叶级数表示。

    # FT 公式推导

    考虑 X(jω)=2πδ(ωω0)X(j\omega)=2\pi\delta(\omega-\omega_0) 所对应的信号:

    x(t)=12π+X(jω)ejωtdω=+δ(ωω0)ejωtdω=ejω0t\begin{aligned} x(t) &=\frac{1}{2\pi} \int_{-\infin}^{+\infin} X(j\omega)\ e^{j\omega t}d\omega\\ &=\int_{-\infin}^{+\infin} \delta(\omega-\omega_0)\ e^{j\omega t}d\omega\\ &= e^{j\omega_0 t} \end{aligned}

    于是对于表示为傅里叶级数的周期信号 x(t)x(t)

    x(t)=k=+akejkω0tx(t)=\sum_{k=-\infin}^{+\infin}a_k\ e^{jk\omega_0 t}

    就有频谱正变换

    X(jω)=2πk=+akδ(ωω0)X(j\omega)=2\pi \sum_{k=-\infin}^{+\infin} a_k\ \delta(\omega-\omega_0)

    这表明周期信号的傅里叶正变换由一系列位于谐波频率处的冲激组成,其中 kω0k\omega_0 处的冲激强度为对应傅里叶级数系数 aka_k2π2\pi 倍。

    # 常见信号的 FT

    # 正弦信号

    x(t)=sin(ω0t)=12j(ejω0tejω0t)x(t)=\sin(\omega_0 t)=\frac{1}{2j}(e^{j\omega_0 t}-e^{-j\omega_0 t})

    有正变换:

    X(jω)=πj[δ(ωω0)δ(ω+ω0)]X(j\omega)=\frac{\pi}{j}[\delta(\omega-\omega_0)-\delta(\omega+\omega_0)]

    正弦信号

    # 余弦信号

    x(t)=cos(ω0t)=12(ejω0t+ejω0t)x(t)=\cos(\omega_0 t)=\frac{1}{2}(e^{j\omega_0 t}+e^{-j\omega_0 t})

    有正变换:

    X(jω)=πj[δ(ωω0)+δ(ω+ω0)]X(j\omega)=\frac{\pi}{j}[\delta(\omega-\omega_0)+\delta(\omega+\omega_0)]

    余弦信号

    # 均匀冲激串

    x(t)=n=+δ(tnT)x(t)=\sum_{n=-\infin}^{+\infin}\delta(t-nT)

    其中 T=2πω0T=\frac{2\pi}{\omega_0}

    其傅里叶系数:

    ak=1TT2T2δ(t)ej2πTktdt=1TT2T2δ(t)dt=1T\begin{aligned} a_k &=\frac{1}{T}\int_{-\frac{T}{2}}^{\frac{T}{2}}\delta(t)\ e^{-j\frac{2\pi}{T}kt}dt\\ &=\frac{1}{T}\int_{-\frac{T}{2}}^{\frac{T}{2}}\delta(t)dt\\ &=\frac{1}{T} \end{aligned}

    可得傅里叶正变换:

    X(jω)=2πTk=+δ(ω2πTk)X(j\omega)=\frac{2\pi}{T}\sum_{k=-\infin}^{+\infin}\delta(\omega-\frac{2\pi}{T}k)

    均匀冲激串

    # 周期性矩形脉冲

    其傅里叶系数:

    ak=2T1TSa(πk2T1T)a_k=\frac{2T_1}{T}\text{Sa}(\pi k\frac{2T_1}{T})

    有傅里叶正变换:

    X(jω)=4πT1Tk=+Sa(πk2T1T)δ(ω2πTk)=2T1ω0k=+Sa(ω0T1k)δ(ωω0k)X(j\omega)=\frac{4\pi T_1}{T}\sum_{k=-\infin}^{+\infin}\text{Sa}(\pi k\frac{2T_1}{T})\ \delta(\omega-\frac{2\pi}{T}k) =2T_1\omega_0\sum_{k=-\infin}^{+\infin}\text{Sa}(\omega_0T_1\cdot k)\ \delta(\omega-\omega_0\cdot k)

    其中 ω0=2πT\omega_0=\frac{2\pi}{T} 为谱线间隔。

    # 特殊信号 FT

    # 符号函数

    符号函数

    可以将符号函数定义为:

    Sgn(t)=lima0[eatu(t)eatu(t)]\text{Sgn}(t)=\lim_{a\to0}[e^{-at}u(t)-e^{at}u(-t)]

    其傅里叶变换为:

    F{Sgn(t)}=lima0[1a+jω1ajω]=lima0j2ωa2+(jω)2=2jω\begin{aligned} \mathscr{F}\{\text{Sgn}(t)\} &=\lim_{a\to0}[\frac{1}{a+j\omega}-\frac{1}{a-j\omega}]\\ &=\lim_{a\to0}\frac{-j2\omega}{a^2+(j\omega)^2}\\ &=\frac{2}{j\omega} \end{aligned}

    # 单位阶跃函数

    u(t)u(t) 分解为偶部和奇部有:

    u(t)=ue(t)+uo(t)ue(t)=12uo(t)=12Sgn(t)\begin{aligned} &u(t)=u_e(t)+u_o(t)\\ &u_e(t)=\frac{1}{2}\\ &u_o(t)=\frac{1}{2}\text{Sgn}(t) \end{aligned}

    1CTFT2πδ(ω)1\stackrel{CTFT}{\longleftrightarrow}2\pi\delta(\omega)Sgn(t)CTFT2jω\text{Sgn}(t)\stackrel{CTFT}{\longleftrightarrow}\frac{2}{j\omega} ,因此有:

    u(t)F1jω+πδ(ω)u(t)\stackrel{F}{\longleftrightarrow}\frac{1}{j\omega}+\pi\delta(\omega)

    # 性质

    # 线性

    x(t)FX(jω),y(t)FY(jω)ax(t)+by(t)FaX(jω)+bY(jω)x(t)\stackrel{F}{\longleftrightarrow}X(j\omega)\ \ ,\ \ y(t)\stackrel{F}{\longleftrightarrow}Y(j\omega)\\ \dArr\\ ax(t)+by(t)\stackrel{F}{\longleftrightarrow}aX(j\omega)+bY(j\omega)

    # 时移特性

    信号的时移会导致其频谱产生一个线性相移

    x(t)FX(jω)x(tt0)FX(jω)ejωt0x(t)\stackrel{F}{\longleftrightarrow}X(j\omega)\\ \dArr\\ x(t-t_0)\stackrel{F}{\longleftrightarrow}X(j\omega)e^{-j\omega t_0}


    证明

    F{x(tt0)}=+x(tt0)ejωtdt=+x(t)ejω(t+t0)dt=ejωt0+x(t)ejωtdt=ejωt0X(jω)\begin{aligned} \mathscr{F}\{x(t-t_0)\} &=\int_{-\infin}^{+\infin}x(t-t_0)\ e^{-j\omega t}dt\\ &=\int_{-\infin}^{+\infin}x(t')\ e^{-j\omega(t'+t_0)}dt'\\ &=e^{-j\omega t_0}\int_{-\infin}^{+\infin}x(t')\ e^{-j\omega t'}dt'\\ &=e^{-j\omega t_0}X(j\omega) \end{aligned}

    # 共轭对称性

    x(t)FX(jω)x(t)FX(jω)x(t)\stackrel{F}{\longleftrightarrow}X(j\omega)\\ \dArr\\ x^*(t)\stackrel{F}{\longleftrightarrow}X^*(-j\omega)

    特别地,当 x(t)x(t) 是实信号 (实函数),则有频谱: X(jω)=X(jω)X(j\omega)=X^*(-j\omega)共轭偶函数实幅偶,虚相奇


    证明

    x(t)x(t) 的傅里叶正变换两侧同时取共轭有:

    X(jω)=[x(t)ejkω0t]dt=x(t)ejkω0tdt\begin{aligned} X^*(j\omega) &={\int}_{-\infin}^{\infin} [x(t) e^{-jk\omega_0 t}]^* dt\\ &={\int}_{-\infin}^{\infin} x(t)^* e^{jk\omega_0 t} dt\\ \end{aligned}

    所以:

    X(jω)=x(t)ejkω0tdt\begin{aligned} X^*(-j\omega) &={\int}_{-\infin}^{\infin} x(t)^* e^{-jk\omega_0 t} dt\\ \end{aligned}

    即得证:

    x(t)FX(jω)x^*(t)\stackrel{F}{\longleftrightarrow}X^*(-j\omega)


    x(t)x(t) 是实信号时有 x(t)=x(t)x(t)=x^*(t) ,于是有:

    X(jω)=X(jω)=Re[X(jω)]+jIm[X(jω)]X(j\omega)=X^*(-j\omega)=\text{Re}[X(-j\omega)]+j\ \text{Im}[X(-j\omega)]

    共轭偶函数:

    • 部是函数:Re[X(jω)]=Re[X(jω)]\text{Re}[X(j\omega)]=\text{Re}[X(-j\omega)]
    • 部是函数:Im[X(jω)]=Im[X(jω)]\text{Im}[X(j\omega)]=-\text{Im}[X(-j\omega)]
    • 幅度函数:X(jω)=X(jω)|X(j\omega)|=|X(-j\omega)|
    • 相位函数:ArgX(jω)=ArgX(jω)\text{Arg}X(j\omega)=-\text{Arg}X(-j\omega)

    # 奇偶对称性

    • 偶信号的傅里叶 (正) 变换还是偶函数
    • 实偶信号的傅里叶 (正) 变换还是实偶函数
    • 奇信号的傅里叶 (正) 变换还是奇函数
    • 实奇信号的傅里叶 (正) 变换是虚奇函数

    用表格总结奇偶性、是否为实函数的不同情况下的傅里叶正变换奇偶情况:

    x(t)x(t)偶函数:x(t)=x(t)x(t)=x(-t)奇函数:x(t)=x(t)x(t)=-x(-t)
    一般的X(jω)=X(jω)X(j\omega)=X(-j\omega)X(jω)=X(jω)X(j\omega)=-X(-j\omega)
    x(t)x(t) 是实函数:x(t)=x(t)x(t)=x^*(t)X(jω)=X(jω)=X(jω)=X(jω)X(j\omega)=X(-j\omega)=X^*(j\omega)=X^*(-j\omega)X(jω)=X(jω)=X(jω)X(j\omega)=-X(-j\omega)=X^*(-j\omega)

    # 奇偶分解

    实信号 x(t)x(t) 若有奇偶分解 x(t)=xe(t)+xo(t)x(t)=x_e(t)+x_o(t) ,则有:

    X(jω)=Xe(jω)+Xo(jω)xe(t)FXe(jω)xo(t)FXo(jω)X(j\omega)=X_e(j\omega)+X_o(j\omega)\\ x_e(t)\stackrel{F}{\longleftrightarrow}X_e(-j\omega)\\ x_o(t)\stackrel{F}{\longleftrightarrow}X_o(-j\omega)

    # 时域微分特性

    可将微分运算转变为代数运算

    x(t)FX(jω)ddtx(t)FjωX(jω)x(t)\stackrel{F}{\longleftrightarrow}X(j\omega)\\ \dArr\\ \frac{d}{dt}x(t)\stackrel{F}{\longleftrightarrow}j\omega X(j\omega)

    可以推广到高次微分,在后面的微分方程分析 LTI 系统中会用到:

    dkdtkx(t)F(jω)kX(jω)\frac{d^k}{dt^k}x(t)\stackrel{F}{\longleftrightarrow}(j\omega)^k X(j\omega)


    证明:傅里叶反变换两侧同时对 tt 微分即可

    x(t)=12π+X(jω)ejωtdωddtx(t)=12π+X(jω)ddtejωtdω=12π+jωX(jω)ejωtdω\begin{aligned} x(t) &=\frac{1}{2\pi} \int_{-\infin}^{+\infin} X(j\omega) e^{j\omega t}d\omega\\ \frac{d}{dt}x(t) &=\frac{1}{2\pi}\int_{-\infin}^{+\infin} X(j\omega)\frac{d}{dt}e^{j\omega t}d\omega\\ &=\frac{1}{2\pi}\int_{-\infin}^{+\infin} j\omega X(j\omega)\ e^{j\omega t}d\omega\\ \end{aligned}

    # 时域积分特性

    可将积分运算转变为代数运算

    x(t)FX(jω)tx(τ)dτF12πX(jω)+πX(0)δ(ω)x(t)\stackrel{F}{\longleftrightarrow}X(j\omega)\\ \dArr\\ \int_{-\infin}^{t}x(\tau)d\tau\stackrel{F}{\longleftrightarrow}\frac{1}{2\pi}X(j\omega)+\pi X(0)\delta(\omega)


    证明:傅里叶正变换两侧同时对 τ\tau 积分即可

    F{tx(τ)dτ}=+[tx(τ)dτ]ejωtdt=+[+x(τ)u(tτ)dτ]ejωtdt=+x(τ)[+ejωtu(tτ)dt]dτ=+x(τ)[+ejω(tτ)jωτu(tτ)d(tτ)]dτ=+x(τ)ejωτ[+ejωsu(s)ds]dτ=[1jω+πδ(ω)]+x(τ)ejωτdτ=[1jω+πδ(ω)]X(jω)=1jωX(jω)+πX(0)δ(ω)\begin{aligned} \mathscr{F}\{\int_{-\infin}^{t}x(\tau)d\tau\} &=\int_{-\infin}^{+\infin}[\int_{-\infin}^{t}x(\tau)d\tau]e^{-j\omega t}dt\\ &=\int_{-\infin}^{+\infin}[\int_{-\infin}^{+\infin}x(\tau)u(t-\tau)d\tau]e^{-j\omega t}dt\\ &=\int_{-\infin}^{+\infin}x(\tau)[\int_{-\infin}^{+\infin}e^{-j\omega t}u(t-\tau)dt]d\tau\\ &=\int_{-\infin}^{+\infin}x(\tau)[\int_{-\infin}^{+\infin}e^{-j\omega(t-\tau)-j\omega\tau}u(t-\tau)d(t-\tau)]d\tau\\ &=\int_{-\infin}^{+\infin}x(\tau)e^{-j\omega\tau}[\int_{-\infin}^{+\infin}e^{-j\omega s}u(s)ds]d\tau\\ &=[\frac{1}{j\omega}+\pi\delta(\omega)]\int_{-\infin}^{+\infin}x(\tau)e^{-j\omega \tau}d\tau\\ &=[\frac{1}{j\omega}+\pi\delta(\omega)]X(j\omega)\\ &=\frac{1}{j\omega}X(j\omega)+\pi X(0)\delta(\omega) \end{aligned}

    其中 u(s)CTFT1jω+πδ(ω)u(s)\stackrel{CTFT}{\longleftrightarrow}\frac{1}{j\omega}+\pi\delta(\omega) 是单位阶跃函数的傅里叶变换

    # 时域和频域的尺度变换

    信号如果在时域扩展 a 倍,则其带宽相应压缩 a 倍,反之亦然

    x(t)FX(jω)x(at)F1aX(jωa)x(t)\stackrel{F}{\longleftrightarrow}X(j\omega)\\ \dArr\\ x(at)\stackrel{F}{\longleftrightarrow}\frac{1}{|a|}X(\frac{j\omega}{a})


    证明:

    F{x(at)}=+x(at)ejωtdt=1a+x(τ)ejωaτdτ=1aX(jωa)\begin{aligned} \mathscr{F}\{x(at)\} &=\int_{-\infin}^{+\infin}x(at)e^{-j\omega t}dt\\ &=\frac{1}{|a|}\int_{-\infin}^{+\infin}x(\tau)e^{-j\frac{\omega}{a}\tau}d\tau\\ &=\frac{1}{|a|}X(\frac{j\omega}{a}) \end{aligned}

    # 对偶性

    由对偶性可以方便地将时域的某些特性对偶到频域

    x(t)CTFTX(jω)X(jt)CTFT2πx(ω)x(t)\stackrel{CTFT}{\longleftrightarrow}X(j\omega)\\ \dArr\\ X(jt)\stackrel{CTFT}{\longleftrightarrow}2\pi x(-\omega)


    证明:

    x(t)=12π+X(jω)ejωtdω2πx(t)=+X(jω)ejωtdω2πx(ω)=+X(jt)ejωtdt2πx(ω)=+X(jt)ejωtdt\begin{aligned} x(t) &=\frac{1}{2\pi} \int_{-\infin}^{+\infin} X(j\omega) e^{j\omega t}d\omega\\ 2\pi x(t) &=\int_{-\infin}^{+\infin} X(j\omega) e^{j\omega t}d\omega\\ 2\pi x(\omega) &=\int_{-\infin}^{+\infin} X(jt) e^{j\omega t}dt\\ 2\pi x(-\omega) &=\int_{-\infin}^{+\infin} X(jt) e^{-j\omega t}dt\\ \end{aligned}

    # 移频特性

    将时域的时移性对偶可以得到频域的移频特性

    x(t)FX(jω)x(t)ejω0tFX[j(ωω0)]x(t)\stackrel{F}{\longleftrightarrow}X(j\omega)\\ \dArr\\ x(t)e^{j\omega_0 t}\stackrel{F}{\longleftrightarrow}X[j(\omega-\omega_0)]


    证明:

    x(t)FX(jω)对偶性X(jt)F2πx(ω)时移特性X(j(tt0))F2πx(ω)ejω0t对偶性2πx(t)ejω0tF2πX[j(ωω0)]x(t)ejω0tFX[j(ωω0)]x(t)\stackrel{F}{\longleftrightarrow}X(j\omega)\\ \dArr\text{对偶性}\\ X(jt)\stackrel{F}{\longleftrightarrow}2\pi x(-\omega)\\ \dArr\text{时移特性}\\ X(j(t-t_0))\stackrel{F}{\longleftrightarrow}2\pi x(-\omega)e^{-j\omega_0 t}\\ \dArr\text{对偶性}\\ 2\pi x(-t)e^{-j\omega_0 t}\stackrel{F}{\longleftrightarrow}2\pi X[j(-\omega-\omega_0)]\\ \dArr\\ x(t)e^{j\omega_0 t}\stackrel{F}{\longleftrightarrow}X[j(\omega-\omega_0)]

    # 频域微分特性

    x(t)FX(jω)jtx(t)FddωX(jω)x(t)\stackrel{F}{\longleftrightarrow}X(j\omega)\\ \dArr\\ -jtx(t)\stackrel{F}{\longleftrightarrow}\frac{d}{d\omega}X(j\omega)

    # 频域积分特性

    x(t)FX(jω)x(t)jt+πx(0)δ(t)FωX(jτ)dτx(t)\stackrel{F}{\longleftrightarrow}X(j\omega)\\ \dArr\\ \frac{x(t)}{-jt}+\pi x(0)\delta(t)\stackrel{F}{\longleftrightarrow}\int_{-\infin}^{\omega}X(j\tau)d\tau

    # Parseval 定理

    x(t)FX(jω)+x(t)2dt=12π+X(jω)2dωx(t)\stackrel{F}{\longleftrightarrow}X(j\omega)\\ \dArr\\ \int_{-\infin}^{+\infin}|x(t)|^2dt=\frac{1}{2\pi}\int_{-\infin}^{+\infin}|X(j\omega)|^2d\omega

    这表明:信号的能量既可以在时域求得,也可以在频域求得。由于 X(jω)2|X(j\omega)^2| 表示了信号能量在频域的分布,因而称其为 ==“能量谱密度” 函数 ==。

    # ⭐️卷积特性

    x(t)FX(jω),h(t)FH(jω)x(t)h(t)FX(jω)H(jω)x(t)\stackrel{F}{\longleftrightarrow}X(j\omega)\ \ \ ,\ \ \ h(t)\stackrel{F}{\longleftrightarrow}H(j\omega)\\ \dArr\\ x(t)*h(t)\stackrel{F}{\longleftrightarrow}X(j\omega)H(j\omega)

    本质上,卷积特性的成立正是因为复指数信号 ejωte^{jωt} 是一切 LTI 系统的特征函数,H(jω)H(j\omega) 则是对应的特征值


    证明

    y(t)=x(t)h(t)=+x(τ)h(tτ)dτy(t)=x(t)*h(t)=\int_{-\infin}^{+\infin}x(\tau)\ h(t-\tau)\ d\tau ,求其傅里叶正变换为:

    F{y(t)}=Y(jω)=+ejωt+x(τ)h(tτ)dτdt=+ejω(tτ)jωτ+x(τ)h(tτ)dτd(tτ)=+ejωsjωτ+x(τ)h(s)dτds=+ejωsejωτ+x(τ)h(s)dτds=+x(τ)ejωτdτ+h(s)ejωsds=X(jω)H(jω)\begin{aligned} \mathscr{F}\{y(t)\}=Y(j\omega) &=\int_{-\infin}^{+\infin} e^{-j\omega t}\int_{-\infin}^{+\infin}x(\tau)\ h(t-\tau)\ d\tau\ dt\\ &=\int_{-\infin}^{+\infin} e^{-j\omega(t-\tau)-j\omega \tau}\int_{-\infin}^{+\infin}x(\tau)\ h(t-\tau)\ d\tau\ d(t-\tau)\\ &=\int_{-\infin}^{+\infin} e^{-j\omega s-j\omega \tau}\int_{-\infin}^{+\infin}x(\tau)\ h(s)\ d\tau\ ds\\ &=\int_{-\infin}^{+\infin}e^{-j\omega s}e^{-j\omega \tau}\int_{-\infin}^{+\infin}x(\tau)\ h(s)\ d\tau\ ds\\ &=\int_{-\infin}^{+\infin}x(\tau)\ e^{-j\omega \tau}\ d\tau\int_{-\infin}^{+\infin}h(s)\ e^{-j\omega s}\ ds\\ &=X(j\omega)H(j\omega) \end{aligned}

    另外,可将 x(t)x(t) 视为复指数分量 ejωte^{j\omega t} 的线性组合,每个 ejωte^{j\omega t} 通过 LTI 系统时都要受到系统对应的特征值 H(jω)H(j\omega) 的加权


    LTI 系统的单位冲激响应 h(t)h(t) 与频率响应 H(jω)H(j\omega) 一一对应,因而 LTI 系统可以由其频率响应完全表征。

    ⚠️ 并非任何系统的频率响应都存在。

    用频率响应表征系统时,一般都限于对稳定系统。

    稳定性保证了 h(t)h(t) 的绝对可积,结合 Dirichlet 的另外两个条件(实际系统一般满足),则频率响应必然存在

    # ⭐️相乘特性

    s(t)FS(jω),p(t)FP(jω)s(t)p(t)F12πS(jω)P(jω)s(t)\stackrel{F}{\longleftrightarrow}S(j\omega)\ \ \ ,\ \ \ p(t)\stackrel{F}{\longleftrightarrow}P(j\omega)\\ \dArr\\ s(t)p(t)\stackrel{F}{\longleftrightarrow}\frac{1}{2\pi}S(j\omega)*P(j\omega)


    证明:利用对偶性可以从卷积性质得出相乘性质

    由对偶性有:

    s(t)FS(jω),p(t)FP(jω)S(jt)F2πS(ω),P(jt)F2πP(ω)s(t)\stackrel{F}{\longleftrightarrow}S(j\omega)\ \ \ ,\ \ \ p(t)\stackrel{F}{\longleftrightarrow}P(j\omega)\\ \dArr\\ S(jt)\stackrel{F}{\longleftrightarrow}2\pi S(-\omega)\ \ \ ,\ \ \ P(jt)\stackrel{F}{\longleftrightarrow}2\pi P(-\omega)\\

    由卷积特性有:

    S(jt)P(jt)F4π2s(ω)p(ω)S(jt)*P(jt)\stackrel{F}{\longleftrightarrow}4\pi^2s(-\omega)p(-\omega)

    由对偶性有:

    4π2s(t)p(t)F2πS(jω)P(jω)s(t)p(t)F12πS(jω)P(jω)\begin{aligned} 4\pi^2s(-t)p(-t)&\stackrel{F}{\longleftrightarrow}2\pi S(-j\omega)*P(-j\omega)\\ s(-t)p(-t)&\stackrel{F}{\longleftrightarrow}\frac{1}{2\pi}S(-j\omega)*P(-j\omega) \end{aligned}

    # 连续时间 LTI 系统的分析

    利用傅里叶变换及其特性对 LTI 系统分析

    # 频域分析法

    根据卷积特性,可以对 LTI 系统进行频域分析,其过程为:

    1. 求出(或已知、根据系统描述直接得出)系统单位冲激响应 h(t)h(t)
    2. 求输入 x(t)x(t) 的傅里叶正变换:x(t)CTFTX(jω)x(t)\stackrel{CTFT}{\longleftrightarrow}X(j\omega)
    3. 求单位冲激响应 h(t)h(t) 的傅里叶正变换:h(t)CTFTH(jω)h(t)\stackrel{CTFT}{\longleftrightarrow}H(j\omega)
    4. 若需要求 Y(jω)Y(j\omega) ,由傅里叶变换卷积特性得:Y(jω)=X(jω)H(jω)Y(j\omega)=X(j\omega)H(j\omega)
    5. 若需要求 y(t)y(t)
      • x(t)x(t) 是非周期的(或非周期部分):
        • 往往由傅里叶反变换求得:y(t)=F1{Y(jω)}=12π+Y(jω)ejωtdωy(t)=\mathscr{F}^{-1}\{Y(j\omega)\}=\frac{1}{2\pi} \int_{-\infin}^{+\infin} Y(j\omega) e^{j\omega t}d\omega
        • 对有理函数求傅里叶反变换通常采用部分分式展开和利用常用变换对进行
      • x(t)x(t) 是周期的(或周期部分):
        • Y(jω)=X(jω)×H(jω)|Y(j\omega)|=|X(j\omega)|\times|H(j\omega)|
        • argY(jω)=argX(jω)+argH(jω)\text{arg}Y(j\omega)=\text{arg}X(j\omega)+\text{arg}H(j\omega)

    # 线性常微分方程

    * 线性常系数微分方程

    对于由线性常系数微分方程描述的 LTI 系统:

    k=0Nakdky(t)dtk=k=0Mbkdkx(t)dtk\sum_{k=0}^{N}a_k\frac{d^ky(t)}{dt^k}=\sum_{k=0}^{M}b_k\frac{d^kx(t)}{dt^k}

    对方程两边进行傅里叶变换,有:

    k=0Nak(jω)kY(jω)=k=0Mbk(jω)kX(jω)Y(jω)k=0Nak(jω)k=X(jω)k=0Mbk(jω)kY(jω)X(jω)=k=0Mbk(jω)kk=0Nak(jω)k\begin{aligned} \sum_{k=0}^{N}a_{k}(j\omega)^{k}Y(j\omega) &= \sum_{k=0}^{M}b_{k}(j\omega)^{k}X(j\omega)\\ Y(j\omega)\sum_{k=0}^{N}a_{k}(j\omega)^{k} &= X(j\omega)\sum_{k=0}^{M}b_{k}(j\omega)^{k}\\ {Y(j\omega)\over X(j\omega)} &= {\sum_{k=0}^{M}b_{k}(j\omega)^{k}\over\sum_{k=0}^{N}a_{k}(j\omega)^{k}} \end{aligned}

    由 CTFT 卷积特性有:

    Y(jω)=X(jω)H(jω)Y(j\omega)=X(j\omega)H(j\omega)

    可得:

    H(jω)=Y(jω)X(jω)=k=0Mbk(jω)kk=0Nak(jω)kH(j\omega)={Y(j\omega)\over X(j\omega)}={\sum_{k=0}^{M}b_{k}(j\omega)^{k}\over\sum_{k=0}^{N}a_{k}(j\omega)^{k}}

    可见由线性常系数微分方程描述的 LTI 系统的频率响应是一个有理函数。

    若要求 h(t)h(t) ,也就是系统的单位冲激响应,则往往可以对有理函数(有理分式)形式的 H(jω)H(j\omega) 变形化得到常用变换对的表示形式,再利用常用变换对反变换得到 h(t)h(t)


    反之,若已知一个 LTI 系统的频率响应 H(jω)=Y(jω)X(jω)H(j\omega)={Y(j\omega)\over X(j\omega)} ,则可以列出两侧分别是关于 Y(jω)Y(j\omega)X(jω)X(j\omega) 的方程,两侧同时进行傅里叶反变换,(注意将 jωj\omega 视为一个整体)则可得到描述该 LTI 系统的线性常系数微分方程

    \ No newline at end of file +M834 80h400000v40h-400000z"/>1=arctanaω

    低通信号(单侧)2

    低通信号(单侧)3

    # 低通信号 (双侧)

    低通信号(双侧)1

    x(t)=eat,a>0x(t)=e^{-a|t|},\ \ \ a\gt0

    有正变换:

    X(jω)=0eatejωtdt+0+eatejωtdt=1ajω+1a+jω=2aa2+ω2X(jω)=X(jω)\begin{aligned} X(j\omega) &=\int_{-\infin}^{0}e^{at}e^{-j\omega t}dt +\int_{0}^{+\infin}e^{-at}e^{-j\omega t}dt\\ &=\frac{1}{a-j\omega}+\frac{1}{a+j\omega}\\ &= \frac{2a}{a^2+\omega^2}\\ \\ |X(j\omega)| &= X(j\omega)\\ \end{aligned}

    低通信号(双侧)2

    # 冲激信号

    x(t)=δ(t)x(t)=\delta(t)

    有正变换:

    X(jω)=+δ(t)ejωtdt=1X(j\omega)=\int_{-\infin}^{+\infin}\delta(t)e^{-j\omega t}dt=1

    冲激信号正变换

    这表明 δ(t)\delta(t) 中包括了所有的频率成分,且所有频率分量的幅度、相位都相同。因此,系统的单位冲激响应 h(t)h(t) 才能完全描述一个 LTI 系统的特性,δ(t)\delta(t) 才在信号与系统分析中具有如此重要的意义。

    # 常数信号

    * 这个名字我乱起的

    x(t)=1x(t)=1

    因为:

    12π+2πδ(ω)ejωtdω=1\begin{aligned} \frac{1}{2\pi} \int_{-\infin}^{+\infin} 2\pi\ \delta(\omega)\ e^{j\omega t}d\omega = 1 \end{aligned}

    所以有反变换:

    X(jω)=2πδ(ω)\begin{aligned} X(j\omega) &= 2\pi\delta(\omega) \end{aligned}


    冲激信号和双侧阶跃信号 (我乱起名) 的对偶关系:

    δ(t)F1F2πδ(ω)\delta(t)\stackrel{F}{\longleftrightarrow}1\stackrel{F}{\longleftrightarrow}2\pi\delta(\omega)

    # 矩形信号

    矩形信号

    x(t)={1,t<T10,t>T1x(t)= \begin{cases} 1,\ \ \ |t|\lt T_1\\ 0,\ \ \ |t|\gt T_1 \end{cases}

    有正变换:

    X(jω)=T1T1ejωtdt=2sinωT1ω=2T1sinωT1ωT1=2T1Sa(ωT1)=2T1sinc(ωT1π)ak=X(jω)T=2T1TSa(kω0T1)=2T1TSa(2T1T)\begin{aligned} X(j\omega) &=\int_{-T_1}^{T_1} e^{-j\omega t}dt\\ &=\frac{2\sin{\omega T_1}}{\omega}\\ &=\frac{2T_1\sin{\omega T_1}}{\omega T_1}\\ &=2T_1\text{Sa}(\omega T_1) = 2T_1\text{sinc}(\frac{\omega T_1}{\pi})\\\\ a_k = \frac{X(j\omega)}{T} &= \frac{2T_1}{T}\text{Sa}(k\omega_0 T_1) = \frac{2T_1}{T}\text{Sa}(\frac{2T_1}{T}) \end{aligned}

    不同脉冲宽度对频谱的影响:

    不同脉冲宽度对矩形信号频谱的影响

    可见,信号宽度在时域和频域之间有一种相反的关系。

    # 采样信号

    X(jω)={1,ω<W0,ω>WX(j\omega)= \begin{cases} 1,\ \ \ |\omega|\lt W\\ 0,\ \ \ |\omega|\gt W \end{cases}

    有反变换:

    x(t)=12πW+Wejωtdω=sinWTπt=WπSa(Wt)=WπSa(Wtπ)\begin{aligned} x(t) &= \frac{1}{2\pi}\int_{-W}^{+W} e^{-j\omega t}d\omega\\ &=\frac{\sin{WT}}{\pi t}\\ &= \frac{W}{\pi}\text{Sa}(Wt) = \frac{W}{\pi}\text{Sa}(\frac{Wt}{\pi}) \end{aligned}

    采样信号

    可见 x(t)x(t) 是采样信号


    矩形信号和采样信号的对偶关系:(上矩形信号、下采样信号)

    矩形信号和采样信号对偶关系

    # 周期信号

    TakTa_k 无论在 TT 有限还是无限的时候都可以是有限的,所以我们可以将 TT 有限时的周期信号和 TT 无限时的非周期信号用傅里叶变换统一起来

    但是 aka_k 却不是,所以我们无法将周期信号和非周期信号都用傅里叶级数表示。

    # FT 公式推导

    考虑 X(jω)=2πδ(ωω0)X(j\omega)=2\pi\delta(\omega-\omega_0) 所对应的信号:

    x(t)=12π+X(jω)ejωtdω=+δ(ωω0)ejωtdω=ejω0t\begin{aligned} x(t) &=\frac{1}{2\pi} \int_{-\infin}^{+\infin} X(j\omega)\ e^{j\omega t}d\omega\\ &=\int_{-\infin}^{+\infin} \delta(\omega-\omega_0)\ e^{j\omega t}d\omega\\ &= e^{j\omega_0 t} \end{aligned}

    于是对于表示为傅里叶级数的周期信号 x(t)x(t)

    x(t)=k=+akejkω0tx(t)=\sum_{k=-\infin}^{+\infin}a_k\ e^{jk\omega_0 t}

    就有频谱正变换

    X(jω)=2πk=+akδ(ωω0)X(j\omega)=2\pi \sum_{k=-\infin}^{+\infin} a_k\ \delta(\omega-\omega_0)

    这表明周期信号的傅里叶正变换由一系列位于谐波频率处的冲激组成,其中 kω0k\omega_0 处的冲激强度为对应傅里叶级数系数 aka_k2π2\pi 倍。

    # 常见信号的 FT

    # 正弦信号

    x(t)=sin(ω0t)=12j(ejω0tejω0t)x(t)=\sin(\omega_0 t)=\frac{1}{2j}(e^{j\omega_0 t}-e^{-j\omega_0 t})

    有正变换:

    X(jω)=πj[δ(ωω0)δ(ω+ω0)]X(j\omega)=\frac{\pi}{j}[\delta(\omega-\omega_0)-\delta(\omega+\omega_0)]

    正弦信号

    # 余弦信号

    x(t)=cos(ω0t)=12(ejω0t+ejω0t)x(t)=\cos(\omega_0 t)=\frac{1}{2}(e^{j\omega_0 t}+e^{-j\omega_0 t})

    有正变换:

    X(jω)=πj[δ(ωω0)+δ(ω+ω0)]X(j\omega)=\frac{\pi}{j}[\delta(\omega-\omega_0)+\delta(\omega+\omega_0)]

    余弦信号

    # 均匀冲激串

    x(t)=n=+δ(tnT)x(t)=\sum_{n=-\infin}^{+\infin}\delta(t-nT)

    其中 T=2πω0T=\frac{2\pi}{\omega_0}

    其傅里叶系数:

    ak=1TT2T2δ(t)ej2πTktdt=1TT2T2δ(t)dt=1T\begin{aligned} a_k &=\frac{1}{T}\int_{-\frac{T}{2}}^{\frac{T}{2}}\delta(t)\ e^{-j\frac{2\pi}{T}kt}dt\\ &=\frac{1}{T}\int_{-\frac{T}{2}}^{\frac{T}{2}}\delta(t)dt\\ &=\frac{1}{T} \end{aligned}

    可得傅里叶正变换:

    X(jω)=2πTk=+δ(ω2πTk)X(j\omega)=\frac{2\pi}{T}\sum_{k=-\infin}^{+\infin}\delta(\omega-\frac{2\pi}{T}k)

    均匀冲激串

    # 周期性矩形脉冲

    其傅里叶系数:

    ak=2T1TSa(πk2T1T)a_k=\frac{2T_1}{T}\text{Sa}(\pi k\frac{2T_1}{T})

    有傅里叶正变换:

    X(jω)=4πT1Tk=+Sa(πk2T1T)δ(ω2πTk)=2T1ω0k=+Sa(ω0T1k)δ(ωω0k)X(j\omega)=\frac{4\pi T_1}{T}\sum_{k=-\infin}^{+\infin}\text{Sa}(\pi k\frac{2T_1}{T})\ \delta(\omega-\frac{2\pi}{T}k) =2T_1\omega_0\sum_{k=-\infin}^{+\infin}\text{Sa}(\omega_0T_1\cdot k)\ \delta(\omega-\omega_0\cdot k)

    其中 ω0=2πT\omega_0=\frac{2\pi}{T} 为谱线间隔。

    # 特殊信号 FT

    # 符号函数

    符号函数

    可以将符号函数定义为:

    Sgn(t)=lima0[eatu(t)eatu(t)]\text{Sgn}(t)=\lim_{a\to0}[e^{-at}u(t)-e^{at}u(-t)]

    其傅里叶变换为:

    F{Sgn(t)}=lima0[1a+jω1ajω]=lima0j2ωa2+(jω)2=2jω\begin{aligned} \mathscr{F}\{\text{Sgn}(t)\} &=\lim_{a\to0}[\frac{1}{a+j\omega}-\frac{1}{a-j\omega}]\\ &=\lim_{a\to0}\frac{-j2\omega}{a^2+(j\omega)^2}\\ &=\frac{2}{j\omega} \end{aligned}

    # 单位阶跃函数

    u(t)u(t) 分解为偶部和奇部有:

    u(t)=ue(t)+uo(t)ue(t)=12uo(t)=12Sgn(t)\begin{aligned} &u(t)=u_e(t)+u_o(t)\\ &u_e(t)=\frac{1}{2}\\ &u_o(t)=\frac{1}{2}\text{Sgn}(t) \end{aligned}

    1CTFT2πδ(ω)1\stackrel{CTFT}{\longleftrightarrow}2\pi\delta(\omega)Sgn(t)CTFT2jω\text{Sgn}(t)\stackrel{CTFT}{\longleftrightarrow}\frac{2}{j\omega} ,因此有:

    u(t)F1jω+πδ(ω)u(t)\stackrel{F}{\longleftrightarrow}\frac{1}{j\omega}+\pi\delta(\omega)

    # 性质

    # 线性

    x(t)FX(jω),y(t)FY(jω)ax(t)+by(t)FaX(jω)+bY(jω)x(t)\stackrel{F}{\longleftrightarrow}X(j\omega)\ \ ,\ \ y(t)\stackrel{F}{\longleftrightarrow}Y(j\omega)\\ \dArr\\ ax(t)+by(t)\stackrel{F}{\longleftrightarrow}aX(j\omega)+bY(j\omega)

    # 时移特性

    信号的时移会导致其频谱产生一个线性相移

    x(t)FX(jω)x(tt0)FX(jω)ejωt0x(t)\stackrel{F}{\longleftrightarrow}X(j\omega)\\ \dArr\\ x(t-t_0)\stackrel{F}{\longleftrightarrow}X(j\omega)e^{-j\omega t_0}


    证明

    F{x(tt0)}=+x(tt0)ejωtdt=+x(t)ejω(t+t0)dt=ejωt0+x(t)ejωtdt=ejωt0X(jω)\begin{aligned} \mathscr{F}\{x(t-t_0)\} &=\int_{-\infin}^{+\infin}x(t-t_0)\ e^{-j\omega t}dt\\ &=\int_{-\infin}^{+\infin}x(t')\ e^{-j\omega(t'+t_0)}dt'\\ &=e^{-j\omega t_0}\int_{-\infin}^{+\infin}x(t')\ e^{-j\omega t'}dt'\\ &=e^{-j\omega t_0}X(j\omega) \end{aligned}

    # 共轭对称性

    x(t)FX(jω)x(t)FX(jω)x(t)\stackrel{F}{\longleftrightarrow}X(j\omega)\\ \dArr\\ x^*(t)\stackrel{F}{\longleftrightarrow}X^*(-j\omega)

    特别地,当 x(t)x(t) 是实信号 (实函数),则有频谱: X(jω)=X(jω)X(j\omega)=X^*(-j\omega)共轭偶函数实幅偶,虚相奇


    证明

    x(t)x(t) 的傅里叶正变换两侧同时取共轭有:

    X(jω)=[x(t)ejkω0t]dt=x(t)ejkω0tdt\begin{aligned} X^*(j\omega) &={\int}_{-\infin}^{\infin} [x(t) e^{-jk\omega_0 t}]^* dt\\ &={\int}_{-\infin}^{\infin} x(t)^* e^{jk\omega_0 t} dt\\ \end{aligned}

    所以:

    X(jω)=x(t)ejkω0tdt\begin{aligned} X^*(-j\omega) &={\int}_{-\infin}^{\infin} x(t)^* e^{-jk\omega_0 t} dt\\ \end{aligned}

    即得证:

    x(t)FX(jω)x^*(t)\stackrel{F}{\longleftrightarrow}X^*(-j\omega)


    x(t)x(t) 是实信号时有 x(t)=x(t)x(t)=x^*(t) ,于是有:

    X(jω)=X(jω)=Re[X(jω)]+jIm[X(jω)]X(j\omega)=X^*(-j\omega)=\text{Re}[X(-j\omega)]+j\ \text{Im}[X(-j\omega)]

    共轭偶函数:

    • 部是函数:Re[X(jω)]=Re[X(jω)]\text{Re}[X(j\omega)]=\text{Re}[X(-j\omega)]
    • 部是函数:Im[X(jω)]=Im[X(jω)]\text{Im}[X(j\omega)]=-\text{Im}[X(-j\omega)]
    • 幅度函数:X(jω)=X(jω)|X(j\omega)|=|X(-j\omega)|
    • 相位函数:ArgX(jω)=ArgX(jω)\text{Arg}X(j\omega)=-\text{Arg}X(-j\omega)

    # 奇偶对称性

    • 偶信号的傅里叶 (正) 变换还是偶函数
    • 实偶信号的傅里叶 (正) 变换还是实偶函数
    • 奇信号的傅里叶 (正) 变换还是奇函数
    • 实奇信号的傅里叶 (正) 变换是虚奇函数

    用表格总结奇偶性、是否为实函数的不同情况下的傅里叶正变换奇偶情况:

    x(t)x(t)偶函数:x(t)=x(t)x(t)=x(-t)奇函数:x(t)=x(t)x(t)=-x(-t)
    一般的X(jω)=X(jω)X(j\omega)=X(-j\omega)X(jω)=X(jω)X(j\omega)=-X(-j\omega)
    x(t)x(t) 是实函数:x(t)=x(t)x(t)=x^*(t)X(jω)=X(jω)=X(jω)=X(jω)X(j\omega)=X(-j\omega)=X^*(j\omega)=X^*(-j\omega)X(jω)=X(jω)=X(jω)X(j\omega)=-X(-j\omega)=X^*(-j\omega)

    # 奇偶分解

    实信号 x(t)x(t) 若有奇偶分解 x(t)=xe(t)+xo(t)x(t)=x_e(t)+x_o(t) ,则有:

    X(jω)=Xe(jω)+Xo(jω)xe(t)FXe(jω)xo(t)FXo(jω)X(j\omega)=X_e(j\omega)+X_o(j\omega)\\ x_e(t)\stackrel{F}{\longleftrightarrow}X_e(-j\omega)\\ x_o(t)\stackrel{F}{\longleftrightarrow}X_o(-j\omega)

    # 时域微分特性

    可将微分运算转变为代数运算

    x(t)FX(jω)ddtx(t)FjωX(jω)x(t)\stackrel{F}{\longleftrightarrow}X(j\omega)\\ \dArr\\ \frac{d}{dt}x(t)\stackrel{F}{\longleftrightarrow}j\omega X(j\omega)

    可以推广到高次微分,在后面的微分方程分析 LTI 系统中会用到:

    dkdtkx(t)F(jω)kX(jω)\frac{d^k}{dt^k}x(t)\stackrel{F}{\longleftrightarrow}(j\omega)^k X(j\omega)


    证明:傅里叶反变换两侧同时对 tt 微分即可

    x(t)=12π+X(jω)ejωtdωddtx(t)=12π+X(jω)ddtejωtdω=12π+jωX(jω)ejωtdω\begin{aligned} x(t) &=\frac{1}{2\pi} \int_{-\infin}^{+\infin} X(j\omega) e^{j\omega t}d\omega\\ \frac{d}{dt}x(t) &=\frac{1}{2\pi}\int_{-\infin}^{+\infin} X(j\omega)\frac{d}{dt}e^{j\omega t}d\omega\\ &=\frac{1}{2\pi}\int_{-\infin}^{+\infin} j\omega X(j\omega)\ e^{j\omega t}d\omega\\ \end{aligned}

    # 时域积分特性

    可将积分运算转变为代数运算

    x(t)FX(jω)tx(τ)dτF12πX(jω)+πX(0)δ(ω)x(t)\stackrel{F}{\longleftrightarrow}X(j\omega)\\ \dArr\\ \int_{-\infin}^{t}x(\tau)d\tau\stackrel{F}{\longleftrightarrow}\frac{1}{2\pi}X(j\omega)+\pi X(0)\delta(\omega)


    证明:傅里叶正变换两侧同时对 τ\tau 积分即可

    F{tx(τ)dτ}=+[tx(τ)dτ]ejωtdt=+[+x(τ)u(tτ)dτ]ejωtdt=+x(τ)[+ejωtu(tτ)dt]dτ=+x(τ)[+ejω(tτ)jωτu(tτ)d(tτ)]dτ=+x(τ)ejωτ[+ejωsu(s)ds]dτ=[1jω+πδ(ω)]+x(τ)ejωτdτ=[1jω+πδ(ω)]X(jω)=1jωX(jω)+πX(0)δ(ω)\begin{aligned} \mathscr{F}\{\int_{-\infin}^{t}x(\tau)d\tau\} &=\int_{-\infin}^{+\infin}[\int_{-\infin}^{t}x(\tau)d\tau]e^{-j\omega t}dt\\ &=\int_{-\infin}^{+\infin}[\int_{-\infin}^{+\infin}x(\tau)u(t-\tau)d\tau]e^{-j\omega t}dt\\ &=\int_{-\infin}^{+\infin}x(\tau)[\int_{-\infin}^{+\infin}e^{-j\omega t}u(t-\tau)dt]d\tau\\ &=\int_{-\infin}^{+\infin}x(\tau)[\int_{-\infin}^{+\infin}e^{-j\omega(t-\tau)-j\omega\tau}u(t-\tau)d(t-\tau)]d\tau\\ &=\int_{-\infin}^{+\infin}x(\tau)e^{-j\omega\tau}[\int_{-\infin}^{+\infin}e^{-j\omega s}u(s)ds]d\tau\\ &=[\frac{1}{j\omega}+\pi\delta(\omega)]\int_{-\infin}^{+\infin}x(\tau)e^{-j\omega \tau}d\tau\\ &=[\frac{1}{j\omega}+\pi\delta(\omega)]X(j\omega)\\ &=\frac{1}{j\omega}X(j\omega)+\pi X(0)\delta(\omega) \end{aligned}

    其中 u(s)CTFT1jω+πδ(ω)u(s)\stackrel{CTFT}{\longleftrightarrow}\frac{1}{j\omega}+\pi\delta(\omega) 是单位阶跃函数的傅里叶变换

    # 时域和频域的尺度变换

    信号如果在时域扩展 a 倍,则其带宽相应压缩 a 倍,反之亦然

    x(t)FX(jω)x(at)F1aX(jωa)x(t)\stackrel{F}{\longleftrightarrow}X(j\omega)\\ \dArr\\ x(at)\stackrel{F}{\longleftrightarrow}\frac{1}{|a|}X(\frac{j\omega}{a})


    证明:

    F{x(at)}=+x(at)ejωtdt=1a+x(τ)ejωaτdτ=1aX(jωa)\begin{aligned} \mathscr{F}\{x(at)\} &=\int_{-\infin}^{+\infin}x(at)e^{-j\omega t}dt\\ &=\frac{1}{|a|}\int_{-\infin}^{+\infin}x(\tau)e^{-j\frac{\omega}{a}\tau}d\tau\\ &=\frac{1}{|a|}X(\frac{j\omega}{a}) \end{aligned}

    # 对偶性

    由对偶性可以方便地将时域的某些特性对偶到频域

    x(t)CTFTX(jω)X(jt)CTFT2πx(ω)x(t)\stackrel{CTFT}{\longleftrightarrow}X(j\omega)\\ \dArr\\ X(jt)\stackrel{CTFT}{\longleftrightarrow}2\pi x(-\omega)


    证明:

    x(t)=12π+X(jω)ejωtdω2πx(t)=+X(jω)ejωtdω2πx(ω)=+X(jt)ejωtdt2πx(ω)=+X(jt)ejωtdt\begin{aligned} x(t) &=\frac{1}{2\pi} \int_{-\infin}^{+\infin} X(j\omega) e^{j\omega t}d\omega\\ 2\pi x(t) &=\int_{-\infin}^{+\infin} X(j\omega) e^{j\omega t}d\omega\\ 2\pi x(\omega) &=\int_{-\infin}^{+\infin} X(jt) e^{j\omega t}dt\\ 2\pi x(-\omega) &=\int_{-\infin}^{+\infin} X(jt) e^{-j\omega t}dt\\ \end{aligned}

    # 移频特性

    将时域的时移性对偶可以得到频域的移频特性

    x(t)FX(jω)x(t)ejω0tFX[j(ωω0)]x(t)\stackrel{F}{\longleftrightarrow}X(j\omega)\\ \dArr\\ x(t)e^{j\omega_0 t}\stackrel{F}{\longleftrightarrow}X[j(\omega-\omega_0)]


    证明:

    x(t)FX(jω)对偶性X(jt)F2πx(ω)时移特性X(j(tt0))F2πx(ω)ejω0t对偶性2πx(t)ejω0tF2πX[j(ωω0)]x(t)ejω0tFX[j(ωω0)]x(t)\stackrel{F}{\longleftrightarrow}X(j\omega)\\ \dArr\text{对偶性}\\ X(jt)\stackrel{F}{\longleftrightarrow}2\pi x(-\omega)\\ \dArr\text{时移特性}\\ X(j(t-t_0))\stackrel{F}{\longleftrightarrow}2\pi x(-\omega)e^{-j\omega_0 t}\\ \dArr\text{对偶性}\\ 2\pi x(-t)e^{-j\omega_0 t}\stackrel{F}{\longleftrightarrow}2\pi X[j(-\omega-\omega_0)]\\ \dArr\\ x(t)e^{j\omega_0 t}\stackrel{F}{\longleftrightarrow}X[j(\omega-\omega_0)]

    # 频域微分特性

    x(t)FX(jω)jtx(t)FddωX(jω)x(t)\stackrel{F}{\longleftrightarrow}X(j\omega)\\ \dArr\\ -jtx(t)\stackrel{F}{\longleftrightarrow}\frac{d}{d\omega}X(j\omega)

    # 频域积分特性

    x(t)FX(jω)x(t)jt+πx(0)δ(t)FωX(jτ)dτx(t)\stackrel{F}{\longleftrightarrow}X(j\omega)\\ \dArr\\ \frac{x(t)}{-jt}+\pi x(0)\delta(t)\stackrel{F}{\longleftrightarrow}\int_{-\infin}^{\omega}X(j\tau)d\tau

    # Parseval 定理

    x(t)FX(jω)+x(t)2dt=12π+X(jω)2dωx(t)\stackrel{F}{\longleftrightarrow}X(j\omega)\\ \dArr\\ \int_{-\infin}^{+\infin}|x(t)|^2dt=\frac{1}{2\pi}\int_{-\infin}^{+\infin}|X(j\omega)|^2d\omega

    这表明:信号的能量既可以在时域求得,也可以在频域求得。由于 X(jω)2|X(j\omega)^2| 表示了信号能量在频域的分布,因而称其为 =="能量谱密度" 函数 ==。

    # ⭐️卷积特性

    x(t)FX(jω),h(t)FH(jω)x(t)h(t)FX(jω)H(jω)x(t)\stackrel{F}{\longleftrightarrow}X(j\omega)\ \ \ ,\ \ \ h(t)\stackrel{F}{\longleftrightarrow}H(j\omega)\\ \dArr\\ x(t)*h(t)\stackrel{F}{\longleftrightarrow}X(j\omega)H(j\omega)

    本质上,卷积特性的成立正是因为复指数信号 ejωte^{jωt} 是一切 LTI 系统的特征函数,H(jω)H(j\omega) 则是对应的特征值


    证明

    y(t)=x(t)h(t)=+x(τ)h(tτ)dτy(t)=x(t)*h(t)=\int_{-\infin}^{+\infin}x(\tau)\ h(t-\tau)\ d\tau ,求其傅里叶正变换为:

    F{y(t)}=Y(jω)=+ejωt+x(τ)h(tτ)dτdt=+ejω(tτ)jωτ+x(τ)h(tτ)dτd(tτ)=+ejωsjωτ+x(τ)h(s)dτds=+ejωsejωτ+x(τ)h(s)dτds=+x(τ)ejωτdτ+h(s)ejωsds=X(jω)H(jω)\begin{aligned} \mathscr{F}\{y(t)\}=Y(j\omega) &=\int_{-\infin}^{+\infin} e^{-j\omega t}\int_{-\infin}^{+\infin}x(\tau)\ h(t-\tau)\ d\tau\ dt\\ &=\int_{-\infin}^{+\infin} e^{-j\omega(t-\tau)-j\omega \tau}\int_{-\infin}^{+\infin}x(\tau)\ h(t-\tau)\ d\tau\ d(t-\tau)\\ &=\int_{-\infin}^{+\infin} e^{-j\omega s-j\omega \tau}\int_{-\infin}^{+\infin}x(\tau)\ h(s)\ d\tau\ ds\\ &=\int_{-\infin}^{+\infin}e^{-j\omega s}e^{-j\omega \tau}\int_{-\infin}^{+\infin}x(\tau)\ h(s)\ d\tau\ ds\\ &=\int_{-\infin}^{+\infin}x(\tau)\ e^{-j\omega \tau}\ d\tau\int_{-\infin}^{+\infin}h(s)\ e^{-j\omega s}\ ds\\ &=X(j\omega)H(j\omega) \end{aligned}

    另外,可将 x(t)x(t) 视为复指数分量 ejωte^{j\omega t} 的线性组合,每个 ejωte^{j\omega t} 通过 LTI 系统时都要受到系统对应的特征值 H(jω)H(j\omega) 的加权


    LTI 系统的单位冲激响应 h(t)h(t) 与频率响应 H(jω)H(j\omega) 一一对应,因而 LTI 系统可以由其频率响应完全表征。

    ⚠️ 并非任何系统的频率响应都存在。

    用频率响应表征系统时,一般都限于对稳定系统。

    稳定性保证了 h(t)h(t) 的绝对可积,结合 Dirichlet 的另外两个条件(实际系统一般满足),则频率响应必然存在

    # ⭐️相乘特性

    s(t)FS(jω),p(t)FP(jω)s(t)p(t)F12πS(jω)P(jω)s(t)\stackrel{F}{\longleftrightarrow}S(j\omega)\ \ \ ,\ \ \ p(t)\stackrel{F}{\longleftrightarrow}P(j\omega)\\ \dArr\\ s(t)p(t)\stackrel{F}{\longleftrightarrow}\frac{1}{2\pi}S(j\omega)*P(j\omega)


    证明:利用对偶性可以从卷积性质得出相乘性质

    由对偶性有:

    s(t)FS(jω),p(t)FP(jω)S(jt)F2πS(ω),P(jt)F2πP(ω)s(t)\stackrel{F}{\longleftrightarrow}S(j\omega)\ \ \ ,\ \ \ p(t)\stackrel{F}{\longleftrightarrow}P(j\omega)\\ \dArr\\ S(jt)\stackrel{F}{\longleftrightarrow}2\pi S(-\omega)\ \ \ ,\ \ \ P(jt)\stackrel{F}{\longleftrightarrow}2\pi P(-\omega)\\

    由卷积特性有:

    S(jt)P(jt)F4π2s(ω)p(ω)S(jt)*P(jt)\stackrel{F}{\longleftrightarrow}4\pi^2s(-\omega)p(-\omega)

    由对偶性有:

    4π2s(t)p(t)F2πS(jω)P(jω)s(t)p(t)F12πS(jω)P(jω)\begin{aligned} 4\pi^2s(-t)p(-t)&\stackrel{F}{\longleftrightarrow}2\pi S(-j\omega)*P(-j\omega)\\ s(-t)p(-t)&\stackrel{F}{\longleftrightarrow}\frac{1}{2\pi}S(-j\omega)*P(-j\omega) \end{aligned}

    # 连续时间 LTI 系统的分析

    利用傅里叶变换及其特性对 LTI 系统分析

    # 频域分析法

    根据卷积特性,可以对 LTI 系统进行频域分析,其过程为:

    1. 求出(或已知、根据系统描述直接得出)系统单位冲激响应 h(t)h(t)
    2. 求输入 x(t)x(t) 的傅里叶正变换:x(t)CTFTX(jω)x(t)\stackrel{CTFT}{\longleftrightarrow}X(j\omega)
    3. 求单位冲激响应 h(t)h(t) 的傅里叶正变换:h(t)CTFTH(jω)h(t)\stackrel{CTFT}{\longleftrightarrow}H(j\omega)
    4. 若需要求 Y(jω)Y(j\omega) ,由傅里叶变换卷积特性得:Y(jω)=X(jω)H(jω)Y(j\omega)=X(j\omega)H(j\omega)
    5. 若需要求 y(t)y(t)
      • x(t)x(t) 是非周期的(或非周期部分):
        • 往往由傅里叶反变换求得:y(t)=F1{Y(jω)}=12π+Y(jω)ejωtdωy(t)=\mathscr{F}^{-1}\{Y(j\omega)\}=\frac{1}{2\pi} \int_{-\infin}^{+\infin} Y(j\omega) e^{j\omega t}d\omega
        • 对有理函数求傅里叶反变换通常采用部分分式展开和利用常用变换对进行
      • x(t)x(t) 是周期的(或周期部分):
        • Y(jω)=X(jω)×H(jω)|Y(j\omega)|=|X(j\omega)|\times|H(j\omega)|
        • argY(jω)=argX(jω)+argH(jω)\text{arg}Y(j\omega)=\text{arg}X(j\omega)+\text{arg}H(j\omega)

    # 线性常微分方程

    * 线性常系数微分方程

    对于由线性常系数微分方程描述的 LTI 系统:

    k=0Nakdky(t)dtk=k=0Mbkdkx(t)dtk\sum_{k=0}^{N}a_k\frac{d^ky(t)}{dt^k}=\sum_{k=0}^{M}b_k\frac{d^kx(t)}{dt^k}

    对方程两边进行傅里叶变换,有:

    k=0Nak(jω)kY(jω)=k=0Mbk(jω)kX(jω)Y(jω)k=0Nak(jω)k=X(jω)k=0Mbk(jω)kY(jω)X(jω)=k=0Mbk(jω)kk=0Nak(jω)k\begin{aligned} \sum_{k=0}^{N}a_{k}(j\omega)^{k}Y(j\omega) &= \sum_{k=0}^{M}b_{k}(j\omega)^{k}X(j\omega)\\ Y(j\omega)\sum_{k=0}^{N}a_{k}(j\omega)^{k} &= X(j\omega)\sum_{k=0}^{M}b_{k}(j\omega)^{k}\\ {Y(j\omega)\over X(j\omega)} &= {\sum_{k=0}^{M}b_{k}(j\omega)^{k}\over\sum_{k=0}^{N}a_{k}(j\omega)^{k}} \end{aligned}

    由 CTFT 卷积特性有:

    Y(jω)=X(jω)H(jω)Y(j\omega)=X(j\omega)H(j\omega)

    可得:

    H(jω)=Y(jω)X(jω)=k=0Mbk(jω)kk=0Nak(jω)kH(j\omega)={Y(j\omega)\over X(j\omega)}={\sum_{k=0}^{M}b_{k}(j\omega)^{k}\over\sum_{k=0}^{N}a_{k}(j\omega)^{k}}

    可见由线性常系数微分方程描述的 LTI 系统的频率响应是一个有理函数。

    若要求 h(t)h(t) ,也就是系统的单位冲激响应,则往往可以对有理函数(有理分式)形式的 H(jω)H(j\omega) 变形化得到常用变换对的表示形式,再利用常用变换对反变换得到 h(t)h(t)


    反之,若已知一个 LTI 系统的频率响应 H(jω)=Y(jω)X(jω)H(j\omega)={Y(j\omega)\over X(j\omega)} ,则可以列出两侧分别是关于 Y(jω)Y(j\omega)X(jω)X(j\omega) 的方程,两侧同时进行傅里叶反变换,(注意将 jωj\omega 视为一个整体)则可得到描述该 LTI 系统的线性常系数微分方程

    \ No newline at end of file diff --git a/ie/signals-and-systems/DTFT/index.html b/ie/signals-and-systems/DTFT/index.html index 2a465c76..eaf44b84 100644 --- a/ie/signals-and-systems/DTFT/index.html +++ b/ie/signals-and-systems/DTFT/index.html @@ -1,4 +1,4 @@ -离散时间信号的傅里叶变换 - 信号与系统 - 电子信息 | Balloon Party = 風船のパーティー = fuusen no party

    # 公式表

    # 基本傅里叶变换对

    基本FT对1基本FT对2

    # 性质

    性质表1性质表2

    # 非周期信号

    # FT 公式推导

    定义 x~[n]\widetilde{x}[n]

    # 公式表

    # 基本傅里叶变换对

    基本FT对1基本FT对2

    # 性质

    性质表1性质表2

    # 非周期信号

    # FT 公式推导

    定义 x~[n]\widetilde{x}[n]1argX(e(jω)=arctan1acosωasinω

    信号1

    # 衰减脉冲 (双侧)

    * 名字是我乱起的

    x[n]=ana<1x[n]=a^{|n|}\ \ \ \ \ |a|\lt1

    x[n]=anu[n1]+anu[n]x[n]=a^{-n}u[-n-1]+a^nu[n]

    有傅里叶正变换:

    X(ejω)=n=1anejωn+n=0+anejωn=n=1+anejωn+n=0+anejωn=n=1+(aejω)n+n=0+(aejω)n=aejω1aejω+11aejω=1a21+a22acosω\begin{aligned} X(e^{j\omega}) &=\sum_{n=-\infin}^{-1}a^{-n}e^{-j\omega n}+\sum_{n=0}^{+\infin}a^ne^{-j\omega n}\\ &=\sum_{n=1}^{+\infin}a^{n}e^{j\omega n}+\sum_{n=0}^{+\infin}a^ne^{-j\omega n}\\ &=\sum_{n=1}^{+\infin}(ae^{j\omega})^n+\sum_{n=0}^{+\infin}(ae^{-j\omega})^n\\ &=\frac{ae^{j\omega}}{1-ae^{j\omega}}+\frac{1}{1-ae^{-j\omega}}\\ &=\frac{1-a^2}{1+a^2-2a\cos\omega} \end{aligned}

    # 冲激信号

    x[n]=δ[n]x[n]=\delta[n]

    有傅里叶正变换:

    X(ejω)=n=+δ[n]ejωn=1X(e^{j\omega})=\sum_{n=-\infin}^{+\infin}\delta[n]e^{-j\omega n}=1

    δ(t)\delta(t) 类似,δ[n]\delta[n] 中也包括所有的频率成分,且所有频率分量的幅度、相位都相同

    # 矩形脉冲

    x[n]={1,nN10,n>N1x[n]= \begin{cases} 1,\ \ \ |n|\leq N_1\\ 0,\ \ \ |n|\gt N_1 \end{cases}

    有傅里叶正变换:

    X(ejω)=n=N1+N1ejωn=sin[(2N1+1)ω2]sinω2X(e^{j\omega})=\sum_{n=-N_1}^{+N_1}e^{-j\omega n}=\frac{\sin[(2N_1+1)\frac{\omega}{2}]}{\sin\frac{\omega}{2}}

    显然,将 X(ejω)X(e^{j\omega}) 中的 ω\omega 代之以 kω0k\omega_0 再乘以 1N\frac{1}{N} ,即是相应周期延拓序列的频谱(傅里叶级数的系数):ak=1NX(ejω)ω=kωa_k=\frac{1}{N}X(e^{j\omega})|_{\omega=k\omega}

    # 采样信号

    X(ejω)={1,ω<W0,W<ωπX(e^{j\omega})= \begin{cases} 1,\ \ \ |\omega|\lt W\\ 0,\ \ \ W\lt|\omega|\leq\pi \end{cases}

    有傅里叶正变换:

    x[n]=12πWWejωndω=sinWnπn=WπSa(Wn)=WπSinc(Wnπ)\begin{aligned} x[n] &=\frac{1}{2\pi}\int_{-W}^{W}e^{j\omega n}d\omega\\ &=\frac{\sin Wn}{\pi n}\\ &=\frac{W}{\pi}\text{Sa}(Wn)\\ &=\frac{W}{\pi}\text{Sinc}(\frac{Wn}{\pi}) \end{aligned}

    # 常数信号

    x[n]=1x[n]=1

    因为:

    ππδ(ω)ejωndω=112πππ2πδ(ω)ejωndω=1\begin{aligned} \int_{-\pi}^{\pi}\delta(\omega)\cdot e^{j\omega n}d\omega&=1\\ \frac{1}{2\pi}\int_{-\pi}^{\pi}2\pi\cdot\delta(\omega)\cdot e^{j\omega n}d\omega&=1 \end{aligned}

    2πδ(ω)2\pi\delta(\omega)2π2\pi 为周期进行周期延拓,得到 x[n]=1x[n]=1 的傅里叶正变换为:

    X(ejω)=2πl=+δ(ω2πl)X(e^{j\omega})=2\pi\sum_{l=-\infin}^{+\infin}\delta(\omega-2\pi l)

    # 周期信号

    # FT 公式推导

    对连续时间信号,有 ejω0tCTFT2πδ(ωω0)e^{j\omega_0 t}\stackrel{CTFT}{\longleftrightarrow}2\pi\delta(\omega-\omega_0) ,由此推断,对离散时间信号或许有相似的情况

    但对于 x[n]=ejkω0nx[n]=e^{jk\omega_0 n} ,由于离散时间傅里叶变换是以 2π2\pi 为周期的,因此频域的冲激应该是周期性的冲激串(周期延拓),也就是:

    X(ejω)=l=+2πδ(ωkω02πl)X(e^{j\omega})=\sum_{l=-\infin}^{+\infin}2\pi\delta(\omega-k\omega_0-2\pi l)

    将反变换积分范围包含 ω0\omega_0 ,可以得到:

    x[n]=12π2πX(ejω)ejωndω=ω0πω0+πδ(ωω0)ejωndω=ejω0n\begin{aligned} x[n] &=\frac{1}{2\pi}\int_{2\pi}X(e^{j\omega})e^{j\omega n}d\omega\\ &=\int_{\omega_0-\pi}^{\omega_0+\pi}\delta(\omega-\omega_0)e^{j\omega n}d\omega\\ &=e^{j\omega_0 n} \end{aligned}

    于是对于表示为傅里叶级数的离散时间周期信号:

    x[n]=k=Nakejkω0n,ω0=2πNx[n]=\sum_{k=\langle N\rangle}a_{k}e^{jk\omega_0 n},\ \ \ \omega_0=\frac{2\pi}{N}

    就有(傅里叶正变换):

    X(ejω)=F{x[n]}=k=NakF{ejkω0n}=k=Nakl=+2πδ(ωkω02πl)=2πl=+k=Nakδ(ωkω0lNω0)=2πl=+k=0N1akδ(ω(lN+k)ω0)=2πl=+lN+k=lNlN+N1alN+kδ(ω(lN+k)ω0)=2πl=+lN+k=lNlN+N1alN+kδ(ω(lN+k)ω0)=2πl=+akδ(ωkω0)\begin{aligned} X(e^{j\omega})=\mathscr{F}\{x[n]\} &=\sum_{k=\langle N\rangle}a_{k}\cdot\mathscr{F}\{e^{jk\omega_{0}n}\}\\ &=\sum_{k=\langle N\rangle}a_{k}\sum_{l=-\infin}^{+\infin}2\pi\delta(\omega-k\omega_0-2\pi l)\\ &=2\pi\sum_{l=-\infin}^{+\infin}\sum_{k=\langle N\rangle}a_{k}\delta(\omega-k\omega_0-lN\omega_0)\\ &=2\pi\sum_{l=-\infin}^{+\infin}\sum_{k=0}^{N-1}a_{k}\delta(\omega-(lN+k)\omega_0)\\ &=2\pi\sum_{l=-\infin}^{+\infin}\ \sum_{lN+k=lN}^{lN+N-1}a_{lN+k}\delta(\omega-(lN+k)\omega_0)\\ &=2\pi\sum_{l=-\infin}^{+\infin}\ \sum_{lN+k=lN}^{lN+N-1}a_{lN+k}\delta(\omega-(lN+k)\omega_0)\\ &=2\pi\sum_{l=-\infin}^{+\infin}a_{k}\delta(\omega-k\omega_0) \end{aligned}

    可以发现 DTFT 与 CTFT 形式一致

    # 性质

    x[n]DTFTX(ejω)x[n]\stackrel{DTFT}{\longleftrightarrow}X(e^{j\omega}) ,则有以下性质成立:

    # 周期性

    X(ej(ω+2π))=X(ejω)X(e^{j(\omega+2\pi)})=X(e^{j\omega})

    # 线性

    x[n]DTFTX(ejω),y[n]DTFTY(ejω)ax[n]+by[n]DTFTaX(ejω)+bY(ejω)x[n]\stackrel{DTFT}{\longleftrightarrow}X(e^{j\omega})\ \ ,\ \ y[n]\stackrel{DTFT}{\longleftrightarrow}Y(e^{j\omega})\\ \dArr\\ ax[n]+by[n]\stackrel{DTFT}{\longleftrightarrow}aX(e^{j\omega})+bY(e^{j\omega})

    # 时移特性

    x[nn0]DTFTX(ejω)ejωn0x[n-n_0]\stackrel{DTFT}{\longleftrightarrow}X(e^{j\omega})e^{-j\omega n_0}

    # 频移特性

    x[n]ejω0nDTFTX(ej(ωω0))x[n]e^{j\omega_0 n}\stackrel{DTFT}{\longleftrightarrow}X(e^{j(\omega-\omega_0)})

    # 时域反转

    x[n]DTFTX(ejω)x[-n]\stackrel{DTFT}{\longleftrightarrow}X(e^{-j\omega})

    # 共轭对称性

    x[n]DTFTX(ejω)x^*[n]\stackrel{DTFT}{\longleftrightarrow}X^*(e^{-j\omega})

    特别地,当 x[n]x[n] 是实信号 (实函数),则有频谱: X(ejω)=X(ejω)X(e^{-j\omega})=X^*(e^{-j\omega})共轭偶函数实幅偶,虚相奇

    共轭偶函数:

    • 部是函数:Re[X(jω)]=Re[X(jω)]\text{Re}[X(j\omega)]=\text{Re}[X(-j\omega)]
    • 部是函数:Im[X(jω)]=Im[X(jω)]\text{Im}[X(j\omega)]=-\text{Im}[X(-j\omega)]
    • 幅度函数:X(jω)=X(jω)|X(j\omega)|=|X(-j\omega)|
    • 相位函数:ArgX(jω)=ArgX(jω)\text{Arg}X(j\omega)=-\text{Arg}X(-j\omega)

    # 奇偶对称性

    与 CTFT 的奇偶对称性一致

    • 偶信号的傅里叶 (正) 变换还是偶函数
    • 实偶信号的傅里叶 (正) 变换还是实偶函数
    • 奇信号的傅里叶 (正) 变换还是奇函数
    • 实奇信号的傅里叶 (正) 变换是虚奇函数

    用表格总结奇偶性、是否为实函数的不同情况下的傅里叶正变换奇偶情况:

    x[n]x[n]偶函数:x[n]=x[n]x[n]=x[-n]奇函数:x[n]=x[n]x[n]=-x[-n]
    一般的X(jω)=X(jω)X(j\omega)=X(-j\omega)X(jω)=X(jω)X(j\omega)=-X(-j\omega)
    实函数:x[n]=x[n]x[n]=x^*[n]X(jω)=X(jω)=X(jω)=X(jω)X(j\omega)=X(-j\omega)=X^*(j\omega)=X^*(-j\omega)X(jω)=X(jω)=X(jω)X(j\omega)=-X(-j\omega)=X^*(-j\omega)

    # 时域差分特性

    x[n]x[n1]DTFT(1ejω)X(ejω)x[n]-x[n-1]\stackrel{DTFT}{\longleftrightarrow}(1-e^{-j\omega})X(e^{j\omega})

    # 时域累加特性

    k=nx[k]DTFTX(ejω)1ejω+πX(ej0)k=+δ(ω2πk)\sum_{k=-\infin}^{n}x[k]\stackrel{DTFT}{\longleftrightarrow}\frac{X(e^{j\omega})}{1-e^{-j\omega}}+\pi X(e^{j0})\sum_{k=-\infin}^{+\infin}\delta(\omega-2\pi k)


    时域差分与累加特性说明:离散时间傅里叶变换中的 1ejω1-e^{-j\omega} 对应于连续时间傅里叶变换中的 jωj\omega

    # 时域内插

    信号的时域特性与频域特性之间有一种相反的关系

    定义:

    xk[n]={x[nk],n为k的整数倍0,其他nx_k[n]= \begin{cases} x[\frac{n}{k}], &\text{n为k的整数倍}\\ 0, & \text{其他n} \end{cases}

    则有:

    xk[n]DTFTX(ejkω)x_k[n]\stackrel{DTFT}{\longleftrightarrow}X(e^{jk\omega})


    证明:

    Xk(ejω)=n=+xk[n]ejωn=r=+xk[rk]ejωrk=r=+x[rkk]ejωrk=r=+x[r]ejkωr=X(ejkω)\begin{aligned} X_k(e^{j\omega}) &=\sum_{n=-\infin}^{+\infin}x_k[n]e^{-j\omega n}\\ &=\sum_{r=-\infin}^{+\infin}x_k[rk]e^{-j\omega rk}\\ &=\sum_{r=-\infin}^{+\infin}x[\frac{rk}{k}]e^{-j\omega rk}\\ &=\sum_{r=-\infin}^{+\infin}x[r]e^{-jk\omega r}\\ &=X(e^{jk\omega}) \end{aligned}

    # 频域微分

    nx[n]DTFTjdX(ejω)dωnx[n]\stackrel{DTFT}{\longleftrightarrow}j\frac{dX(e^{j\omega})}{d\omega}

    # Parseval 定理

    n=+x[n]2=12π2πX(ejω)2dω\sum_{n=-\infin}^{+\infin}|x[n]|^2=\frac{1}{2\pi}\int_{2\pi}|X(e^{j\omega})|^2d\omega

    # 卷积特性

    x[n]DTFTX(ejω),h[n]DTFTH(ejω)x[n]h[n]DTFTX(ejω)H(ejω)x[n]\stackrel{DTFT}{\longleftrightarrow}X(e^{j\omega})\ \ ,\ \ h[n]\stackrel{DTFT}{\longleftrightarrow}H(e^{j\omega})\\ \dArr\\ x[n]*h[n]\stackrel{DTFT}{\longleftrightarrow}X(e^{j\omega})H(e^{j\omega})

    该卷积性质使离散时间 LTI 系统的频域分析非常便利。本质上,卷积性质的成立仍是因为复指数信号 e^{j𝜔n} 是一切离散时间 LTI 系统的特征函数,H(ejω)H(e^{j\omega}) 则是对应的特征值。

    # 相乘特性

    如果:

    y[n]=x1[n]x2[n]y[n]=x_1[n]\cdot x_2[n]

    则:

    Y(ejω)=12π2πX1(ejθ)X2(ej(ωθ))dθ=12πX1(ejω)X2(ejω)\begin{aligned} Y(e^{j\omega}) &=\frac{1}{2\pi}\int_{2\pi}X_1(e^{j\theta})X_2(e^{j(\omega-\theta)})d\theta\\ &=\frac{1}{2\pi}X_1(e^{j\omega})\otimes X_2(e^{j\omega}) \end{aligned}

    因此上述卷积称为周期卷积,因为 X1(ejω)X_1(e^{j\omega})X2(ejω)X_2(e^{j\omega}) 都是以 2π2\pi 为周期的

    # 离散时间 LTI 系统的分析

    # 频域分析法

    ⚠️并非所有的 LTI 系统都存在频率响应!一般只考虑稳定的 LTI 系统的频率响应,因为此时 h[n]h[n] 绝对可和,离散时间傅里叶变换(即频率响应)H(ejω)H(e^{j\omega}) 存在。

    与连续时间 LTI 系统的频域分析法基本一致

    根据卷积特性,可以对 LTI 系统进行频域分析,其过程为:

    1. 求出(或已知、根据系统描述直接得出)系统单位冲激响应 h[n]h[n]
    2. 求输入 x[n]x[n] 的傅里叶正变换:x[n]DTFTX(ejω)x[n]\stackrel{DTFT}{\longleftrightarrow}X(e^{j\omega})
    3. 求单位冲激响应 h[n]h[n] 的傅里叶正变换:h[n]DTFTH(ejω)h[n]\stackrel{DTFT}{\longleftrightarrow}H(e^{j\omega})
    4. 若需要求 Y(ejω)Y(e^{j\omega}) ,由傅里叶变换卷积特性得:Y(ejω)=X(ejω)H(ejω)Y(e^{j\omega})=X(e^{j\omega})H(e^{j\omega})
    5. 若需要求 y[n]y[n]
      • x[n]x[n] 是非周期的(或非周期部分):
        • 往往由傅里叶反变换求得:y[n]=F1{Y(ejω)}=12π+Y(ejω)ejωtdωy[n]=\mathscr{F}^{-1}\{Y(e^{j\omega})\}=\frac{1}{2\pi}\int_{-\infin}^{+\infin} Y(e^{j\omega}) e^{j\omega t}d\omega
        • 对有理函数求傅里叶反变换通常采用部分分式展开和利用常用变换对进行
      • x[n]x[n] 是周期的(或周期部分):
        • Y(ejω)=X(ejω)×H(ejω)|Y(e^{j\omega})|=|X(e^{j\omega})|\times|H(e^{j\omega})|
        • argY(ejω)=argX(ejω)+argH(ejω)\text{arg}Y(e^{j\omega})=\text{arg}X(e^{j\omega})+\text{arg}H(e^{j\omega})

    # 线性常差分方程

    * 线性常系数差分方程

    对于由线性常差分方程描述的 LTI 系统:

    k=0Naky[nk]=k=0Mbkx[nk]\sum_{k=0}^{N}a_ky[n-k]=\sum_{k=0}^{M}b_kx[n-k]

    对方程两边进行傅里叶变换,有:

    k=0NakejkωY(ejω)=k=0MbkejkωX(ejω)Y(ejω)k=0Nakejkω=X(ejω)k=0MbkejkωY(ejω)X(ejω)=k=0Mbkejkωk=0Nakejkω\begin{aligned} \sum_{k=0}^{N}a_{k}e^{-jk\omega}Y(e^{j\omega}) &= \sum_{k=0}^{M}b_{k}e^{-jk\omega}X(e^{j\omega})\\ Y(e^{j\omega})\sum_{k=0}^{N}a_{k}e^{-jk\omega} &= X(e^{j\omega})\sum_{k=0}^{M}b_{k}e^{-jk\omega}\\ {Y(e^{j\omega})\over X(e^{j\omega})} &= {\sum_{k=0}^{M}b_{k}e^{-jk\omega}\over\sum_{k=0}^{N}a_{k}e^{-jk\omega}} \end{aligned}

    由 DTFT 卷积特性有:

    Y(ejω)=X(ejω)H(ejω)Y(e^{j\omega})=X(e^{j\omega})H(e^{j\omega})

    可得:

    H(ejω)=Y(ejω)X(ejω)=k=0Mbkejkωk=0NakejkωH(e^{j\omega})={Y(e^{j\omega})\over X(e^{j\omega})}={\sum_{k=0}^{M}b_{k}e^{-jk\omega}\over\sum_{k=0}^{N}a_{k}e^{-jk\omega}}

    可见由线性常系数差分方程描述的 LTI 系统的频率响应是一个有理函数。

    若要求 h[n]h[n] ,也就是系统的单位冲激响应,则往往可以对有理函数(有理分式)形式的 H(ejω)H(e^{j\omega}) 变形化得到常用变换对的表示形式,再利用常用变换对反变换得到 h[t]h[t]


    反之,若已知一个 LTI 系统的频率响应 H(ejω)=Y(ejω)X(ejω)H(e^{j\omega})={Y(e^{j\omega})\over X(e^{j\omega})} ,则可以列出两侧分别是关于 Y(ejω)Y(e^{j\omega})X(ejω)X(e^{j\omega}) 的方程,两侧同时进行傅里叶反变换,(注意将 ejωe^{j\omega} 视为一个整体)则可得到描述该 LTI 系统的线性常系数微分方程

    \ No newline at end of file +M834 80h400000v40h-400000z"/>1argX(e(jω)=arctan1acosωasinω

    信号1

    # 衰减脉冲 (双侧)

    * 名字是我乱起的

    x[n]=ana<1x[n]=a^{|n|}\ \ \ \ \ |a|\lt1

    x[n]=anu[n1]+anu[n]x[n]=a^{-n}u[-n-1]+a^nu[n]

    有傅里叶正变换:

    X(ejω)=n=1anejωn+n=0+anejωn=n=1+anejωn+n=0+anejωn=n=1+(aejω)n+n=0+(aejω)n=aejω1aejω+11aejω=1a21+a22acosω\begin{aligned} X(e^{j\omega}) &=\sum_{n=-\infin}^{-1}a^{-n}e^{-j\omega n}+\sum_{n=0}^{+\infin}a^ne^{-j\omega n}\\ &=\sum_{n=1}^{+\infin}a^{n}e^{j\omega n}+\sum_{n=0}^{+\infin}a^ne^{-j\omega n}\\ &=\sum_{n=1}^{+\infin}(ae^{j\omega})^n+\sum_{n=0}^{+\infin}(ae^{-j\omega})^n\\ &=\frac{ae^{j\omega}}{1-ae^{j\omega}}+\frac{1}{1-ae^{-j\omega}}\\ &=\frac{1-a^2}{1+a^2-2a\cos\omega} \end{aligned}

    # 冲激信号

    x[n]=δ[n]x[n]=\delta[n]

    有傅里叶正变换:

    X(ejω)=n=+δ[n]ejωn=1X(e^{j\omega})=\sum_{n=-\infin}^{+\infin}\delta[n]e^{-j\omega n}=1

    δ(t)\delta(t) 类似,δ[n]\delta[n] 中也包括所有的频率成分,且所有频率分量的幅度、相位都相同

    # 矩形脉冲

    x[n]={1,nN10,n>N1x[n]= \begin{cases} 1,\ \ \ |n|\leq N_1\\ 0,\ \ \ |n|\gt N_1 \end{cases}

    有傅里叶正变换:

    X(ejω)=n=N1+N1ejωn=sin[(2N1+1)ω2]sinω2X(e^{j\omega})=\sum_{n=-N_1}^{+N_1}e^{-j\omega n}=\frac{\sin[(2N_1+1)\frac{\omega}{2}]}{\sin\frac{\omega}{2}}

    显然,将 X(ejω)X(e^{j\omega}) 中的 ω\omega 代之以 kω0k\omega_0 再乘以 1N\frac{1}{N} ,即是相应周期延拓序列的频谱(傅里叶级数的系数):ak=1NX(ejω)ω=kωa_k=\frac{1}{N}X(e^{j\omega})|_{\omega=k\omega}

    # 采样信号

    X(ejω)={1,ω<W0,W<ωπX(e^{j\omega})= \begin{cases} 1,\ \ \ |\omega|\lt W\\ 0,\ \ \ W\lt|\omega|\leq\pi \end{cases}

    有傅里叶正变换:

    x[n]=12πWWejωndω=sinWnπn=WπSa(Wn)=WπSinc(Wnπ)\begin{aligned} x[n] &=\frac{1}{2\pi}\int_{-W}^{W}e^{j\omega n}d\omega\\ &=\frac{\sin Wn}{\pi n}\\ &=\frac{W}{\pi}\text{Sa}(Wn)\\ &=\frac{W}{\pi}\text{Sinc}(\frac{Wn}{\pi}) \end{aligned}

    # 常数信号

    x[n]=1x[n]=1

    因为:

    ππδ(ω)ejωndω=112πππ2πδ(ω)ejωndω=1\begin{aligned} \int_{-\pi}^{\pi}\delta(\omega)\cdot e^{j\omega n}d\omega&=1\\ \frac{1}{2\pi}\int_{-\pi}^{\pi}2\pi\cdot\delta(\omega)\cdot e^{j\omega n}d\omega&=1 \end{aligned}

    2πδ(ω)2\pi\delta(\omega)2π2\pi 为周期进行周期延拓,得到 x[n]=1x[n]=1 的傅里叶正变换为:

    X(ejω)=2πl=+δ(ω2πl)X(e^{j\omega})=2\pi\sum_{l=-\infin}^{+\infin}\delta(\omega-2\pi l)

    # 周期信号

    # FT 公式推导

    对连续时间信号,有 ejω0tCTFT2πδ(ωω0)e^{j\omega_0 t}\stackrel{CTFT}{\longleftrightarrow}2\pi\delta(\omega-\omega_0) ,由此推断,对离散时间信号或许有相似的情况

    但对于 x[n]=ejkω0nx[n]=e^{jk\omega_0 n} ,由于离散时间傅里叶变换是以 2π2\pi 为周期的,因此频域的冲激应该是周期性的冲激串(周期延拓),也就是:

    X(ejω)=l=+2πδ(ωkω02πl)X(e^{j\omega})=\sum_{l=-\infin}^{+\infin}2\pi\delta(\omega-k\omega_0-2\pi l)

    将反变换积分范围包含 ω0\omega_0 ,可以得到:

    x[n]=12π2πX(ejω)ejωndω=ω0πω0+πδ(ωω0)ejωndω=ejω0n\begin{aligned} x[n] &=\frac{1}{2\pi}\int_{2\pi}X(e^{j\omega})e^{j\omega n}d\omega\\ &=\int_{\omega_0-\pi}^{\omega_0+\pi}\delta(\omega-\omega_0)e^{j\omega n}d\omega\\ &=e^{j\omega_0 n} \end{aligned}

    于是对于表示为傅里叶级数的离散时间周期信号:

    x[n]=k=Nakejkω0n,ω0=2πNx[n]=\sum_{k=\langle N\rangle}a_{k}e^{jk\omega_0 n},\ \ \ \omega_0=\frac{2\pi}{N}

    就有(傅里叶正变换):

    X(ejω)=F{x[n]}=k=NakF{ejkω0n}=k=Nakl=+2πδ(ωkω02πl)=2πl=+k=Nakδ(ωkω0lNω0)=2πl=+k=0N1akδ(ω(lN+k)ω0)=2πl=+lN+k=lNlN+N1alN+kδ(ω(lN+k)ω0)=2πl=+lN+k=lNlN+N1alN+kδ(ω(lN+k)ω0)=2πl=+akδ(ωkω0)\begin{aligned} X(e^{j\omega})=\mathscr{F}\{x[n]\} &=\sum_{k=\langle N\rangle}a_{k}\cdot\mathscr{F}\{e^{jk\omega_{0}n}\}\\ &=\sum_{k=\langle N\rangle}a_{k}\sum_{l=-\infin}^{+\infin}2\pi\delta(\omega-k\omega_0-2\pi l)\\ &=2\pi\sum_{l=-\infin}^{+\infin}\sum_{k=\langle N\rangle}a_{k}\delta(\omega-k\omega_0-lN\omega_0)\\ &=2\pi\sum_{l=-\infin}^{+\infin}\sum_{k=0}^{N-1}a_{k}\delta(\omega-(lN+k)\omega_0)\\ &=2\pi\sum_{l=-\infin}^{+\infin}\ \sum_{lN+k=lN}^{lN+N-1}a_{lN+k}\delta(\omega-(lN+k)\omega_0)\\ &=2\pi\sum_{l=-\infin}^{+\infin}\ \sum_{lN+k=lN}^{lN+N-1}a_{lN+k}\delta(\omega-(lN+k)\omega_0)\\ &=2\pi\sum_{l=-\infin}^{+\infin}a_{k}\delta(\omega-k\omega_0) \end{aligned}

    可以发现 DTFT 与 CTFT 形式一致

    # 性质

    x[n]DTFTX(ejω)x[n]\stackrel{DTFT}{\longleftrightarrow}X(e^{j\omega}) ,则有以下性质成立:

    # 周期性

    X(ej(ω+2π))=X(ejω)X(e^{j(\omega+2\pi)})=X(e^{j\omega})

    # 线性

    x[n]DTFTX(ejω),y[n]DTFTY(ejω)ax[n]+by[n]DTFTaX(ejω)+bY(ejω)x[n]\stackrel{DTFT}{\longleftrightarrow}X(e^{j\omega})\ \ ,\ \ y[n]\stackrel{DTFT}{\longleftrightarrow}Y(e^{j\omega})\\ \dArr\\ ax[n]+by[n]\stackrel{DTFT}{\longleftrightarrow}aX(e^{j\omega})+bY(e^{j\omega})

    # 时移特性

    x[nn0]DTFTX(ejω)ejωn0x[n-n_0]\stackrel{DTFT}{\longleftrightarrow}X(e^{j\omega})e^{-j\omega n_0}

    # 频移特性

    x[n]ejω0nDTFTX(ej(ωω0))x[n]e^{j\omega_0 n}\stackrel{DTFT}{\longleftrightarrow}X(e^{j(\omega-\omega_0)})

    # 时域反转

    x[n]DTFTX(ejω)x[-n]\stackrel{DTFT}{\longleftrightarrow}X(e^{-j\omega})

    # 共轭对称性

    x[n]DTFTX(ejω)x^*[n]\stackrel{DTFT}{\longleftrightarrow}X^*(e^{-j\omega})

    特别地,当 x[n]x[n] 是实信号 (实函数),则有频谱: X(ejω)=X(ejω)X(e^{-j\omega})=X^*(e^{-j\omega})共轭偶函数实幅偶,虚相奇

    共轭偶函数:

    • 部是函数:Re[X(jω)]=Re[X(jω)]\text{Re}[X(j\omega)]=\text{Re}[X(-j\omega)]
    • 部是函数:Im[X(jω)]=Im[X(jω)]\text{Im}[X(j\omega)]=-\text{Im}[X(-j\omega)]
    • 幅度函数:X(jω)=X(jω)|X(j\omega)|=|X(-j\omega)|
    • 相位函数:ArgX(jω)=ArgX(jω)\text{Arg}X(j\omega)=-\text{Arg}X(-j\omega)

    # 奇偶对称性

    与 CTFT 的奇偶对称性一致

    • 偶信号的傅里叶 (正) 变换还是偶函数
    • 实偶信号的傅里叶 (正) 变换还是实偶函数
    • 奇信号的傅里叶 (正) 变换还是奇函数
    • 实奇信号的傅里叶 (正) 变换是虚奇函数

    用表格总结奇偶性、是否为实函数的不同情况下的傅里叶正变换奇偶情况:

    x[n]x[n]偶函数:x[n]=x[n]x[n]=x[-n]奇函数:x[n]=x[n]x[n]=-x[-n]
    一般的X(jω)=X(jω)X(j\omega)=X(-j\omega)X(jω)=X(jω)X(j\omega)=-X(-j\omega)
    实函数:x[n]=x[n]x[n]=x^*[n]X(jω)=X(jω)=X(jω)=X(jω)X(j\omega)=X(-j\omega)=X^*(j\omega)=X^*(-j\omega)X(jω)=X(jω)=X(jω)X(j\omega)=-X(-j\omega)=X^*(-j\omega)

    # 时域差分特性

    x[n]x[n1]DTFT(1ejω)X(ejω)x[n]-x[n-1]\stackrel{DTFT}{\longleftrightarrow}(1-e^{-j\omega})X(e^{j\omega})

    # 时域累加特性

    k=nx[k]DTFTX(ejω)1ejω+πX(ej0)k=+δ(ω2πk)\sum_{k=-\infin}^{n}x[k]\stackrel{DTFT}{\longleftrightarrow}\frac{X(e^{j\omega})}{1-e^{-j\omega}}+\pi X(e^{j0})\sum_{k=-\infin}^{+\infin}\delta(\omega-2\pi k)


    时域差分与累加特性说明:离散时间傅里叶变换中的 1ejω1-e^{-j\omega} 对应于连续时间傅里叶变换中的 jωj\omega

    # 时域内插

    信号的时域特性与频域特性之间有一种相反的关系

    定义:

    xk[n]={x[nk],n为k的整数倍0,其他nx_k[n]= \begin{cases} x[\frac{n}{k}], &\text{n为k的整数倍}\\ 0, & \text{其他n} \end{cases}

    则有:

    xk[n]DTFTX(ejkω)x_k[n]\stackrel{DTFT}{\longleftrightarrow}X(e^{jk\omega})


    证明:

    Xk(ejω)=n=+xk[n]ejωn=r=+xk[rk]ejωrk=r=+x[rkk]ejωrk=r=+x[r]ejkωr=X(ejkω)\begin{aligned} X_k(e^{j\omega}) &=\sum_{n=-\infin}^{+\infin}x_k[n]e^{-j\omega n}\\ &=\sum_{r=-\infin}^{+\infin}x_k[rk]e^{-j\omega rk}\\ &=\sum_{r=-\infin}^{+\infin}x[\frac{rk}{k}]e^{-j\omega rk}\\ &=\sum_{r=-\infin}^{+\infin}x[r]e^{-jk\omega r}\\ &=X(e^{jk\omega}) \end{aligned}

    # 频域微分

    nx[n]DTFTjdX(ejω)dωnx[n]\stackrel{DTFT}{\longleftrightarrow}j\frac{dX(e^{j\omega})}{d\omega}

    # Parseval 定理

    n=+x[n]2=12π2πX(ejω)2dω\sum_{n=-\infin}^{+\infin}|x[n]|^2=\frac{1}{2\pi}\int_{2\pi}|X(e^{j\omega})|^2d\omega

    # 卷积特性

    x[n]DTFTX(ejω),h[n]DTFTH(ejω)x[n]h[n]DTFTX(ejω)H(ejω)x[n]\stackrel{DTFT}{\longleftrightarrow}X(e^{j\omega})\ \ ,\ \ h[n]\stackrel{DTFT}{\longleftrightarrow}H(e^{j\omega})\\ \dArr\\ x[n]*h[n]\stackrel{DTFT}{\longleftrightarrow}X(e^{j\omega})H(e^{j\omega})

    该卷积性质使离散时间 LTI 系统的频域分析非常便利。本质上,卷积性质的成立仍是因为复指数信号 e^{j𝜔n} 是一切离散时间 LTI 系统的特征函数,H(ejω)H(e^{j\omega}) 则是对应的特征值。

    # 相乘特性

    如果:

    y[n]=x1[n]x2[n]y[n]=x_1[n]\cdot x_2[n]

    则:

    Y(ejω)=12π2πX1(ejθ)X2(ej(ωθ))dθ=12πX1(ejω)X2(ejω)\begin{aligned} Y(e^{j\omega}) &=\frac{1}{2\pi}\int_{2\pi}X_1(e^{j\theta})X_2(e^{j(\omega-\theta)})d\theta\\ &=\frac{1}{2\pi}X_1(e^{j\omega})\otimes X_2(e^{j\omega}) \end{aligned}

    因此上述卷积称为周期卷积,因为 X1(ejω)X_1(e^{j\omega})X2(ejω)X_2(e^{j\omega}) 都是以 2π2\pi 为周期的

    # 离散时间 LTI 系统的分析

    # 频域分析法

    ⚠️并非所有的 LTI 系统都存在频率响应!一般只考虑稳定的 LTI 系统的频率响应,因为此时 h[n]h[n] 绝对可和,离散时间傅里叶变换(即频率响应)H(ejω)H(e^{j\omega}) 存在。

    与连续时间 LTI 系统的频域分析法基本一致

    根据卷积特性,可以对 LTI 系统进行频域分析,其过程为:

    1. 求出(或已知、根据系统描述直接得出)系统单位冲激响应 h[n]h[n]
    2. 求输入 x[n]x[n] 的傅里叶正变换:x[n]DTFTX(ejω)x[n]\stackrel{DTFT}{\longleftrightarrow}X(e^{j\omega})
    3. 求单位冲激响应 h[n]h[n] 的傅里叶正变换:h[n]DTFTH(ejω)h[n]\stackrel{DTFT}{\longleftrightarrow}H(e^{j\omega})
    4. 若需要求 Y(ejω)Y(e^{j\omega}) ,由傅里叶变换卷积特性得:Y(ejω)=X(ejω)H(ejω)Y(e^{j\omega})=X(e^{j\omega})H(e^{j\omega})
    5. 若需要求 y[n]y[n]
      • x[n]x[n] 是非周期的(或非周期部分):
        • 往往由傅里叶反变换求得:y[n]=F1{Y(ejω)}=12π+Y(ejω)ejωtdωy[n]=\mathscr{F}^{-1}\{Y(e^{j\omega})\}=\frac{1}{2\pi}\int_{-\infin}^{+\infin} Y(e^{j\omega}) e^{j\omega t}d\omega
        • 对有理函数求傅里叶反变换通常采用部分分式展开和利用常用变换对进行
      • x[n]x[n] 是周期的(或周期部分):
        • Y(ejω)=X(ejω)×H(ejω)|Y(e^{j\omega})|=|X(e^{j\omega})|\times|H(e^{j\omega})|
        • argY(ejω)=argX(ejω)+argH(ejω)\text{arg}Y(e^{j\omega})=\text{arg}X(e^{j\omega})+\text{arg}H(e^{j\omega})

    # 线性常差分方程

    * 线性常系数差分方程

    对于由线性常差分方程描述的 LTI 系统:

    k=0Naky[nk]=k=0Mbkx[nk]\sum_{k=0}^{N}a_ky[n-k]=\sum_{k=0}^{M}b_kx[n-k]

    对方程两边进行傅里叶变换,有:

    k=0NakejkωY(ejω)=k=0MbkejkωX(ejω)Y(ejω)k=0Nakejkω=X(ejω)k=0MbkejkωY(ejω)X(ejω)=k=0Mbkejkωk=0Nakejkω\begin{aligned} \sum_{k=0}^{N}a_{k}e^{-jk\omega}Y(e^{j\omega}) &= \sum_{k=0}^{M}b_{k}e^{-jk\omega}X(e^{j\omega})\\ Y(e^{j\omega})\sum_{k=0}^{N}a_{k}e^{-jk\omega} &= X(e^{j\omega})\sum_{k=0}^{M}b_{k}e^{-jk\omega}\\ {Y(e^{j\omega})\over X(e^{j\omega})} &= {\sum_{k=0}^{M}b_{k}e^{-jk\omega}\over\sum_{k=0}^{N}a_{k}e^{-jk\omega}} \end{aligned}

    由 DTFT 卷积特性有:

    Y(ejω)=X(ejω)H(ejω)Y(e^{j\omega})=X(e^{j\omega})H(e^{j\omega})

    可得:

    H(ejω)=Y(ejω)X(ejω)=k=0Mbkejkωk=0NakejkωH(e^{j\omega})={Y(e^{j\omega})\over X(e^{j\omega})}={\sum_{k=0}^{M}b_{k}e^{-jk\omega}\over\sum_{k=0}^{N}a_{k}e^{-jk\omega}}

    可见由线性常系数差分方程描述的 LTI 系统的频率响应是一个有理函数。

    若要求 h[n]h[n] ,也就是系统的单位冲激响应,则往往可以对有理函数(有理分式)形式的 H(ejω)H(e^{j\omega}) 变形化得到常用变换对的表示形式,再利用常用变换对反变换得到 h[t]h[t]


    反之,若已知一个 LTI 系统的频率响应 H(ejω)=Y(ejω)X(ejω)H(e^{j\omega})={Y(e^{j\omega})\over X(e^{j\omega})} ,则可以列出两侧分别是关于 Y(ejω)Y(e^{j\omega})X(ejω)X(e^{j\omega}) 的方程,两侧同时进行傅里叶反变换,(注意将 ejωe^{j\omega} 视为一个整体)则可得到描述该 LTI 系统的线性常系数微分方程

    \ No newline at end of file diff --git a/ie/signals-and-systems/FS/index.html b/ie/signals-and-systems/FS/index.html index 6fc4b34f..80fb6401 100644 --- a/ie/signals-and-systems/FS/index.html +++ b/ie/signals-and-systems/FS/index.html @@ -1,4 +1,4 @@ -周期信号的傅里叶级数表示 - 信号与系统 - 电子信息 | Balloon Party = 風船のパーティー = fuusen no party

    # 连续时间周期信号的傅里叶级数表示

    # 连续时间傅里叶级数

    谐波关系的复指数信号集:ejkω0t,k=0,±1,±2,e^{jk\omega_0 t},\ \ \ k=0,\pm 1,\pm 2,\cdots ,满足:

    • 每个信号周期为:2πω=2πkω0\frac{2\pi}{|\omega|}=\frac{2\pi}{|k\omega_0|}
    • 公共周期为:T=2πω0T=\frac{2\pi}{|\omega_0|}
    • 各信号彼此独立

    这些信号的线性组合称为 (连续) 傅里叶级数

    x(t)=k=+akejkω0t,k=0,±1,±2,x(t)=\sum_{k=-\infin}^{+\infin}a_k e^{jk\omega_0 t},\ \ \ k=0,\pm 1,\pm 2,\cdots

    则有:

    • 基波频率 ω0\omega_0,即 k=1k=1 时的频率 ω=kω0\omega=k\omega_0
    • (基波) 周期: T=2πω0T=\frac{2\pi}{|\omega_0|}
    • aka_k 称为傅立叶级数的系数
    • k = 0 对应直流分量
    • k = ±1 对应一次谐波
    • k = ±2 对应二次谐波

    # 频谱图

    在傅里叶级数中,各个信号分量(谐波分量)间的区别仅仅是系数频率不同。

    因此,可以用线段来表示分量的系数,用线段的位置表示相应的频率(谐波次数)

    实系数傅里叶级数

    x(t)=k=+akejkω0t,k=0,±1,±2,,akRx(t)=\sum_{k=-\infin}^{+\infin}a_k e^{jk\omega_0 t},\ \ \ k=0,\pm 1,\pm 2,\cdots,\ \ \ a_k\in\mathbb{R}

    可以表示为:

    频谱图1

    akCa_k\in\mathbb{C} 为复系数傅里叶级数时,通常分幅度与相位分别作图:

    频谱图2

    # 确定系数

    如果周期信号 x (t) 可以表示为傅里叶级数:

    x(t)=k=+akejkω0t,k=0,±1,±2,,ω0=2πTx(t)=\sum_{k=-\infin}^{+\infin}a_k e^{jk\omega_0 t},\ \ \ k=0,\pm 1,\pm 2,\cdots,\ \ \ \omega_0=\frac{2\pi}{T}

    则有:

    x(t)ejnω0t=k=+akej(kn)ω0tx(t) e^{-jn\omega_0 t} = \sum_{k=-\infin}^{+\infin}a_k e^{j(k-n)\omega_0 t}

    对两边同时在一个周期内积分,有:

    0Tx(t)ejnω0tdy=k=+ak0Tej(kn)ω0tdt\begin{aligned} \int_{0}^{T}x(t) e^{-jn\omega_0 t}dy &= \sum_{k=-\infin}^{+\infin} a_k \int_{0}^{T} e^{j(k-n)\omega_0 t}dt \end{aligned}

    其中:

    0Tej(kn)ω0tdt=0Tcos((kn)ω0t)dt+j0Tsin((kn)ω0t)dt={0,knT,k=n\begin{aligned} \int_{0}^{T} e^{j(k-n)\omega_0 t}dt &=\int_{0}^{T}cos((k-n)\omega_0 t)dt+j\int_{0}^{T}sin((k-n)\omega_0 t)dt\\ &= \begin{cases} 0,\ \ \ k\neq n\\ T,\ \ \ k=n \end{cases} \end{aligned}

    所以:

    0Tx(t)ejnω0tdt=k=+ak0Tej(kn)ω0tdt=anT\begin{aligned} \int_{0}^{T}x(t) e^{-jn\omega_0 t}dt &= \sum_{k=-\infin}^{+\infin} a_k \int_{0}^{T} e^{j(k-n)\omega_0 t}dt\\ &= a_n T \end{aligned}

    可得 ana_n

    an=1T0Tx(t)ejnω0tdt\begin{aligned} a_n &= \frac{1}{T}\int_{0}^{T}x(t) e^{-jn\omega_0 t}dt \end{aligned}

    在确定此积分时,只要积分区间是一个周期即可,对积分区间的起止并无特别要求,因此可表示为:

    ak=1TTx(t)ejkω0tdt=1TTx(t)[cos(kω0t)jsin(kω0t)]dt\begin{aligned} a_k &= \frac{1}{T}{\int}_{T} x(t) e^{-jk\omega_0 t}dt\\ &= \frac{1}{T}{\int}_{T} x(t) [\text{cos}(k\omega_0t)-j\ \text{sin}(k\omega_0t)]dt \end{aligned}

    n=0n=0 时信号在一个周期的平均值,通常称直流分量

    a0=1TTx(t)dt\begin{aligned} a_0 &= \frac{1}{T}{\int}_{T} x(t) dt \end{aligned}

    # 周期性矩形脉冲信号

    周期性矩形脉冲信号

    ak=1TTx(t)ejkω0tdt=1TT1T1ejkω0tdt=1jkω0Tejkω0T1T1=2sin(kω0T1)kω0T=2T1T2sin(kω0T1)kω0T1(ω0=2πT)=2T1TSa(kπ2T1T)\begin{aligned} a_k &= \frac{1}{T}{\int}_{T} x(t) e^{-jk\omega_0 t}dt\\ &= \frac{1}{T}{\int}_{-T_1}^{T_1}e^{-jk\omega_0 t}dt\\ &= -\frac{1}{jk\omega_0 T} e^{-jk\omega_0}|_{-T_1}^{T_1}\\ &= \frac{2\text{sin}(k\omega_0 T_1)}{k\omega_0 T}\\ &= \frac{2T_1}{T}\frac{2\text{sin}(k\omega_0 T_1)}{k\omega_0 T_1}\\ (\omega_0=\frac{2\pi}{T}) &= \frac{2T_1}{T}\text{Sa}(k\pi\frac{2T_1}{T}) \end{aligned}

    其中 Sa(...)\text{Sa}(...) 是抽样信号:

    Sa(x)=sinxx\text{Sa}(x)=\frac{\text{sin}x}{x}

    基波频率为:ω0=2πT\omega_0=\frac{2\pi}{T}

    类似地:

    sinc(x)=sinπxπx\text{sinc}(x)=\frac{\text{sin}\pi x}{\pi x}

    抽样信号

    不同占空比的周期性矩形脉冲信号的傅里叶级数表示:

    不同占空比的周期性矩形脉冲信号的傅里叶级数表示

    # 收敛性

    x(t)x(t) 以 T 为周期,用有限项级数:

    xN(t)=k=NNakejkω0tx_N(t)=\sum_{k=-N}^{N}a_k e^{jk\omega_0 t}

    近似 x(t)x(t) 时,定义误差:

    eN(t)=x(t)xN(t)=x(t)k=NNakejkω0te_N(t)=x(t)-x_N(t)=x(t)-\sum_{k=-N}^{N}a_k e^{jk\omega_0 t}

    误差在一个周期内的能量为:

    EN=TeN(t)2dtE_N=\int_{T}|e_N(t)|^2dt

    可以证明,要使 ENE_N 最小,应有:

    ak=1TTx(t)ejkω0tdta_k= \frac{1}{T}{\int}_{T} x(t) e^{-jk\omega_0 t}dt

    结论:在误差能量ENE_N 最小的准则下,傅里叶级数是对周期信号的最佳近似

    # 收敛的充分条件

    两组条件并不完全等价。它们都是傅里叶级数收敛的充分条件。

    相当广泛的信号都能满足这两组条件中的一组,因而用傅里叶级数表示周期信号具有相当的普遍适用性

    # 平方可积条件

    如果:

    Tx(t)2dt<\int_{T}|x(t)|^2dt \lt \infin

    aka_k 必存在。

    此时 x (t) 与其傅里叶级数表示之间没有能量上的区别

    # Dirichlet 条件

    (1)x(t)x(t) 在任何周期内信号绝对可积,即:

    Tx(t)dt<\int_{T}|x(t)|dt \lt \infin

    (2)在任何有限区间内,只有有限个极值点,且极值为有限值

    (3)在任何有限区间内,只有有限个间断点,且在这些间断点上,函数值有限

    Dirichlet 条件保证在间断点之外 x(t)x(t) 等于它的傅里叶级数;而在间断点上,该无穷级数收敛于间断点两边值的平均值

    # 实信号的傅里叶级数其他形式 (如考)

    x(t)x(t) 是实信号,则有 :

    x(t)=x(t)x(t)=x^*(t)

    从而有:

    x(t)=[k=+akejkω0t]x^*(t)=[\sum_{k=-\infin}^{+\infin}a_k e^{jk\omega_0 t}]^*


    # Gibbs 现象

    满足 Dirichlet 条件的信号,其傅里叶级数是如何收敛于 x(t)x(t) 的?特别当 x(t)x(t) 具有间断点时,在间断点附近,如何收敛于 x(t)x(t) ?

    Gibbs 现象表明:

    用有限项傅里叶级数表示有间断点的信号时,在间断点附近不可避免的会出现振荡和超量。超量的幅度不会随所取项数的增加而减小。只是随着项数的增多,振荡频率变高,并向间断点处压缩,从而使它所占有的能量减少


    # 离散时间周期信号的傅里叶级数表示

    # 离散时间傅里叶级数

    谐波关系的复指数信号集:Φk[n]={ej2πNkt},k=0,±1,±2,\Phi_k[n]=\{e^{j\frac{2\pi}{N}kt}\},\ \ \ k=0,\pm 1,\pm 2,\cdots ,满足:

    • 每个信号周期为:Nk\frac{N}{k}
    • 公共周期为:NN
    • 各信号彼此独立

    这些信号的线性组合称为离散傅里叶级数

    x[n]=k=Nakej2πNkn,k=0,±1,±2,x[n]=\sum_{k=\langle N\rangle}a_k e^{j\frac{2\pi}{N}kn},\ \ \ k=0,\pm 1,\pm 2,\cdots

    ⭐️其中 N\langle N\rangle 表示一个总共 NN 个的连续序列,起始和结束没有要求。

    则有:

    • 基波频率 ω0=2πN\omega_0=\frac{2\pi}{N},即 k=1k=1 时的频率 ω=kω0=2kπN\omega=k\omega_0=\frac{2k\pi}{N}
    • (基波) 周期:NN
    • aka_k 为傅立叶级数的系数,也称为 x[n]x[n]频谱系数

    # 确定系数

    x[n]=k=Nakej2πNktx[n]=\sum_{k=\langle N\rangle}a_{k}e^{j\frac{2\pi}{N}kt} 两边同时乘 ej2πNrne^{-j\frac{2\pi}{N}rn} ,可得:

    x[n]ej2πNrn=k=Nakej2πN(kr)nx[n]e^{-j\frac{2\pi}{N}rn}=\sum_{k=\langle N\rangle}a_k e^{j\frac{2\pi}{N}(k-r)n}

    显然 x[n]ej2πNrnx[n]e^{-j\frac{2\pi}{N}rn} 仍然是以 N 为周期的,对两边求和:

    n=Nx[n]ej2πNrn=n=Nk=Nakej2πN(kr)n=k=Nakn=Nej2πN(kr)n\sum_{n=\langle N\rangle}x[n]e^{-j\frac{2\pi}{N}rn} =\sum_{n=\langle N\rangle}\sum_{k=\langle N\rangle}a_k e^{j\frac{2\pi}{N}(k-r)n} =\sum_{k=\langle N\rangle}a_k \sum_{n=\langle N\rangle}e^{j\frac{2\pi}{N}(k-r)n}

    其中:

    n=Nej2πN(kr)n=n=0N1ej2πN(kr)n=1ej(kr)2π1ej(kr)2πN={0,krN,k=r\begin{aligned} \sum_{n=\langle N\rangle}e^{j\frac{2\pi}{N}(k-r)n} &= \sum_{n=0}^{N-1}e^{j\frac{2\pi}{N}(k-r)n}\\ &= \frac{1-e^{j(k-r)2\pi}}{1-e^{j(k-r)\frac{2\pi}{N}}}\\ &= \begin{cases} 0,\ \ \ k\neq r\\ N,\ \ \ k=r \end{cases} \end{aligned}

    从而:

    n=Nx[n]ej2πNrn=Nar\sum_{n=\langle N\rangle}x[n]e^{-j\frac{2\pi}{N}rn}=Na_r

    即可得:(r 换成了 k)

    ak=1Nn=Nx[n]ej2πNkna_k=\frac{1}{N}\sum_{n=\langle N\rangle}x[n]e^{-j\frac{2\pi}{N}kn}

    显然上式满足 ak+N=aka_{k+N}=a_k ,即也是以 NN 为周期的,或者说 aka_k 中只有 NN 个是独立的

    # 周期性方波序列

    周期性方波序列

    ak=1Nn=Nx[n]ej2πNkn=1Nn=N1N1ej2πNkn=1Nej2πNkN1ej2πNk(N1+1)1ej2πNk=1NejπNkejπNkej2πNk(N1+12)ej2πNk(N1+12)ejπNkejπNk=1NsinπNk(2N1+1)sinπNk,k0,±N,±2N,\begin{aligned} a_k &= \frac{1}{N}\sum_{n=\langle N\rangle}x[n]e^{-j\frac{2\pi}{N}kn} \\ &= \frac{1}{N}\sum_{n=-N_1}^{N_1}e^{-j\frac{2\pi}{N}kn} \\ &= \frac{1}{N}\frac {e^{j\frac{2\pi}{N}kN_1}-e^{-j\frac{2\pi}{N}k(N_1+1)}} {1-e^{-j\frac{2\pi}{N}k}} \\ &= \frac{1}{N}\frac{e^{-j\frac{\pi}{N}k}}{e^{-j\frac{\pi}{N}k}}\frac {e^{j\frac{2\pi}{N}k(N_1+\frac{1}{2})}-e^{-j\frac{2\pi}{N}k(N_1+\frac{1}{2})}} {e^{j\frac{\pi}{N}k}-e^{-j\frac{\pi}{N}k}} \\ &= \frac{1}{N}\frac{\text{sin}\frac{\pi}{N}k(2N_1+1)}{\text{sin}\frac{\pi}{N}k},\ \ \ \ k\neq 0,\pm N,\pm 2N,\cdots \end{aligned}

    ak=2N1+1N,k=rNa_k = \frac{2N_1+1}{N},\ \ \ \ k=rN

    # 收敛性

    离散时间傅里叶级数是一个有限项的级数,确定 aka_k 的关系式也是有限项的和式,因而不存在收敛问题,也不会产生 Gibbs 现象

    # LTI 系统对复指数信号的响应

    # 特征函数与特征值

    考查 LTI 系统对复指数信号 este^{st} 作为输入时的响应

    连续复指数信号输入

    y(t)=+es(tτ)h(τ)dτ=est+esτh(τ)dτ=estH(s)\begin{aligned} y(t) &=\int_{-\infin}^{+\infin}e^{s(t-\tau)}h(\tau)d\tau \\ &= e^{st}\int_{-\infin}^{+\infin}e^{-s\tau}h(\tau)d\tau \\ &= e^{st}H(s) \end{aligned}

    其中:

    H(s)=+esτh(τ)dτH(s) = \int_{-\infin}^{+\infin}e^{-s\tau}h(\tau)d\tau

    考查 LTI 系统对复指数信号 znz^n 作为输入时的响应

    离散复指数信号输入

    y[n]=k=+znkh[k]=znk=+zkh[k]=znH(z)\begin{aligned} y[n] &=\sum_{k=-\infin}^{+\infin}z^{n-k}h[k] \\ &=z^{n}\sum_{k=-\infin}^{+\infin}z^{-k}h[k] \\ &= z^{n}H(z) \end{aligned}

    其中:

    H(z)=k=+zkh[k]H(z) = \sum_{k=-\infin}^{+\infin}z^{-k}h[k]

    如果系统对某一信号的响应是该信号乘以一个常数,则称该信号是这个系统的特征函数,称该常数为系统与特征函数相对应的特征值

    类比线性代数中的特征向量和特征值

    复指数函数 este^{st}znz^n一切 LTI 系统的特征函数H(s)H(s)H(z)H(z) 分别是 LTI 系统与复指数信号相对应的特征值

    # 频率响应

    对时域的任何一个信号 x(t)x(t) 或者 x[n]x[n],若能将其表示为下列形式:

    x(t)=kakesktx[n]=kakzknx(t) = \sum_{k}a_k\ e^{s_k t}\\ x[n] = \sum_{k}a_k\ z_{k}^{n}

    s=jωs=j\omega ,考虑形如 ejωte^{j\omega t} 的基本信号,则有连续时间傅里叶级数与傅里叶级数变换:

    H(jω)H(j\omega) 被称为连续时间 LTI 系统的频率响应

    H(jω)=+h(t)ejωtdtH(j\omega) = \int_{-\infin}^{+\infin}h(t)e^{-j\omega t}dt

    z=ejωz=e^{j\omega} ,考虑形如 ejωne^{j\omega n} 的基本信号,则有离散时间傅里叶级数与傅里叶级数变换;

    H(ejω)H(e^{j\omega}) 称为离散时间 LTI 系统的频率响应

    H(ejω)=n=+h[n]ejωnH(e^{j\omega}) = \sum_{n=-\infin}^{+\infin}h[n]e^{-j\omega n}

    利用 LTI 系统的特征函数以及齐次性与叠加性,则有:

    输入响应\text{输入}\to\text{响应}

    ejωtejωtH(jω)x(t)=kakejkω0ty(t)=kakejkω0tH(jkω0)e^{j\omega t}\to e^{j\omega t}\ H(j\omega ) \\ x(t)= \sum_{k}a_k\ e^{jk\omega_0 t}\to y(t)= \sum_{k}a_k\ e^{jk\omega_0 t}\ H(jk\omega_0)

    ejωnejωnH(ejω)x[n]=kakej2πNkny[n]=kakej2πNknH(ej2πNk)e^{j\omega n}\to e^{j\omega n}\ H(e^{j\omega})\\ x[n] = \sum_{k}a_k\ e^{j\frac{2\pi}{N}kn}\to y[n]=\sum_{k}a_k\ e^{j\frac{2\pi}{N}kn}\ H(e^{j\frac{2\pi}{N}k})

    有的题目里会有ω=kω0\omega=k\omega_0 关系,直接给出H(jω)H(j\omega)H(ejω)H(e^{j\omega})

    LTI 系统对周期信号的响应仍是一个周期信号;

    LTI 系统频率响应的意义在于对各个谐波频率的信号分量进行不同的加权处理.

    # 滤波

    滤波:改变信号中各频率分量的相对大小或消除某些频率分量的过程。

    • 频率成形滤波器:改变信号频谱形状的滤波器
    • 频率选择性滤波器:基本上无失真地通过某些频率,而显著地衰减或消除另一些频率的滤波器

    # 微分器

    属于频率成形滤波器

    H(jω)=+u1(t)ejωtdt=jωH(j\omega) = \int_{-\infin}^{+\infin}u_1(t)e^{-j\omega t}dt=j\omega

    幅度特性:

    幅度特性

    相位特性:

    相位特性

    # 低通滤波器 LPF

    Lowpass filter,通过低频分量,衰减或阻止高频分量

    # 高通滤波器 HPF

    Highpass filter,通过高频而衰减或阻止低频

    # 带通滤波器 BPF

    Bandpass filter,通过一频带范围,衰减或阻止这一频带之外的频率分量

    # 带阻滤波器

    Band-stop filter 或 band-rejection filter

    # 微分方程描述的连续时间滤波器

    # RC 低通滤波器

    RC-C电压的响应

    频率响应 H(jω)H(j\omega),设系统初始松弛,则该系统为 LTI 系统,输入 vs(t)=ejωtv_s(t) = e^{jωt} 将产生输出 vc(t)=H(jω)ejωtv_c(t) = H(j\omega) e^{jωt} ,因此:

    RCddtvc(t)+vc(t)=vs(t)RCddt[H(jω)ejωt]+H(jω)ejωt=ejωt\begin{aligned} RC\frac{d}{dt}v_c(t)+v_c(t)&=v_s(t)\\ RC\frac{d}{dt}[H(j\omega)e^{j\omega t}]+H(j\omega)e^{j\omega t}&=e^{j\omega t} \end{aligned}

    可得:

    H(jω)=11+RCjωH(jω)=1s1+(RCω)2H(jω)=arctan(RCω)\begin{aligned} H(j\omega)&=\frac{1}{1+RCj\omega} \\ |H(j\omega)|&=\frac{1}{s\sqrt{1+(RC\omega)^2}}\\ \angle H(j\omega)&=-\text{arctan}(RC\omega) \end{aligned}

    # 连续时间周期信号的傅里叶级数表示

    # 连续时间傅里叶级数

    谐波关系的复指数信号集:ejkω0t,k=0,±1,±2,e^{jk\omega_0 t},\ \ \ k=0,\pm 1,\pm 2,\cdots ,满足:

    • 每个信号周期为:2πω=2πkω0\frac{2\pi}{|\omega|}=\frac{2\pi}{|k\omega_0|}
    • 公共周期为:T=2πω0T=\frac{2\pi}{|\omega_0|}
    • 各信号彼此独立

    这些信号的线性组合称为 (连续) 傅里叶级数

    x(t)=k=+akejkω0t,k=0,±1,±2,x(t)=\sum_{k=-\infin}^{+\infin}a_k e^{jk\omega_0 t},\ \ \ k=0,\pm 1,\pm 2,\cdots

    则有:

    • 基波频率 ω0\omega_0,即 k=1k=1 时的频率 ω=kω0\omega=k\omega_0
    • (基波) 周期: T=2πω0T=\frac{2\pi}{|\omega_0|}
    • aka_k 称为傅立叶级数的系数
    • k = 0 对应直流分量
    • k = ±1 对应一次谐波
    • k = ±2 对应二次谐波

    # 频谱图

    在傅里叶级数中,各个信号分量(谐波分量)间的区别仅仅是系数频率不同。

    因此,可以用线段来表示分量的系数,用线段的位置表示相应的频率(谐波次数)

    实系数傅里叶级数

    x(t)=k=+akejkω0t,k=0,±1,±2,,akRx(t)=\sum_{k=-\infin}^{+\infin}a_k e^{jk\omega_0 t},\ \ \ k=0,\pm 1,\pm 2,\cdots,\ \ \ a_k\in\mathbb{R}

    可以表示为:

    频谱图1

    akCa_k\in\mathbb{C} 为复系数傅里叶级数时,通常分幅度与相位分别作图:

    频谱图2

    # 确定系数

    如果周期信号 x (t) 可以表示为傅里叶级数:

    x(t)=k=+akejkω0t,k=0,±1,±2,,ω0=2πTx(t)=\sum_{k=-\infin}^{+\infin}a_k e^{jk\omega_0 t},\ \ \ k=0,\pm 1,\pm 2,\cdots,\ \ \ \omega_0=\frac{2\pi}{T}

    则有:

    x(t)ejnω0t=k=+akej(kn)ω0tx(t) e^{-jn\omega_0 t} = \sum_{k=-\infin}^{+\infin}a_k e^{j(k-n)\omega_0 t}

    对两边同时在一个周期内积分,有:

    0Tx(t)ejnω0tdy=k=+ak0Tej(kn)ω0tdt\begin{aligned} \int_{0}^{T}x(t) e^{-jn\omega_0 t}dy &= \sum_{k=-\infin}^{+\infin} a_k \int_{0}^{T} e^{j(k-n)\omega_0 t}dt \end{aligned}

    其中:

    0Tej(kn)ω0tdt=0Tcos((kn)ω0t)dt+j0Tsin((kn)ω0t)dt={0,knT,k=n\begin{aligned} \int_{0}^{T} e^{j(k-n)\omega_0 t}dt &=\int_{0}^{T}cos((k-n)\omega_0 t)dt+j\int_{0}^{T}sin((k-n)\omega_0 t)dt\\ &= \begin{cases} 0,\ \ \ k\neq n\\ T,\ \ \ k=n \end{cases} \end{aligned}

    所以:

    0Tx(t)ejnω0tdt=k=+ak0Tej(kn)ω0tdt=anT\begin{aligned} \int_{0}^{T}x(t) e^{-jn\omega_0 t}dt &= \sum_{k=-\infin}^{+\infin} a_k \int_{0}^{T} e^{j(k-n)\omega_0 t}dt\\ &= a_n T \end{aligned}

    可得 ana_n

    an=1T0Tx(t)ejnω0tdt\begin{aligned} a_n &= \frac{1}{T}\int_{0}^{T}x(t) e^{-jn\omega_0 t}dt \end{aligned}

    在确定此积分时,只要积分区间是一个周期即可,对积分区间的起止并无特别要求,因此可表示为:

    ak=1TTx(t)ejkω0tdt=1TTx(t)[cos(kω0t)jsin(kω0t)]dt\begin{aligned} a_k &= \frac{1}{T}{\int}_{T} x(t) e^{-jk\omega_0 t}dt\\ &= \frac{1}{T}{\int}_{T} x(t) [\text{cos}(k\omega_0t)-j\ \text{sin}(k\omega_0t)]dt \end{aligned}

    n=0n=0 时信号在一个周期的平均值,通常称直流分量

    a0=1TTx(t)dt\begin{aligned} a_0 &= \frac{1}{T}{\int}_{T} x(t) dt \end{aligned}

    # 周期性矩形脉冲信号

    周期性矩形脉冲信号

    ak=1TTx(t)ejkω0tdt=1TT1T1ejkω0tdt=1jkω0Tejkω0T1T1=2sin(kω0T1)kω0T=2T1T2sin(kω0T1)kω0T1(ω0=2πT)=2T1TSa(kπ2T1T)\begin{aligned} a_k &= \frac{1}{T}{\int}_{T} x(t) e^{-jk\omega_0 t}dt\\ &= \frac{1}{T}{\int}_{-T_1}^{T_1}e^{-jk\omega_0 t}dt\\ &= -\frac{1}{jk\omega_0 T} e^{-jk\omega_0}|_{-T_1}^{T_1}\\ &= \frac{2\text{sin}(k\omega_0 T_1)}{k\omega_0 T}\\ &= \frac{2T_1}{T}\frac{2\text{sin}(k\omega_0 T_1)}{k\omega_0 T_1}\\ (\omega_0=\frac{2\pi}{T}) &= \frac{2T_1}{T}\text{Sa}(k\pi\frac{2T_1}{T}) \end{aligned}

    其中 Sa(...)\text{Sa}(...) 是抽样信号:

    Sa(x)=sinxx\text{Sa}(x)=\frac{\text{sin}x}{x}

    基波频率为:ω0=2πT\omega_0=\frac{2\pi}{T}

    类似地:

    sinc(x)=sinπxπx\text{sinc}(x)=\frac{\text{sin}\pi x}{\pi x}

    抽样信号

    不同占空比的周期性矩形脉冲信号的傅里叶级数表示:

    不同占空比的周期性矩形脉冲信号的傅里叶级数表示

    # 收敛性

    x(t)x(t) 以 T 为周期,用有限项级数:

    xN(t)=k=NNakejkω0tx_N(t)=\sum_{k=-N}^{N}a_k e^{jk\omega_0 t}

    近似 x(t)x(t) 时,定义误差:

    eN(t)=x(t)xN(t)=x(t)k=NNakejkω0te_N(t)=x(t)-x_N(t)=x(t)-\sum_{k=-N}^{N}a_k e^{jk\omega_0 t}

    误差在一个周期内的能量为:

    EN=TeN(t)2dtE_N=\int_{T}|e_N(t)|^2dt

    可以证明,要使 ENE_N 最小,应有:

    ak=1TTx(t)ejkω0tdta_k= \frac{1}{T}{\int}_{T} x(t) e^{-jk\omega_0 t}dt

    结论:在误差能量ENE_N 最小的准则下,傅里叶级数是对周期信号的最佳近似

    # 收敛的充分条件

    两组条件并不完全等价。它们都是傅里叶级数收敛的充分条件。

    相当广泛的信号都能满足这两组条件中的一组,因而用傅里叶级数表示周期信号具有相当的普遍适用性

    # 平方可积条件

    如果:

    Tx(t)2dt<\int_{T}|x(t)|^2dt \lt \infin

    aka_k 必存在。

    此时 x (t) 与其傅里叶级数表示之间没有能量上的区别

    # Dirichlet 条件

    (1)x(t)x(t) 在任何周期内信号绝对可积,即:

    Tx(t)dt<\int_{T}|x(t)|dt \lt \infin

    (2)在任何有限区间内,只有有限个极值点,且极值为有限值

    (3)在任何有限区间内,只有有限个间断点,且在这些间断点上,函数值有限

    Dirichlet 条件保证在间断点之外 x(t)x(t) 等于它的傅里叶级数;而在间断点上,该无穷级数收敛于间断点两边值的平均值

    # 实信号的傅里叶级数其他形式 (如考)

    x(t)x(t) 是实信号,则有 :

    x(t)=x(t)x(t)=x^*(t)

    从而有:

    x(t)=[k=+akejkω0t]x^*(t)=[\sum_{k=-\infin}^{+\infin}a_k e^{jk\omega_0 t}]^*


    # Gibbs 现象

    满足 Dirichlet 条件的信号,其傅里叶级数是如何收敛于 x(t)x(t) 的?特别当 x(t)x(t) 具有间断点时,在间断点附近,如何收敛于 x(t)x(t) ?

    Gibbs 现象表明:

    用有限项傅里叶级数表示有间断点的信号时,在间断点附近不可避免的会出现振荡和超量。超量的幅度不会随所取项数的增加而减小。只是随着项数的增多,振荡频率变高,并向间断点处压缩,从而使它所占有的能量减少


    # 离散时间周期信号的傅里叶级数表示

    # 离散时间傅里叶级数

    谐波关系的复指数信号集:Φk[n]={ej2πNkt},k=0,±1,±2,\Phi_k[n]=\{e^{j\frac{2\pi}{N}kt}\},\ \ \ k=0,\pm 1,\pm 2,\cdots ,满足:

    • 每个信号周期为:Nk\frac{N}{k}
    • 公共周期为:NN
    • 各信号彼此独立

    这些信号的线性组合称为离散傅里叶级数

    x[n]=k=Nakej2πNkn,k=0,±1,±2,x[n]=\sum_{k=\langle N\rangle}a_k e^{j\frac{2\pi}{N}kn},\ \ \ k=0,\pm 1,\pm 2,\cdots

    ⭐️其中 N\langle N\rangle 表示一个总共 NN 个的连续序列,起始和结束没有要求。

    则有:

    • 基波频率 ω0=2πN\omega_0=\frac{2\pi}{N},即 k=1k=1 时的频率 ω=kω0=2kπN\omega=k\omega_0=\frac{2k\pi}{N}
    • (基波) 周期:NN
    • aka_k 为傅立叶级数的系数,也称为 x[n]x[n]频谱系数

    # 确定系数

    x[n]=k=Nakej2πNktx[n]=\sum_{k=\langle N\rangle}a_{k}e^{j\frac{2\pi}{N}kt} 两边同时乘 ej2πNrne^{-j\frac{2\pi}{N}rn} ,可得:

    x[n]ej2πNrn=k=Nakej2πN(kr)nx[n]e^{-j\frac{2\pi}{N}rn}=\sum_{k=\langle N\rangle}a_k e^{j\frac{2\pi}{N}(k-r)n}

    显然 x[n]ej2πNrnx[n]e^{-j\frac{2\pi}{N}rn} 仍然是以 N 为周期的,对两边求和:

    n=Nx[n]ej2πNrn=n=Nk=Nakej2πN(kr)n=k=Nakn=Nej2πN(kr)n\sum_{n=\langle N\rangle}x[n]e^{-j\frac{2\pi}{N}rn} =\sum_{n=\langle N\rangle}\sum_{k=\langle N\rangle}a_k e^{j\frac{2\pi}{N}(k-r)n} =\sum_{k=\langle N\rangle}a_k \sum_{n=\langle N\rangle}e^{j\frac{2\pi}{N}(k-r)n}

    其中:

    n=Nej2πN(kr)n=n=0N1ej2πN(kr)n=1ej(kr)2π1ej(kr)2πN={0,krN,k=r\begin{aligned} \sum_{n=\langle N\rangle}e^{j\frac{2\pi}{N}(k-r)n} &= \sum_{n=0}^{N-1}e^{j\frac{2\pi}{N}(k-r)n}\\ &= \frac{1-e^{j(k-r)2\pi}}{1-e^{j(k-r)\frac{2\pi}{N}}}\\ &= \begin{cases} 0,\ \ \ k\neq r\\ N,\ \ \ k=r \end{cases} \end{aligned}

    从而:

    n=Nx[n]ej2πNrn=Nar\sum_{n=\langle N\rangle}x[n]e^{-j\frac{2\pi}{N}rn}=Na_r

    即可得:(r 换成了 k)

    ak=1Nn=Nx[n]ej2πNkna_k=\frac{1}{N}\sum_{n=\langle N\rangle}x[n]e^{-j\frac{2\pi}{N}kn}

    显然上式满足 ak+N=aka_{k+N}=a_k ,即也是以 NN 为周期的,或者说 aka_k 中只有 NN 个是独立的

    # 周期性方波序列

    周期性方波序列

    ak=1Nn=Nx[n]ej2πNkn=1Nn=N1N1ej2πNkn=1Nej2πNkN1ej2πNk(N1+1)1ej2πNk=1NejπNkejπNkej2πNk(N1+12)ej2πNk(N1+12)ejπNkejπNk=1NsinπNk(2N1+1)sinπNk,k0,±N,±2N,\begin{aligned} a_k &= \frac{1}{N}\sum_{n=\langle N\rangle}x[n]e^{-j\frac{2\pi}{N}kn} \\ &= \frac{1}{N}\sum_{n=-N_1}^{N_1}e^{-j\frac{2\pi}{N}kn} \\ &= \frac{1}{N}\frac {e^{j\frac{2\pi}{N}kN_1}-e^{-j\frac{2\pi}{N}k(N_1+1)}} {1-e^{-j\frac{2\pi}{N}k}} \\ &= \frac{1}{N}\frac{e^{-j\frac{\pi}{N}k}}{e^{-j\frac{\pi}{N}k}}\frac {e^{j\frac{2\pi}{N}k(N_1+\frac{1}{2})}-e^{-j\frac{2\pi}{N}k(N_1+\frac{1}{2})}} {e^{j\frac{\pi}{N}k}-e^{-j\frac{\pi}{N}k}} \\ &= \frac{1}{N}\frac{\text{sin}\frac{\pi}{N}k(2N_1+1)}{\text{sin}\frac{\pi}{N}k},\ \ \ \ k\neq 0,\pm N,\pm 2N,\cdots \end{aligned}

    ak=2N1+1N,k=rNa_k = \frac{2N_1+1}{N},\ \ \ \ k=rN

    # 收敛性

    离散时间傅里叶级数是一个有限项的级数,确定 aka_k 的关系式也是有限项的和式,因而不存在收敛问题,也不会产生 Gibbs 现象

    # LTI 系统对复指数信号的响应

    # 特征函数与特征值

    考查 LTI 系统对复指数信号 este^{st} 作为输入时的响应

    连续复指数信号输入

    y(t)=+es(tτ)h(τ)dτ=est+esτh(τ)dτ=estH(s)\begin{aligned} y(t) &=\int_{-\infin}^{+\infin}e^{s(t-\tau)}h(\tau)d\tau \\ &= e^{st}\int_{-\infin}^{+\infin}e^{-s\tau}h(\tau)d\tau \\ &= e^{st}H(s) \end{aligned}

    其中:

    H(s)=+esτh(τ)dτH(s) = \int_{-\infin}^{+\infin}e^{-s\tau}h(\tau)d\tau

    考查 LTI 系统对复指数信号 znz^n 作为输入时的响应

    离散复指数信号输入

    y[n]=k=+znkh[k]=znk=+zkh[k]=znH(z)\begin{aligned} y[n] &=\sum_{k=-\infin}^{+\infin}z^{n-k}h[k] \\ &=z^{n}\sum_{k=-\infin}^{+\infin}z^{-k}h[k] \\ &= z^{n}H(z) \end{aligned}

    其中:

    H(z)=k=+zkh[k]H(z) = \sum_{k=-\infin}^{+\infin}z^{-k}h[k]

    如果系统对某一信号的响应是该信号乘以一个常数,则称该信号是这个系统的特征函数,称该常数为系统与特征函数相对应的特征值

    类比线性代数中的特征向量和特征值

    复指数函数 este^{st}znz^n一切 LTI 系统的特征函数H(s)H(s)H(z)H(z) 分别是 LTI 系统与复指数信号相对应的特征值

    # 频率响应

    对时域的任何一个信号 x(t)x(t) 或者 x[n]x[n],若能将其表示为下列形式:

    x(t)=kakesktx[n]=kakzknx(t) = \sum_{k}a_k\ e^{s_k t}\\ x[n] = \sum_{k}a_k\ z_{k}^{n}

    s=jωs=j\omega ,考虑形如 ejωte^{j\omega t} 的基本信号,则有连续时间傅里叶级数与傅里叶级数变换:

    H(jω)H(j\omega) 被称为连续时间 LTI 系统的频率响应

    H(jω)=+h(t)ejωtdtH(j\omega) = \int_{-\infin}^{+\infin}h(t)e^{-j\omega t}dt

    z=ejωz=e^{j\omega} ,考虑形如 ejωne^{j\omega n} 的基本信号,则有离散时间傅里叶级数与傅里叶级数变换;

    H(ejω)H(e^{j\omega}) 称为离散时间 LTI 系统的频率响应

    H(ejω)=n=+h[n]ejωnH(e^{j\omega}) = \sum_{n=-\infin}^{+\infin}h[n]e^{-j\omega n}

    利用 LTI 系统的特征函数以及齐次性与叠加性,则有:

    输入响应\text{输入}\to\text{响应}

    ejωtejωtH(jω)x(t)=kakejkω0ty(t)=kakejkω0tH(jkω0)e^{j\omega t}\to e^{j\omega t}\ H(j\omega ) \\ x(t)= \sum_{k}a_k\ e^{jk\omega_0 t}\to y(t)= \sum_{k}a_k\ e^{jk\omega_0 t}\ H(jk\omega_0)

    ejωnejωnH(ejω)x[n]=kakej2πNkny[n]=kakej2πNknH(ej2πNk)e^{j\omega n}\to e^{j\omega n}\ H(e^{j\omega})\\ x[n] = \sum_{k}a_k\ e^{j\frac{2\pi}{N}kn}\to y[n]=\sum_{k}a_k\ e^{j\frac{2\pi}{N}kn}\ H(e^{j\frac{2\pi}{N}k})

    有的题目里会有ω=kω0\omega=k\omega_0 关系,直接给出H(jω)H(j\omega)H(ejω)H(e^{j\omega})

    LTI 系统对周期信号的响应仍是一个周期信号;

    LTI 系统频率响应的意义在于对各个谐波频率的信号分量进行不同的加权处理.

    # 滤波

    滤波:改变信号中各频率分量的相对大小或消除某些频率分量的过程。

    • 频率成形滤波器:改变信号频谱形状的滤波器
    • 频率选择性滤波器:基本上无失真地通过某些频率,而显著地衰减或消除另一些频率的滤波器

    # 微分器

    属于频率成形滤波器

    H(jω)=+u1(t)ejωtdt=jωH(j\omega) = \int_{-\infin}^{+\infin}u_1(t)e^{-j\omega t}dt=j\omega

    幅度特性:

    幅度特性

    相位特性:

    相位特性

    # 低通滤波器 LPF

    Lowpass filter,通过低频分量,衰减或阻止高频分量

    # 高通滤波器 HPF

    Highpass filter,通过高频而衰减或阻止低频

    # 带通滤波器 BPF

    Bandpass filter,通过一频带范围,衰减或阻止这一频带之外的频率分量

    # 带阻滤波器

    Band-stop filter 或 band-rejection filter

    # 微分方程描述的连续时间滤波器

    # RC 低通滤波器

    RC-C电压的响应

    频率响应 H(jω)H(j\omega),设系统初始松弛,则该系统为 LTI 系统,输入 vs(t)=ejωtv_s(t) = e^{jωt} 将产生输出 vc(t)=H(jω)ejωtv_c(t) = H(j\omega) e^{jωt} ,因此:

    RCddtvc(t)+vc(t)=vs(t)RCddt[H(jω)ejωt]+H(jω)ejωt=ejωt\begin{aligned} RC\frac{d}{dt}v_c(t)+v_c(t)&=v_s(t)\\ RC\frac{d}{dt}[H(j\omega)e^{j\omega t}]+H(j\omega)e^{j\omega t}&=e^{j\omega t} \end{aligned}

    可得:

    H(jω)=11+RCjωH(jω)=1s1+(RCω)2H(jω)=arctan(RCω)\begin{aligned} H(j\omega)&=\frac{1}{1+RCj\omega} \\ |H(j\omega)|&=\frac{1}{s\sqrt{1+(RC\omega)^2}}\\ \angle H(j\omega)&=-\text{arctan}(RC\omega) \end{aligned}

    RC高通

    # 差分方程描述的离散时间滤波器

    这一部分在我刚学这门课时觉得比较难,但还是记了这些笔记。

    但后面学了 CTFT、DTFT、LT、zT 后运用后面的方法会发现这里的分析其实很简单,不必按照这里的方法分析。

    # 一阶递归离散时间滤波器

    对于由一阶差分方程描述的 LTI 系统

    y[n]ay[n1]=x[n]y[n]-ay[n-1]=x[n]

    输入 x[n]=ejωnx[n] = e^{j\omega n} 将产生输出 y[n]=H(ejω)ejωny[n] = H(e^{j\omega}) e^{j\omega n} ,因此:

    H(ejω)ejωnaH(ejω)ejω(n1)=ejωnH(e^{j\omega}) e^{j\omega n}-a H(e^{j\omega}) e^{jω(n-1)}=e^{j\omega n}

    H(ejω)=11aejωH(e^{j\omega})=\frac{1}{1-ae^{-j\omega}}

    # 非递归离散时间滤波器

    # 移动均值滤波器

    输出信号 = 输入信号的时移和

    # 三点

    三点移动均值滤波器中,输入与输出的关系是:

    y[n]=13(x[n1]+x[n]+x[n+1])y[n]=\frac{1}{3}(x[n-1]+x[n]+x[n+1])

    单位脉冲响应为:

    h[n]=13(δ[n1]+δ[n]+δ[n+1])h[n]=\frac{1}{3}(\delta[n-1]+\delta[n]+\delta[n+1])

    则频率响应为:

    H(ejω)=n=+h[n]ejωn=13[ejω+1+ejω]=13+23cosω\begin{aligned} H(e^{j\omega}) &= \sum_{n=-\infin}^{+\infin}h[n]e^{-j\omega n} \\ &= \frac{1}{3}[e^{-j\omega}+1+e^{j\omega}] \\ &= \frac{1}{3}+\frac{2}{3}\text{cos}\omega \end{aligned}

    # N+M+1 点

    N+M+1 移动均值滤波器中,输入与输出的关系是:

    y[n]=1N+M+1k=NMx[nk]y[n]=\frac{1}{N+M+1}\sum_{k=-N}^{M}x[n-k]

    单位脉冲响应为:

    h[n]=1N+M+1k=NMδ[nk]h[n]=\frac{1}{N+M+1}\sum_{k=-N}^{M}\delta[n-k]

    则频率响应为:

    H(ejω)=1N+M+1n=+k=NMδ[nk]ejωn=1N+M+1k=NMn=+δ[nk]ejωn=1N+M+1k=NMδ[nk]ejωk=1N+M+1ejωNM2sinω(N+M+1)2sin(ω2)\begin{aligned} H(e^{j\omega}) &= \frac{1}{N+M+1}\sum_{n=-\infin}^{+\infin}\sum_{k=-N}^{M}\delta[n-k]e^{-j\omega n} \\ &= \frac{1}{N+M+1}\sum_{k=-N}^{M}\sum_{n=-\infin}^{+\infin}\delta[n-k]e^{-j\omega n} \\ &= \frac{1}{N+M+1}\sum_{k=-N}^{M}\delta[n-k]e^{-j\omega k} \\ &= \frac{1}{N+M+1}e^{j\omega\frac{N-M}{2}}\frac{\text{sin}\frac{\omega(N+M+1)}{2}}{\sin(\frac{\omega}{2})} \\ \end{aligned}

    # 差分滤波器

    这个名字是我乱起的,为了和移动均值滤波器 “对偶” 也好分类

    输出信号 = 输入信号的时移差,也就是输出信号 = 输入信号的差分

    # (两点)

    两点差分滤波器这个名字也是我乱起的

    输入与输出的关系是:

    y[n]=12(x[n]x[n1])y[n]=\frac{1}{2}(x[n]-x[n-1])

    单位脉冲响应为:

    h[n]=12(δ[n]δ[n1])h[n]=\frac{1}{2}(\delta[n]-\delta[n-1])

    则频率响应为:

    H(ejω)=n=+h[n]ejωn=12[1ejω]=je12jωsin(ω2)\begin{aligned} H(e^{j\omega}) &= \sum_{n=-\infin}^{+\infin}h[n]e^{-j\omega n} \\ &= \frac{1}{2}[1-e^{-j\omega}] \\ &= je^{-\frac{1}{2}j\omega}\sin(\frac{\omega}{2}) \end{aligned}

    H(ejω)=sin(ω2)|H(e^{j\omega})|=|\sin(\frac{\omega}{2})|

    两点差分滤波器

    \ No newline at end of file +M1001 80h400000v40h-400000z"/>RCω=arctan(RCω1)

    RC高通

    # 差分方程描述的离散时间滤波器

    这一部分在我刚学这门课时觉得比较难,但还是记了这些笔记。

    但后面学了 CTFT、DTFT、LT、zT 后运用后面的方法会发现这里的分析其实很简单,不必按照这里的方法分析。

    # 一阶递归离散时间滤波器

    对于由一阶差分方程描述的 LTI 系统

    y[n]ay[n1]=x[n]y[n]-ay[n-1]=x[n]

    输入 x[n]=ejωnx[n] = e^{j\omega n} 将产生输出 y[n]=H(ejω)ejωny[n] = H(e^{j\omega}) e^{j\omega n} ,因此:

    H(ejω)ejωnaH(ejω)ejω(n1)=ejωnH(e^{j\omega}) e^{j\omega n}-a H(e^{j\omega}) e^{jω(n-1)}=e^{j\omega n}

    H(ejω)=11aejωH(e^{j\omega})=\frac{1}{1-ae^{-j\omega}}

    # 非递归离散时间滤波器

    # 移动均值滤波器

    输出信号 = 输入信号的时移和

    # 三点

    三点移动均值滤波器中,输入与输出的关系是:

    y[n]=13(x[n1]+x[n]+x[n+1])y[n]=\frac{1}{3}(x[n-1]+x[n]+x[n+1])

    单位脉冲响应为:

    h[n]=13(δ[n1]+δ[n]+δ[n+1])h[n]=\frac{1}{3}(\delta[n-1]+\delta[n]+\delta[n+1])

    则频率响应为:

    H(ejω)=n=+h[n]ejωn=13[ejω+1+ejω]=13+23cosω\begin{aligned} H(e^{j\omega}) &= \sum_{n=-\infin}^{+\infin}h[n]e^{-j\omega n} \\ &= \frac{1}{3}[e^{-j\omega}+1+e^{j\omega}] \\ &= \frac{1}{3}+\frac{2}{3}\text{cos}\omega \end{aligned}

    # N+M+1 点

    N+M+1 移动均值滤波器中,输入与输出的关系是:

    y[n]=1N+M+1k=NMx[nk]y[n]=\frac{1}{N+M+1}\sum_{k=-N}^{M}x[n-k]

    单位脉冲响应为:

    h[n]=1N+M+1k=NMδ[nk]h[n]=\frac{1}{N+M+1}\sum_{k=-N}^{M}\delta[n-k]

    则频率响应为:

    H(ejω)=1N+M+1n=+k=NMδ[nk]ejωn=1N+M+1k=NMn=+δ[nk]ejωn=1N+M+1k=NMδ[nk]ejωk=1N+M+1ejωNM2sinω(N+M+1)2sin(ω2)\begin{aligned} H(e^{j\omega}) &= \frac{1}{N+M+1}\sum_{n=-\infin}^{+\infin}\sum_{k=-N}^{M}\delta[n-k]e^{-j\omega n} \\ &= \frac{1}{N+M+1}\sum_{k=-N}^{M}\sum_{n=-\infin}^{+\infin}\delta[n-k]e^{-j\omega n} \\ &= \frac{1}{N+M+1}\sum_{k=-N}^{M}\delta[n-k]e^{-j\omega k} \\ &= \frac{1}{N+M+1}e^{j\omega\frac{N-M}{2}}\frac{\text{sin}\frac{\omega(N+M+1)}{2}}{\sin(\frac{\omega}{2})} \\ \end{aligned}

    # 差分滤波器

    这个名字是我乱起的,为了和移动均值滤波器 “对偶” 也好分类

    输出信号 = 输入信号的时移差,也就是输出信号 = 输入信号的差分

    # (两点)

    两点差分滤波器这个名字也是我乱起的

    输入与输出的关系是:

    y[n]=12(x[n]x[n1])y[n]=\frac{1}{2}(x[n]-x[n-1])

    单位脉冲响应为:

    h[n]=12(δ[n]δ[n1])h[n]=\frac{1}{2}(\delta[n]-\delta[n-1])

    则频率响应为:

    H(ejω)=n=+h[n]ejωn=12[1ejω]=je12jωsin(ω2)\begin{aligned} H(e^{j\omega}) &= \sum_{n=-\infin}^{+\infin}h[n]e^{-j\omega n} \\ &= \frac{1}{2}[1-e^{-j\omega}] \\ &= je^{-\frac{1}{2}j\omega}\sin(\frac{\omega}{2}) \end{aligned}

    H(ejω)=sin(ω2)|H(e^{j\omega})|=|\sin(\frac{\omega}{2})|

    两点差分滤波器

    \ No newline at end of file diff --git a/ie/signals-and-systems/LT/index.html b/ie/signals-and-systems/LT/index.html index cf4e81ed..4a190056 100644 --- a/ie/signals-and-systems/LT/index.html +++ b/ie/signals-and-systems/LT/index.html @@ -1,4 +1,4 @@ -拉普拉斯变换 - 信号与系统 - 电子信息 | Balloon Party = 風船のパーティー = fuusen no party

    # 公式表

    # 基本变换对

    基本变换对

    # 性质表

    性质表

    # 双边拉普拉斯变换

    # 定义

    X(s)=+x(t)estdtX(s)=\int_{-\infin}^{+\infin}x(t)e^{-st}dt

    称为 x(t)x(t) 的拉普拉斯变换,后面简记为 LT,其中 s=σ+jωs=\sigma+j\omega 为复数。

    σ=0\sigma=0s=jωs=j\omega 为纯虚数则为傅里叶变换,也就是说连续时间傅里叶变换是双边拉普拉斯变换在 𝜎=0 或是在 jωj\omega 轴上的特例,即 X(s)s=jω=X(jω)X(s)|_{s=j\omega}=X(j\omega)

    # LT 与 CTFT 的关系

    X(s)=+x(t)eσtejωtdt=+[x(t)eσt]ejωtdt=F{x(t)eσt}\begin{aligned} X(s)&=\int_{-\infin}^{+\infin}x(t)e^{-\sigma t}e^{-j\omega t}dt\\ &=\int_{-\infin}^{+\infin}[x(t)e^{-\sigma t}]e^{-j\omega t}dt\\ &=\mathscr{F}\{x(t)e^{-\sigma t}\} \end{aligned}

    所以 x(t)x(t) 的 LT 就是 x(t)eσtx(t)e^{-\sigma t} 的 CTFT 。

    # 零极点图

    X(s)X(s) 是有理函数:

    X(s)=N(s)D(s)=Mi(sβi)i(sαi)X(s)=\frac{N(s)}{D(s)}=M\frac{\prod_i(s-\beta_i)}{\prod_i(s-\alpha_i)}

    • 分子多项式的根称为零点,画为圆圈 ⭕️
    • 分母多项式的根称为极点,画为交叉 ❌

    X(s)X(s) 的全部零点极点表示在 ss 平面上,就构成了零极点图

    零极点图及其收敛域可以表示一个 X(s)X(s) ,最多与真实的 X(s)X(s) 相差一个常数因子 𝑀𝑀

    # ROC

    ⚠️不同的信号可能会有完全相同的拉普拉斯变换表达式,只是它们的收敛域不同。拉普拉斯变换的表达式只有连同相应的收敛域,才能和信号建立一一对应的关系。

    常见的 x(t)=ektu(at+b)x(t)=e^{kt}u(at+b) 收敛域:

    X(s)=ba+e(ks)tdt,a>0X(s)=bae(ks)tdt,a<0X(s)=\int_{-\frac{b}{a}}^{+\infin}e^{(k-s)t}dt,\ \ \ a>0\\ X(s)=\int^{-\frac{b}{a}}_{-\infin}e^{(k-s)t}dt,\ \ \ a<0

    tt 积分到 ++\infin 时,tt 前的系数的实部需要小于零,该积分收敛域为 Re{ks}<0\text{Re}\{k-s\}<0

    tt 积分到 -\infin 时,tt 前的系数的实部需要大于零,该积分收敛域为 Re{ks}>0\text{Re}\{k-s\}>0

    并非 ss 平面上的任何复数都能使拉普拉斯变换 X(s)X(s) 收敛,也不是任何信号的拉普拉斯变换 X(s)X(s) 都存在。

    使拉普拉斯变换 X(s)X(s) 收敛的复数 ss 的集合,称为拉普拉斯变换 (LT) 的收敛域ROC,Region of Convergence)

    x(t)x(t) 可以写为若干部分的线性组合,各个部分分别进行 LT 有各自的收敛域,则 X(s)X(s) 的收敛域是各个收敛域的交集

    • ROC 总是以平行于 jωj\omega 轴的直线作为边界的
    • ROC 的边界总是与 X(s)X(s) 的分母的根 (极点) 相对应
    • (因为)有理拉普拉斯变换在其 ROC 内无任何极点
    • 绝对可积的时限信号其 ROC 是整个 ss 平面

    按照 ROC 在 ss 平面上的分布可以分为:

    • 右边信号的 ROC 位于 ss 平面内一条平行于 jωj\omega 轴的直线的右边
    • 左边信号的 ROC 位于 ss 平面内一条平行于 jωj\omega 轴的直线的左边
    • 双边信号的 ROC 如果存在,一定是 ss 平面内平行于 jωj\omega 轴的带形区域

    # 拉普拉斯反变换

    # 定义

    利用 LT 和 CTFT 的关系、CTFT 的反变换可以得到 LT 的反变换。

    (LT 和 CTFT 的关系:)

    X(s)=+x(t)estdtX(σ+jω)=+x(t)eσtejωtdt=F{x(t)eσt}\begin{aligned} X(s)&=\int_{-\infin}^{+\infin}x(t)e^{-st}dt\\ X(\sigma+j\omega)&=\int_{-\infin}^{+\infin}x(t)e^{-\sigma t}e^{-j\omega t}dt\\ &=\mathscr{F}\{x(t)e^{-\sigma t}\} \end{aligned}

    进而由 CTFT 的反变换有:

    x(t)eσt=12π+X(σ+jω)ejωtdωx(t)=12π+X(σ+jω)eσtejωtdωx(t)=12π+X(s)estdω\begin{aligned} x(t)e^{-\sigma t}&=\frac{1}{2\pi}\int_{-\infin}^{+\infin}X(\sigma+j\omega)e^{j\omega t}d\omega\\ x(t)&=\frac{1}{2\pi}\int_{-\infin}^{+\infin}X(\sigma+j\omega)e^{\sigma t}e^{j\omega t}d\omega\\ x(t)&=\frac{1}{2\pi}\int_{-\infin}^{+\infin}X(s)e^{st}d\omega \end{aligned}

    s=σ+jωs=\sigma+j\omega 两侧同时微分得到:ds=jdωds=jd\omega ;同时,当 ω\omega+-\infin\to+\infin 时,ssσjσ+j\sigma-j\infin\to\sigma+j\infin ,所以可得到拉普拉斯反变换为

    x(t)=12πjσjσ+jX(s)estdsx(t)=\frac{1}{2\pi j}\int_{\sigma-j\infin}^{\sigma+j\infin}X(s)e^{st}ds

    # 部分分式展开法

    1. X(s)X(s) 展开为部分分式
    2. 根据 X(s)X(s) 的 ROC ,确定每一项的 ROC
    3. 利用常用信号变换对与拉普拉斯变换性质,对每一项进行反变换

    # LT 的几何求值

    # 单零点

    X(s)=saX(s)=s-a

    有唯一一个零点:s=as=a

    单零点

    s1a\vec{s_1}-\vec{a}

    # 公式表

    # 基本变换对

    基本变换对

    # 性质表

    性质表

    # 双边拉普拉斯变换

    # 定义

    X(s)=+x(t)estdtX(s)=\int_{-\infin}^{+\infin}x(t)e^{-st}dt

    称为 x(t)x(t) 的拉普拉斯变换,后面简记为 LT,其中 s=σ+jωs=\sigma+j\omega 为复数。

    σ=0\sigma=0s=jωs=j\omega 为纯虚数则为傅里叶变换,也就是说连续时间傅里叶变换是双边拉普拉斯变换在 𝜎=0 或是在 jωj\omega 轴上的特例,即 X(s)s=jω=X(jω)X(s)|_{s=j\omega}=X(j\omega)

    # LT 与 CTFT 的关系

    X(s)=+x(t)eσtejωtdt=+[x(t)eσt]ejωtdt=F{x(t)eσt}\begin{aligned} X(s)&=\int_{-\infin}^{+\infin}x(t)e^{-\sigma t}e^{-j\omega t}dt\\ &=\int_{-\infin}^{+\infin}[x(t)e^{-\sigma t}]e^{-j\omega t}dt\\ &=\mathscr{F}\{x(t)e^{-\sigma t}\} \end{aligned}

    所以 x(t)x(t) 的 LT 就是 x(t)eσtx(t)e^{-\sigma t} 的 CTFT 。

    # 零极点图

    X(s)X(s) 是有理函数:

    X(s)=N(s)D(s)=Mi(sβi)i(sαi)X(s)=\frac{N(s)}{D(s)}=M\frac{\prod_i(s-\beta_i)}{\prod_i(s-\alpha_i)}

    • 分子多项式的根称为零点,画为圆圈 ⭕️
    • 分母多项式的根称为极点,画为交叉 ❌

    X(s)X(s) 的全部零点极点表示在 ss 平面上,就构成了零极点图

    零极点图及其收敛域可以表示一个 X(s)X(s) ,最多与真实的 X(s)X(s) 相差一个常数因子 𝑀𝑀

    # ROC

    ⚠️不同的信号可能会有完全相同的拉普拉斯变换表达式,只是它们的收敛域不同。拉普拉斯变换的表达式只有连同相应的收敛域,才能和信号建立一一对应的关系。

    常见的 x(t)=ektu(at+b)x(t)=e^{kt}u(at+b) 收敛域:

    X(s)=ba+e(ks)tdt,a>0X(s)=bae(ks)tdt,a<0X(s)=\int_{-\frac{b}{a}}^{+\infin}e^{(k-s)t}dt,\ \ \ a>0\\ X(s)=\int^{-\frac{b}{a}}_{-\infin}e^{(k-s)t}dt,\ \ \ a<0

    tt 积分到 ++\infin 时,tt 前的系数的实部需要小于零,该积分收敛域为 Re{ks}<0\text{Re}\{k-s\}<0

    tt 积分到 -\infin 时,tt 前的系数的实部需要大于零,该积分收敛域为 Re{ks}>0\text{Re}\{k-s\}>0

    并非 ss 平面上的任何复数都能使拉普拉斯变换 X(s)X(s) 收敛,也不是任何信号的拉普拉斯变换 X(s)X(s) 都存在。

    使拉普拉斯变换 X(s)X(s) 收敛的复数 ss 的集合,称为拉普拉斯变换 (LT) 的收敛域ROC,Region of Convergence)

    x(t)x(t) 可以写为若干部分的线性组合,各个部分分别进行 LT 有各自的收敛域,则 X(s)X(s) 的收敛域是各个收敛域的交集

    • ROC 总是以平行于 jωj\omega 轴的直线作为边界的
    • ROC 的边界总是与 X(s)X(s) 的分母的根 (极点) 相对应
    • (因为)有理拉普拉斯变换在其 ROC 内无任何极点
    • 绝对可积的时限信号其 ROC 是整个 ss 平面

    按照 ROC 在 ss 平面上的分布可以分为:

    • 右边信号的 ROC 位于 ss 平面内一条平行于 jωj\omega 轴的直线的右边
    • 左边信号的 ROC 位于 ss 平面内一条平行于 jωj\omega 轴的直线的左边
    • 双边信号的 ROC 如果存在,一定是 ss 平面内平行于 jωj\omega 轴的带形区域

    # 拉普拉斯反变换

    # 定义

    利用 LT 和 CTFT 的关系、CTFT 的反变换可以得到 LT 的反变换。

    (LT 和 CTFT 的关系:)

    X(s)=+x(t)estdtX(σ+jω)=+x(t)eσtejωtdt=F{x(t)eσt}\begin{aligned} X(s)&=\int_{-\infin}^{+\infin}x(t)e^{-st}dt\\ X(\sigma+j\omega)&=\int_{-\infin}^{+\infin}x(t)e^{-\sigma t}e^{-j\omega t}dt\\ &=\mathscr{F}\{x(t)e^{-\sigma t}\} \end{aligned}

    进而由 CTFT 的反变换有:

    x(t)eσt=12π+X(σ+jω)ejωtdωx(t)=12π+X(σ+jω)eσtejωtdωx(t)=12π+X(s)estdω\begin{aligned} x(t)e^{-\sigma t}&=\frac{1}{2\pi}\int_{-\infin}^{+\infin}X(\sigma+j\omega)e^{j\omega t}d\omega\\ x(t)&=\frac{1}{2\pi}\int_{-\infin}^{+\infin}X(\sigma+j\omega)e^{\sigma t}e^{j\omega t}d\omega\\ x(t)&=\frac{1}{2\pi}\int_{-\infin}^{+\infin}X(s)e^{st}d\omega \end{aligned}

    s=σ+jωs=\sigma+j\omega 两侧同时微分得到:ds=jdωds=jd\omega ;同时,当 ω\omega+-\infin\to+\infin 时,ssσjσ+j\sigma-j\infin\to\sigma+j\infin ,所以可得到拉普拉斯反变换为

    x(t)=12πjσjσ+jX(s)estdsx(t)=\frac{1}{2\pi j}\int_{\sigma-j\infin}^{\sigma+j\infin}X(s)e^{st}ds

    # 部分分式展开法

    1. X(s)X(s) 展开为部分分式
    2. 根据 X(s)X(s) 的 ROC ,确定每一项的 ROC
    3. 利用常用信号变换对与拉普拉斯变换性质,对每一项进行反变换

    # LT 的几何求值

    # 单零点

    X(s)=saX(s)=s-a

    有唯一一个零点:s=as=a

    单零点

    s1a\vec{s_1}-\vec{a}

    从所有零点向 ss 点作零点矢量,从所有极点向 ss 点作极点矢量:

    • 所有零点矢量的长度之积 (不存在则为 1) 除以所有极点矢量的长度之积即为 X(s)|X(s)|
    • 所有零点矢量的幅角之和 (不存在则为 0) 减去所有极点矢量的幅角之和即为 X(s)\angle X(s)

    ss 取为 jωj\omega 轴上的点时,即为傅里叶变换的几何求值。考查 ssjωj\omega 轴上移动时所有零、极点矢量的长度和幅角的变化,即可得出 X(jω)X(j\omega) 的幅频特性和相频特性

    # 全通系统

    全通系统被广泛用于对系统进行相位均衡,它满足:

    • 零点和极点分布关于 jωj\omega 轴对称、成对分布;
    • H(jω)|H(j\omega)| 在任何时候都等于 11

    一般还称具有 nn 对零点和极点的全通系统为 nn 阶全通系统。

    # LT 性质

    LT 具有很多与 FT 相似的重要性质,主要不同是这里需要考虑 ROC

    # 线性

    x1(t)LTX1(s),ROC:R1x2(t)LTX2(s),ROC:R2ax1(t)+bx2(t)LTaX1(s)+bX2(s)x_1(t)\stackrel{LT}{\longleftrightarrow}X_1(s),\ \ \text{ROC}:R_1\\ x_2(t)\stackrel{LT}{\longleftrightarrow}X_2(s),\ \ \text{ROC}:R_2\\ \dArr\\ ax_1(t)+bx_2(t)\stackrel{LT}{\longleftrightarrow}aX_1(s)+bX_2(s)

    后者的 ROC 至少R1R2R_1\cap R_2 ,否则若 R1R2=R_1\cap R_2=\varnothing ,则 ax1(t)+bx2(t)ax_1(t)+bx_2(t) 的 LT 不存在。

    ROC 也有可能扩大:X1(s)X_1(s)X2(s)X_2(s) 线性组合时,若发生了零极点相抵消的现象,且当被抵消的极点恰好在 ROC 的边界上时,就会使 ROC 扩大。

    # 时移特性

    x(t)LTX(s),ROC:Rx(tt0)LTX(s)est0,ROC:Rx(t)\stackrel{LT}{\longleftrightarrow}X(s),\ \ \text{ROC}:R\\ \dArr\\ x(t-t_0)\stackrel{LT}{\longleftrightarrow}X(s)e^{-st_0},\ \ \text{ROC}:R

    # s 域平移

    x(t)LTX(t),ROC:Rx(t)esotLTX(ss0),ROC:R+Re[s0]x(t)\stackrel{LT}{\longleftrightarrow}X(t),\ \ \text{ROC}:R\\ \dArr\\ x(t)e^{s_o t}\stackrel{LT}{\longleftrightarrow}X(s-s_0),\ \ \text{ROC}:R+\text{Re}[s_0]

    ⚠️这里 R+Re[s0]R+\text{Re}[s_0] 的意思是将 RR 平移了一个 Re[s0]\text{Re}[s_0]⚠️,与时域平移不同,ss 域平移的时候收敛域 ROC 也发生了平移。

    • 加法➕是 ROC 向右平移
    • 剑法➖是 ROC 向左平移

    # 时域尺度变换

    若信号在时域尺度变换,其 LT 的 ROC 在 ss 平面上作相反的尺度变换。

    x(t)LTX(s),ROC:Rx(at)LT1aX(sa),ROC:aRx(t)\stackrel{LT}{\longleftrightarrow}X(s),\ \ \text{ROC}:R\\ \dArr\\ x(at)\stackrel{LT}{\longleftrightarrow}\frac{1}{|a|}X(\frac{s}{a}),\ \ \text{ROC}:aR

    ROC 的推导比较简单:sRs\in RX(s)X(s) 收敛,所以 s/aRs/a\in RX(sa)X(\frac{s}{a}) 收敛。

    ⚠️这里 aRaR 的意思是将 RR 线性伸缩了 aa 倍⚠️

    # 共轭对称性

    x(t)LTX(s),ROC:Rx(t)LTX(s),ROC:Rx(t)\stackrel{LT}{\longleftrightarrow}X(s),\ \ \text{ROC}:R\\ \dArr\\ x^*(t)\stackrel{LT}{\longleftrightarrow}X^*(s^*),\ \ \text{ROC}:R

    这里可以结合 X(s)=X(s)|X(s)|=|X(s^*)| 理解,模相同意味着二者总是零点极点相同......

    特别地,当 x(t)x(t)实信号时,有:

    x(t)=x(t)LTX(s)=X(s)=X(s)=X(s)x^*(t)=x(t)\stackrel{LT}{\longleftrightarrow} X^*(s^*)=X(s)=X^*(s)=X(s^*)

    这里 X(s)=X(s)X^*(s^*)=X(s) 等价于 X(s)=X(s)X^*(s)=X(s^*)

    因此,如果 x(t)x(t) 是实信号,且 X(s)X(s)s0s_0 有极点 (或零点),则 X(s)X(s) 一定在 s0s_0^* 也有极点 (或零点)。即实信号的拉普拉斯变换其零、极点必共轭成对出现

    # 卷积性质

    x1(t)LTX1(s),ROC:R1x2(t)LTX2(s),ROC:R2x1(t)x2(t)LTX1(s)X2(s)x_1(t)\stackrel{LT}{\longleftrightarrow}X_1(s),\ \ \text{ROC}:R_1\\ x_2(t)\stackrel{LT}{\longleftrightarrow}X_2(s),\ \ \text{ROC}:R_2\\ \dArr\\ x_1(t)*x_2(t)\stackrel{LT}{\longleftrightarrow}X_1(s)X_2(s)

    类似线性性质中,后者的 ROC 包括 R1R2R_1\cap R_2 ,否则若 R1R2=R_1\cap R_2=\varnothing ,则 x1(t)x2(t)x_1(t)*x_2(t) 的 LT 不存在。

    ROC 也有可能扩大:X1(s)X_1(s)X2(s)X_2(s) 相乘时,若发生了零极点相抵消的现象,且当被抵消的极点恰好在 ROC 的边界上时,就会使 ROC 扩大。

    # 时域微分

    x(t)LTX(s),ROC:Rddtx(t)LTsX(s)x(t)\stackrel{LT}{\longleftrightarrow}X(s),\ \ \text{ROC}:R\\ \dArr\\ \frac{d}{dt}x(t)\stackrel{LT}{\longleftrightarrow}sX(s)

    后者的 ROC 包括 R1R2R_1\cap R_2 ,有可能扩大。

    证明可由 LT 逆变换两侧同时对 tt 微分得到。

    # s 域微分

    x(t)LTX(s),ROC:Rtx(t)LTddsX(s),ROC:Rx(t)\stackrel{LT}{\longleftrightarrow}X(s),\ \ \text{ROC}:R\\ \dArr\\ -tx(t)\stackrel{LT}{\longleftrightarrow}\frac{d}{ds}X(s),\ \ \text{ROC}:R\\

    # 时域积分

    x(t)LTX(s),ROC:Rtx(τ)dτLT1sX(s)x(t)\stackrel{LT}{\longleftrightarrow}X(s),\ \ \text{ROC}:R\\ \dArr\\ \int_{-\infin}^{t}x(\tau)d\tau\stackrel{LT}{\longleftrightarrow}\frac{1}{s}X(s)

    后者的 ROC 包括 R{sRe[s]>0}R\cap\{s|\text{Re}[s]>0\} ,也就是包括 RRssjωj\omega 轴右侧的部分 (ss 平面的右半部分)。这是因为:

    tx(τ)dτ=x(t)u(t)\int_{-\infin}^{t}x(\tau)d\tau=x(t)*u(t)

    # 初值与终值定理

    # 初值定理

    如果 x(t)x(t) 是因果信号,且在 t=0t=0 不包含奇异函数,则有初值定理

    x(0+)=limssX(s)x(0^+)=\lim_{s\to\infin}sX(s)


    证明

    t<0t<0x(t)=0x(t)=0x=0x=0 时不为奇异函数,则有: x(t)=x(t)u(t)x(t)=x(t)u(t)

    x(t)=[x(0+)+x(0+)t+x(0+)t22++x(n)(0+)tnn!]u(t)\begin{aligned} x(t)&=[x(0^+)+x'(0^+)t+x''(0^+)\frac{t^2}{2}+\cdots+x^{(n)}(0^+)\frac{t^n}{n!}]u(t) \end{aligned}

    作 LT 可得:

    X(s)=[1sx(0+)+1s2x(0+)+1s3x(0+)++1sn+1x(n)(0+)]u(t)=n=0+1sn+1x(n)(0+)\begin{aligned} X(s)&=[\frac{1}{s}x(0^+)+\frac{1}{s^2}x'(0^+)+\frac{1}{s^3}x''(0^+)+\cdots+\frac{1}{s^{n+1}}x^{(n)}(0^+)]u(t)\\ &=\sum_{n=0}^{+\infin}\frac{1}{s^{n+1}}\cdot x^{(n)}(0^+) \end{aligned}

    那么:

    sX(s)=sn=0+1sn+1x(n)(0+)=x(0+)+n=1+1snx(n)(0+)limssX(s)=x(0+)\begin{aligned} sX(s)&=s\sum_{n=0}^{+\infin}\frac{1}{s^{n+1}}\cdot x^{(n)}(0^+)\\ &=x(0^+)+\sum_{n=1}^{+\infin}\frac{1}{s^{n}}\cdot x^{(n)}(0^+)\\ \lim_{s\to\infin}sX(s)&=x(0^+) \end{aligned}

    # 终值定理

    如果 x(t)x(t) 是因果信号,且在 t=0t=0 不包含奇异函数,X(s)X(s) 除了在 s=0s=0 可以有单阶极点外,其余极点均在 ss 平面的左半边 (Re[s]<0\text{Re}[s]<0​)

    上述条件的目的在于保证终值存在

    则有终值定理

    limtx(t)=lims0sX(s)\lim_{t\to\infin}x(t)=\lim_{s\to0}sX(s)


    证明

    x(t)x(t) 是因果信号,且在 t=0t=0 不包含奇异函数,则有:

    0+estddtx(t)dt=x(t)est0++s0+estx(t)dt=x(t)est0++sX(s)\begin{aligned} \int_{0^+}^{\infin}e^{-st}\frac{d}{dt}x(t)dt &=x(t)e^{-st}\big|_{0^+}^{\infin}+s\int_{0^+}^{\infin}e^{-st}x(t)dt\\ &=x(t)e^{-st}\big|_{0^+}^{\infin}+sX(s) \end{aligned}

    X(s)X(s) 除了在 s=0s=0 可以有单阶极点外,其余极点均在 ss 平面的左半边 (Re[s]<0\text{Re}[s]<0) ,所以 sX(s)sX(s) 的 ROC 中必包含正实数轴。

    x(t)est0+=x(0+)x(t)e^{-st}\big|_{0^+}^{\infin}=-x(0^+)

    于是:

    0+estddtx(t)dt=x(0+)+sX(s)\begin{aligned} \int_{0^+}^{\infin}e^{-st}\frac{d}{dt}x(t)dt &=-x(0^+)+sX(s) \end{aligned}

    两侧同时令 s0+s\to0^+ ,可得:

    lims0+0+estddtx(t)dt=x(0+)+lims0+sX(s)0+ddtx(t)dt=x(0+)+lims0+sX(s)0+dx(t)=x(0+)+lims0+sX(s)limtx(t)x(0+)=x(0+)+lims0+sX(s)limtx(t)=lims0+sX(s)\begin{aligned} \lim_{s\to0^+}\int_{0^+}^{\infin}e^{-st}\frac{d}{dt}x(t)dt &=-x(0^+)+\lim_{s\to0^+}sX(s)\\ \int_{0^+}^{\infin}\frac{d}{dt}x(t)dt &=-x(0^+)+\lim_{s\to0^+}sX(s)\\ \int_{0^+}^{\infin}dx(t) &=-x(0^+)+\lim_{s\to0^+}sX(s)\\ \lim_{t\to\infin}x(t)-x(0^+) &=-x(0^+)+\lim_{s\to0^+}sX(s)\\ \lim_{t\to\infin}x(t) &=\lim_{s\to0^+}sX(s)\\ \end{aligned}

    # LT 分析 LTI 系统

    以卷积特性为基础,可建立 LTI 系统的拉普拉斯变换分析方法,即:

    Y(s)=X(s)H(s)Y(s)=X(s)\cdot H(s)

    其中 H(s)H(s)h(t)h(t) 的拉普拉斯变换,称为系统函数转移函数传递函数

    这些方法成立的本质原因在于复指数函数 este^{st} 是一切 LTI 系统的特征函数。

    # 特征函数

    x(t)=es0tx(t)=e^{s_0t} 时,响应为:

    y(t)=H(s0)es0ty(t)=H(s_0)e^{s_0t}

    # 因果性

    • 如果 t<0t<0h(t)=0h(t)=0,则系统是因果的;
    • 如果 t>0t>0h(t)=0h(t)=0,则系统是反因果的;

    因果系统的 h(t)h(t) 是右边信号,其 H(s)H(s) 的 ROC 必是最右边极点的右边;当 H(s)H(s) 是有理函数时,逆命题成立。

    • ⚠️但是当 H(s)H(s) 不是有理函数时, H(s)H(s) 的 ROC 是最右边极点的右边不能推出 h(t)h(t) 不是因果的

    反因果系统的 h(t)h(t) 是左边信号,H(s)H(s) 的 ROC 必是最左边极点的左边;当 H(s)H(s) 是有理函数时,逆命题成立。

    # 稳定性

    LTI 系统稳定的充要条件是其 H(s)H(s) 收敛域包括 jωj\omega 轴。

    因果稳定系统的 H(s)H(s) 收敛域为包括 jωj\omega 轴的 ss 右半平面。

    # 线性常微分方程

    * 线性常系数微分方程

    如果由线性常系数微分方程描述的系统满足初始松弛条件,则系统是因果 LTI 的,其 H(s)H(s) 的 ROC 必是最右边极点的右边。

    对于由线性常系数微分方程描述的 LTI 系统:

    k=0Nakdky(t)dtk=k=0Mbkdkx(t)dtk\sum_{k=0}^{N}a_k\frac{d^ky(t)}{dt^k}=\sum_{k=0}^{M}b_k\frac{d^kx(t)}{dt^k}

    两边进行拉普拉斯变换,可得:

    H(s)=Y(s)X(s)=k=0Mbkskk=0NakskH(s)={Y(s)\over X(s)}={\sum_{k=0}^{M}b_{k}s^{k}\over\sum_{k=0}^{N}a_{k}s^{k}}

    可得到 H(s)H(s) 是一个有理函数。系统的单位冲激响应 h(t)h(t) 可由 H(s)H(s) 反变换获得。

    # 单边拉普拉斯变换

    Unilateral Laplace Transform

    单边拉普拉斯变换是仅考虑 00^- 之后部分信号的双边拉普拉斯变换:

    χ(s)=0+x(t)estdt\chi(s)=\int_{0^-}^{+\infin}x(t)e^{-st}dt

    单边拉普拉斯变换的反变换与同 ROC 的双边拉普拉斯变换的反变换相同:

    x(t)=12πjσjσ+jχ(s)estdsx(t)=\frac{1}{2\pi j}\int_{\sigma-j\infin}^{\sigma+j\infin}\chi(s)e^{st}ds

    X(s)X(s)χ(s)\chi(s) 不同,是因为 x(t)x(t)t<0t<0 的部分对 X(s)X(s) 有作用,而对 χ(s)\chi(s) 没有任何作用所致。

    由于单边拉普拉斯变换不存在类似双边拉普拉斯变换中与 ROC 相关的多义性,一般不再强调其 ROC ,任何单边拉普拉斯变换的收敛域总是某一右半平面。

    单边拉普拉斯变换的 ROC 位于最右边极点右边。

    因果信号的双边拉普拉斯变换和单边拉普拉斯变换完全相同。⭐️从而可以利用双边 LT 的基本变换对。

    # 性质

    由于单边拉普拉斯变换是特殊的双边拉普拉斯变换,因此其大部分性质与双边拉普拉斯变换相同,但也有一些区别:

    # 共轭对称性

    x(t)ULTχ(s)x(t)ULTχ(s)x(t)\stackrel{ULT}{\longleftrightarrow}\chi(s)\\ \dArr\\ x^*(t)\stackrel{ULT}{\longleftrightarrow}\chi^*(s)

    # 时域尺度变换

    x(at)x(at) 中的 aa 需要满足 a>0a\gt0

    x(t)ULTχ(s)x(at)ULT1aχ(sa)x(t)\stackrel{ULT}{\longleftrightarrow}\chi(s)\\ \dArr\\ x(at)\stackrel{ULT}{\longleftrightarrow}\frac{1}{a}\chi(\frac{s}{a})

    # 时域微分

    x(t)ULTχ(s)ddtx(t)ULTsχ(s)x(0)d2dt2x(t)ULTs2χ(s)sx(0)ddtx(0)d3dt3x(t)ULTs3χ(s)s2x(0)sddtx(0)d2dt2x(0)......x(n)(t)ULTsnχ(s)sn1x(0)sn2x(0)sn3x(0)x(n1)(0)x(t)\stackrel{ULT}{\longleftrightarrow}\chi(s)\\ \dArr\\ \frac{d}{dt}x(t)\stackrel{ULT}{\longleftrightarrow}s\chi(s)-x(0^-)\\ \frac{d^2}{dt^2}x(t)\stackrel{ULT}{\longleftrightarrow}s^2\chi(s)-sx(0^-)-\frac{d}{dt}x(0^-)\\ \frac{d^3}{dt^3}x(t)\stackrel{ULT}{\longleftrightarrow}s^3\chi(s)-s^2x(0^-)-s\frac{d}{dt}x(0^-)-\frac{d^2}{dt^2}x(0^-)\\ ......\\ x^{(n)}(t)\stackrel{ULT}{\longleftrightarrow}s^{n}\chi(s)-s^{n-1}x(0^-)-s^{n-2}x'(0^-)-s^{n-3}x''(0^-)-\cdots-x^{(n-1)}(0^-)

    # 线性常微分方程

    单边拉普拉斯变换特别适合于求解具有非零初始条件的线性常系数微分方程(增量线性系统)。

    往往假定系统为因果系统。该条件下得到的结果在进行单边拉普拉斯反变换的时候可以确定为右边信号

    以二阶为例(最高二阶微分),将初始条件带入后,整理得到如下形式:

    Aγ(s)=By(0)+Cy(0)+Dχ(s)γ(s)=BAy(0)+CAy(0)+DAχ(s)\begin{aligned} A\gamma(s)&=By(0^-)+Cy'(0^-)+D\chi(s)\\ \gamma(s)&=\frac{B}{A}y(0^-)+\frac{C}{A}y'(0^-)+\frac{D}{A}\chi(s) \end{aligned}

    其中 DAχ(s)\frac{D}{A}\chi(s)零状态响应BAy(0)+CAy(0)\frac{B}{A}y(0^-)+\frac{C}{A}y'(0^-)零输入响应

    γ(s)=k01s+k11s+1+k21s+2+\begin{aligned} \gamma(s)=k_0\frac{1}{s}+k_1\frac{1}{s+1}+k_2\frac{1}{s+2}+\cdots \end{aligned}

    其中 k01sk_0\frac{1}{s}强迫响应k0k_0 为常数,也就是对应的单边拉普拉斯反变换为 k0u(t)k_0u(t) 形式的才是强迫响应;;k11s+1+k21s+2+k_1\frac{1}{s+1}+k_2\frac{1}{s+2}+\cdots自然响应

    \ No newline at end of file +c-16-25.333-24-45-24-59z"/>)

    从所有零点向 ss 点作零点矢量,从所有极点向 ss 点作极点矢量:

    • 所有零点矢量的长度之积 (不存在则为 1) 除以所有极点矢量的长度之积即为 X(s)|X(s)|
    • 所有零点矢量的幅角之和 (不存在则为 0) 减去所有极点矢量的幅角之和即为 X(s)\angle X(s)

    ss 取为 jωj\omega 轴上的点时,即为傅里叶变换的几何求值。考查 ssjωj\omega 轴上移动时所有零、极点矢量的长度和幅角的变化,即可得出 X(jω)X(j\omega) 的幅频特性和相频特性

    # 全通系统

    全通系统被广泛用于对系统进行相位均衡,它满足:

    • 零点和极点分布关于 jωj\omega 轴对称、成对分布;
    • H(jω)|H(j\omega)| 在任何时候都等于 11

    一般还称具有 nn 对零点和极点的全通系统为 nn 阶全通系统。

    # LT 性质

    LT 具有很多与 FT 相似的重要性质,主要不同是这里需要考虑 ROC

    # 线性

    x1(t)LTX1(s),ROC:R1x2(t)LTX2(s),ROC:R2ax1(t)+bx2(t)LTaX1(s)+bX2(s)x_1(t)\stackrel{LT}{\longleftrightarrow}X_1(s),\ \ \text{ROC}:R_1\\ x_2(t)\stackrel{LT}{\longleftrightarrow}X_2(s),\ \ \text{ROC}:R_2\\ \dArr\\ ax_1(t)+bx_2(t)\stackrel{LT}{\longleftrightarrow}aX_1(s)+bX_2(s)

    后者的 ROC 至少R1R2R_1\cap R_2 ,否则若 R1R2=R_1\cap R_2=\varnothing ,则 ax1(t)+bx2(t)ax_1(t)+bx_2(t) 的 LT 不存在。

    ROC 也有可能扩大:X1(s)X_1(s)X2(s)X_2(s) 线性组合时,若发生了零极点相抵消的现象,且当被抵消的极点恰好在 ROC 的边界上时,就会使 ROC 扩大。

    # 时移特性

    x(t)LTX(s),ROC:Rx(tt0)LTX(s)est0,ROC:Rx(t)\stackrel{LT}{\longleftrightarrow}X(s),\ \ \text{ROC}:R\\ \dArr\\ x(t-t_0)\stackrel{LT}{\longleftrightarrow}X(s)e^{-st_0},\ \ \text{ROC}:R

    # s 域平移

    x(t)LTX(t),ROC:Rx(t)esotLTX(ss0),ROC:R+Re[s0]x(t)\stackrel{LT}{\longleftrightarrow}X(t),\ \ \text{ROC}:R\\ \dArr\\ x(t)e^{s_o t}\stackrel{LT}{\longleftrightarrow}X(s-s_0),\ \ \text{ROC}:R+\text{Re}[s_0]

    ⚠️这里 R+Re[s0]R+\text{Re}[s_0] 的意思是将 RR 平移了一个 Re[s0]\text{Re}[s_0]⚠️,与时域平移不同,ss 域平移的时候收敛域 ROC 也发生了平移。

    • 加法➕是 ROC 向右平移
    • 剑法➖是 ROC 向左平移

    # 时域尺度变换

    若信号在时域尺度变换,其 LT 的 ROC 在 ss 平面上作相反的尺度变换。

    x(t)LTX(s),ROC:Rx(at)LT1aX(sa),ROC:aRx(t)\stackrel{LT}{\longleftrightarrow}X(s),\ \ \text{ROC}:R\\ \dArr\\ x(at)\stackrel{LT}{\longleftrightarrow}\frac{1}{|a|}X(\frac{s}{a}),\ \ \text{ROC}:aR

    ROC 的推导比较简单:sRs\in RX(s)X(s) 收敛,所以 s/aRs/a\in RX(sa)X(\frac{s}{a}) 收敛。

    ⚠️这里 aRaR 的意思是将 RR 线性伸缩了 aa 倍⚠️

    # 共轭对称性

    x(t)LTX(s),ROC:Rx(t)LTX(s),ROC:Rx(t)\stackrel{LT}{\longleftrightarrow}X(s),\ \ \text{ROC}:R\\ \dArr\\ x^*(t)\stackrel{LT}{\longleftrightarrow}X^*(s^*),\ \ \text{ROC}:R

    这里可以结合 X(s)=X(s)|X(s)|=|X(s^*)| 理解,模相同意味着二者总是零点极点相同......

    特别地,当 x(t)x(t)实信号时,有:

    x(t)=x(t)LTX(s)=X(s)=X(s)=X(s)x^*(t)=x(t)\stackrel{LT}{\longleftrightarrow} X^*(s^*)=X(s)=X^*(s)=X(s^*)

    这里 X(s)=X(s)X^*(s^*)=X(s) 等价于 X(s)=X(s)X^*(s)=X(s^*)

    因此,如果 x(t)x(t) 是实信号,且 X(s)X(s)s0s_0 有极点 (或零点),则 X(s)X(s) 一定在 s0s_0^* 也有极点 (或零点)。即实信号的拉普拉斯变换其零、极点必共轭成对出现

    # 卷积性质

    x1(t)LTX1(s),ROC:R1x2(t)LTX2(s),ROC:R2x1(t)x2(t)LTX1(s)X2(s)x_1(t)\stackrel{LT}{\longleftrightarrow}X_1(s),\ \ \text{ROC}:R_1\\ x_2(t)\stackrel{LT}{\longleftrightarrow}X_2(s),\ \ \text{ROC}:R_2\\ \dArr\\ x_1(t)*x_2(t)\stackrel{LT}{\longleftrightarrow}X_1(s)X_2(s)

    类似线性性质中,后者的 ROC 包括 R1R2R_1\cap R_2 ,否则若 R1R2=R_1\cap R_2=\varnothing ,则 x1(t)x2(t)x_1(t)*x_2(t) 的 LT 不存在。

    ROC 也有可能扩大:X1(s)X_1(s)X2(s)X_2(s) 相乘时,若发生了零极点相抵消的现象,且当被抵消的极点恰好在 ROC 的边界上时,就会使 ROC 扩大。

    # 时域微分

    x(t)LTX(s),ROC:Rddtx(t)LTsX(s)x(t)\stackrel{LT}{\longleftrightarrow}X(s),\ \ \text{ROC}:R\\ \dArr\\ \frac{d}{dt}x(t)\stackrel{LT}{\longleftrightarrow}sX(s)

    后者的 ROC 包括 R1R2R_1\cap R_2 ,有可能扩大。

    证明可由 LT 逆变换两侧同时对 tt 微分得到。

    # s 域微分

    x(t)LTX(s),ROC:Rtx(t)LTddsX(s),ROC:Rx(t)\stackrel{LT}{\longleftrightarrow}X(s),\ \ \text{ROC}:R\\ \dArr\\ -tx(t)\stackrel{LT}{\longleftrightarrow}\frac{d}{ds}X(s),\ \ \text{ROC}:R\\

    # 时域积分

    x(t)LTX(s),ROC:Rtx(τ)dτLT1sX(s)x(t)\stackrel{LT}{\longleftrightarrow}X(s),\ \ \text{ROC}:R\\ \dArr\\ \int_{-\infin}^{t}x(\tau)d\tau\stackrel{LT}{\longleftrightarrow}\frac{1}{s}X(s)

    后者的 ROC 包括 R{sRe[s]>0}R\cap\{s|\text{Re}[s]>0\} ,也就是包括 RRssjωj\omega 轴右侧的部分 (ss 平面的右半部分)。这是因为:

    tx(τ)dτ=x(t)u(t)\int_{-\infin}^{t}x(\tau)d\tau=x(t)*u(t)

    # 初值与终值定理

    # 初值定理

    如果 x(t)x(t) 是因果信号,且在 t=0t=0 不包含奇异函数,则有初值定理

    x(0+)=limssX(s)x(0^+)=\lim_{s\to\infin}sX(s)


    证明

    t<0t<0x(t)=0x(t)=0x=0x=0 时不为奇异函数,则有: x(t)=x(t)u(t)x(t)=x(t)u(t)

    x(t)=[x(0+)+x(0+)t+x(0+)t22++x(n)(0+)tnn!]u(t)\begin{aligned} x(t)&=[x(0^+)+x'(0^+)t+x''(0^+)\frac{t^2}{2}+\cdots+x^{(n)}(0^+)\frac{t^n}{n!}]u(t) \end{aligned}

    作 LT 可得:

    X(s)=[1sx(0+)+1s2x(0+)+1s3x(0+)++1sn+1x(n)(0+)]u(t)=n=0+1sn+1x(n)(0+)\begin{aligned} X(s)&=[\frac{1}{s}x(0^+)+\frac{1}{s^2}x'(0^+)+\frac{1}{s^3}x''(0^+)+\cdots+\frac{1}{s^{n+1}}x^{(n)}(0^+)]u(t)\\ &=\sum_{n=0}^{+\infin}\frac{1}{s^{n+1}}\cdot x^{(n)}(0^+) \end{aligned}

    那么:

    sX(s)=sn=0+1sn+1x(n)(0+)=x(0+)+n=1+1snx(n)(0+)limssX(s)=x(0+)\begin{aligned} sX(s)&=s\sum_{n=0}^{+\infin}\frac{1}{s^{n+1}}\cdot x^{(n)}(0^+)\\ &=x(0^+)+\sum_{n=1}^{+\infin}\frac{1}{s^{n}}\cdot x^{(n)}(0^+)\\ \lim_{s\to\infin}sX(s)&=x(0^+) \end{aligned}

    # 终值定理

    如果 x(t)x(t) 是因果信号,且在 t=0t=0 不包含奇异函数,X(s)X(s) 除了在 s=0s=0 可以有单阶极点外,其余极点均在 ss 平面的左半边 (Re[s]<0\text{Re}[s]<0​)

    上述条件的目的在于保证终值存在

    则有终值定理

    limtx(t)=lims0sX(s)\lim_{t\to\infin}x(t)=\lim_{s\to0}sX(s)


    证明

    x(t)x(t) 是因果信号,且在 t=0t=0 不包含奇异函数,则有:

    0+estddtx(t)dt=x(t)est0++s0+estx(t)dt=x(t)est0++sX(s)\begin{aligned} \int_{0^+}^{\infin}e^{-st}\frac{d}{dt}x(t)dt &=x(t)e^{-st}\big|_{0^+}^{\infin}+s\int_{0^+}^{\infin}e^{-st}x(t)dt\\ &=x(t)e^{-st}\big|_{0^+}^{\infin}+sX(s) \end{aligned}

    X(s)X(s) 除了在 s=0s=0 可以有单阶极点外,其余极点均在 ss 平面的左半边 (Re[s]<0\text{Re}[s]<0) ,所以 sX(s)sX(s) 的 ROC 中必包含正实数轴。

    x(t)est0+=x(0+)x(t)e^{-st}\big|_{0^+}^{\infin}=-x(0^+)

    于是:

    0+estddtx(t)dt=x(0+)+sX(s)\begin{aligned} \int_{0^+}^{\infin}e^{-st}\frac{d}{dt}x(t)dt &=-x(0^+)+sX(s) \end{aligned}

    两侧同时令 s0+s\to0^+ ,可得:

    lims0+0+estddtx(t)dt=x(0+)+lims0+sX(s)0+ddtx(t)dt=x(0+)+lims0+sX(s)0+dx(t)=x(0+)+lims0+sX(s)limtx(t)x(0+)=x(0+)+lims0+sX(s)limtx(t)=lims0+sX(s)\begin{aligned} \lim_{s\to0^+}\int_{0^+}^{\infin}e^{-st}\frac{d}{dt}x(t)dt &=-x(0^+)+\lim_{s\to0^+}sX(s)\\ \int_{0^+}^{\infin}\frac{d}{dt}x(t)dt &=-x(0^+)+\lim_{s\to0^+}sX(s)\\ \int_{0^+}^{\infin}dx(t) &=-x(0^+)+\lim_{s\to0^+}sX(s)\\ \lim_{t\to\infin}x(t)-x(0^+) &=-x(0^+)+\lim_{s\to0^+}sX(s)\\ \lim_{t\to\infin}x(t) &=\lim_{s\to0^+}sX(s)\\ \end{aligned}

    # LT 分析 LTI 系统

    以卷积特性为基础,可建立 LTI 系统的拉普拉斯变换分析方法,即:

    Y(s)=X(s)H(s)Y(s)=X(s)\cdot H(s)

    其中 H(s)H(s)h(t)h(t) 的拉普拉斯变换,称为系统函数转移函数传递函数

    这些方法成立的本质原因在于复指数函数 este^{st} 是一切 LTI 系统的特征函数。

    # 特征函数

    x(t)=es0tx(t)=e^{s_0t} 时,响应为:

    y(t)=H(s0)es0ty(t)=H(s_0)e^{s_0t}

    # 因果性

    • 如果 t<0t<0h(t)=0h(t)=0,则系统是因果的;
    • 如果 t>0t>0h(t)=0h(t)=0,则系统是反因果的;

    因果系统的 h(t)h(t) 是右边信号,其 H(s)H(s) 的 ROC 必是最右边极点的右边;当 H(s)H(s) 是有理函数时,逆命题成立。

    • ⚠️但是当 H(s)H(s) 不是有理函数时, H(s)H(s) 的 ROC 是最右边极点的右边不能推出 h(t)h(t) 不是因果的

    反因果系统的 h(t)h(t) 是左边信号,H(s)H(s) 的 ROC 必是最左边极点的左边;当 H(s)H(s) 是有理函数时,逆命题成立。

    # 稳定性

    LTI 系统稳定的充要条件是其 H(s)H(s) 收敛域包括 jωj\omega 轴。

    因果稳定系统的 H(s)H(s) 收敛域为包括 jωj\omega 轴的 ss 右半平面。

    # 线性常微分方程

    * 线性常系数微分方程

    如果由线性常系数微分方程描述的系统满足初始松弛条件,则系统是因果 LTI 的,其 H(s)H(s) 的 ROC 必是最右边极点的右边。

    对于由线性常系数微分方程描述的 LTI 系统:

    k=0Nakdky(t)dtk=k=0Mbkdkx(t)dtk\sum_{k=0}^{N}a_k\frac{d^ky(t)}{dt^k}=\sum_{k=0}^{M}b_k\frac{d^kx(t)}{dt^k}

    两边进行拉普拉斯变换,可得:

    H(s)=Y(s)X(s)=k=0Mbkskk=0NakskH(s)={Y(s)\over X(s)}={\sum_{k=0}^{M}b_{k}s^{k}\over\sum_{k=0}^{N}a_{k}s^{k}}

    可得到 H(s)H(s) 是一个有理函数。系统的单位冲激响应 h(t)h(t) 可由 H(s)H(s) 反变换获得。

    # 单边拉普拉斯变换

    Unilateral Laplace Transform

    单边拉普拉斯变换是仅考虑 00^- 之后部分信号的双边拉普拉斯变换:

    χ(s)=0+x(t)estdt\chi(s)=\int_{0^-}^{+\infin}x(t)e^{-st}dt

    单边拉普拉斯变换的反变换与同 ROC 的双边拉普拉斯变换的反变换相同:

    x(t)=12πjσjσ+jχ(s)estdsx(t)=\frac{1}{2\pi j}\int_{\sigma-j\infin}^{\sigma+j\infin}\chi(s)e^{st}ds

    X(s)X(s)χ(s)\chi(s) 不同,是因为 x(t)x(t)t<0t<0 的部分对 X(s)X(s) 有作用,而对 χ(s)\chi(s) 没有任何作用所致。

    由于单边拉普拉斯变换不存在类似双边拉普拉斯变换中与 ROC 相关的多义性,一般不再强调其 ROC ,任何单边拉普拉斯变换的收敛域总是某一右半平面。

    单边拉普拉斯变换的 ROC 位于最右边极点右边。

    因果信号的双边拉普拉斯变换和单边拉普拉斯变换完全相同。⭐️从而可以利用双边 LT 的基本变换对。

    # 性质

    由于单边拉普拉斯变换是特殊的双边拉普拉斯变换,因此其大部分性质与双边拉普拉斯变换相同,但也有一些区别:

    # 共轭对称性

    x(t)ULTχ(s)x(t)ULTχ(s)x(t)\stackrel{ULT}{\longleftrightarrow}\chi(s)\\ \dArr\\ x^*(t)\stackrel{ULT}{\longleftrightarrow}\chi^*(s)

    # 时域尺度变换

    x(at)x(at) 中的 aa 需要满足 a>0a\gt0

    x(t)ULTχ(s)x(at)ULT1aχ(sa)x(t)\stackrel{ULT}{\longleftrightarrow}\chi(s)\\ \dArr\\ x(at)\stackrel{ULT}{\longleftrightarrow}\frac{1}{a}\chi(\frac{s}{a})

    # 时域微分

    x(t)ULTχ(s)ddtx(t)ULTsχ(s)x(0)d2dt2x(t)ULTs2χ(s)sx(0)ddtx(0)d3dt3x(t)ULTs3χ(s)s2x(0)sddtx(0)d2dt2x(0)......x(n)(t)ULTsnχ(s)sn1x(0)sn2x(0)sn3x(0)x(n1)(0)x(t)\stackrel{ULT}{\longleftrightarrow}\chi(s)\\ \dArr\\ \frac{d}{dt}x(t)\stackrel{ULT}{\longleftrightarrow}s\chi(s)-x(0^-)\\ \frac{d^2}{dt^2}x(t)\stackrel{ULT}{\longleftrightarrow}s^2\chi(s)-sx(0^-)-\frac{d}{dt}x(0^-)\\ \frac{d^3}{dt^3}x(t)\stackrel{ULT}{\longleftrightarrow}s^3\chi(s)-s^2x(0^-)-s\frac{d}{dt}x(0^-)-\frac{d^2}{dt^2}x(0^-)\\ ......\\ x^{(n)}(t)\stackrel{ULT}{\longleftrightarrow}s^{n}\chi(s)-s^{n-1}x(0^-)-s^{n-2}x'(0^-)-s^{n-3}x''(0^-)-\cdots-x^{(n-1)}(0^-)

    # 线性常微分方程

    单边拉普拉斯变换特别适合于求解具有非零初始条件的线性常系数微分方程(增量线性系统)。

    往往假定系统为因果系统。该条件下得到的结果在进行单边拉普拉斯反变换的时候可以确定为右边信号

    以二阶为例(最高二阶微分),将初始条件带入后,整理得到如下形式:

    Aγ(s)=By(0)+Cy(0)+Dχ(s)γ(s)=BAy(0)+CAy(0)+DAχ(s)\begin{aligned} A\gamma(s)&=By(0^-)+Cy'(0^-)+D\chi(s)\\ \gamma(s)&=\frac{B}{A}y(0^-)+\frac{C}{A}y'(0^-)+\frac{D}{A}\chi(s) \end{aligned}

    其中 DAχ(s)\frac{D}{A}\chi(s)零状态响应BAy(0)+CAy(0)\frac{B}{A}y(0^-)+\frac{C}{A}y'(0^-)零输入响应

    γ(s)=k01s+k11s+1+k21s+2+\begin{aligned} \gamma(s)=k_0\frac{1}{s}+k_1\frac{1}{s+1}+k_2\frac{1}{s+2}+\cdots \end{aligned}

    其中 k01sk_0\frac{1}{s}强迫响应k0k_0 为常数,也就是对应的单边拉普拉斯反变换为 k0u(t)k_0u(t) 形式的才是强迫响应;;k11s+1+k21s+2+k_1\frac{1}{s+1}+k_2\frac{1}{s+2}+\cdots自然响应

    \ No newline at end of file diff --git a/ie/signals-and-systems/communication/index.html b/ie/signals-and-systems/communication/index.html index bfc64571..fde3f15b 100644 --- a/ie/signals-and-systems/communication/index.html +++ b/ie/signals-and-systems/communication/index.html @@ -1 +1 @@ -通信系统 - 信号与系统 - 电子信息 | Balloon Party = 風船のパーティー = fuusen no party

    there are still some bugs rendering katex here...

    LTI 系统对输入信号所起的作用包括两个方面:

    • 改变输入信号各频率分量的幅度
    • 改变输入信号各频率分量的相对相位

    Y(jω)=X(jω)H(jω)Y(jω)=X(jω)H(jω)Y(jω)=X(jω)+H(jω)\begin{aligned} Y(j\omega)&=X(j\omega)H(j\omega)\\ |Y(j\omega)|&=|X(j\omega)|\cdot|H(j\omega)|\\ \angle Y(j\omega)&=\angle X(j\omega)+\angle H(j\omega) \end{aligned}

    线性与非线性相位

    信号在传输过程中,相位特性或幅度特性发生改变都可能引起信号波形的改变,即发生失真

    当相位特性仅仅是产生一个线性相移时,只引起信号在时间上的平移。如连续时间 LTI 系统,信号的时移会导致其频谱产生一个线性相移

    x(t)FX(jω)x(tt0)FX(jω)ejωt0x(t)\stackrel{F}{\longleftrightarrow}X(j\omega)\\ \dArr\\ x(t-t_0)\stackrel{F}{\longleftrightarrow}X(j\omega)e^{-j\omega t_0}

    此时信号波形不发生改变,只是发生时间上的延迟,因此未丢失信号所携带的任何信息。

    如果系统的相位特性是非线性的,由于不同频率分量所产生的时移不同,叠加起来会变成一个与原来信号不相同的信号波形。

    系统响应与输入信号满足:

    y(t)=kx(tt0)y[n]=kx[nn0]y(t)=kx(t-t_0)\ \ \ y[n]=kx[n-n_0]

    也就是:

    则可视为在传输中未发生失真

    H(jω)=k,H(jω)=ωt0|H(j\omega)|=k,\ \ \angle H(j\omega)=-\omega t_0

    在传输信号的带宽范围内

    理想滤波器

    • 非因果系统,因而是物理不可实现
    • h(t)h(t) 存在震荡

    # 采样

    # 采样的概念

    在某些离散的时间点上提取连续时间信号值的过程就是采样(sampling)

    与之相对应的,从离散时间信号等间隔地提取信号,则叫做抽取(decimation)

    一般情况下,从连续时间信号采样所得到的样本序列不能唯一地代表原来的连续时间信号,决定是否能从采样结果恢复原信号的关键在于采样定理

    # 采样的数学模型

    采样的数学模型

    在时域:

    xp(t)=x(t)p(t)x_p(t)=x(t)\cdot p(t)

    在频域:

    Xp(jω)=12πX(jω)P(jω)X_p(j\omega)=\frac{1}{2\pi}X(j\omega)*P(j\omega)

    # 冲激串采样 (理想采样)

    TT 为采样间隔的脉冲采样信号为:

    p(t)=n=+δ(tnT)p(t)=\sum_{n=-\infin}^{+\infin}\delta(t-nT)

    • p(t)p(t)采样函数
    • p(t)p(t) 的基波周期 TT采样周期
    • p(t)p(t) 的基波频率 ωs=2πT\omega_s=\frac{2\pi}{T}采样频率

    采样结果为:

    xp(t)=x(t)p(t)=n=+x(nT)δ(tnT)x_p(t)=x(t)\cdot p(t)=\sum_{n=-\infin}^{+\infin}x(nT)\delta(t-nT)

    这样的采样就是理想采样,也就是冲激串采样

    理想采样

    由 CTFT 在频域有:

    p(t)CTFTP(jω)=2πTn=+δ(ω2πTk)p(t)\stackrel{CTFT}{\longleftrightarrow}P(j\omega)=\frac{2\pi}{T}\sum_{n=-\infin}^{+\infin}\delta(\omega-\frac{2\pi}{T}k)

    从而可得到 Xp(jω)X_p(j\omega)

    Xp(jω)=12πX(jω)P(jω)=12πX(jω)2πTn=+δ(ω2πTk)=1Tn=+X(j(ω2πTk))=1Tn=+X(j(ωkωs))\begin{aligned} X_p(j\omega) &=\frac{1}{2\pi}X(j\omega)*P(j\omega)\\ &=\frac{1}{2\pi}X(j\omega)*\frac{2\pi}{T}\sum_{n=-\infin}^{+\infin}\delta(\omega-\frac{2\pi}{T}k)\\ &=\frac{1}{T}\sum_{n=-\infin}^{+\infin}X(j(\omega-\frac{2\pi}{T}k))\\ &=\frac{1}{T}\sum_{n=-\infin}^{+\infin}X(j(\omega-k\omega_s)) \end{aligned}

    可见,在时域对连续时间信号进行理想采样,就相当于在频域将连续时间信号的频谱以 ωs=2πT\omega_s=\frac{2\pi}{T} 为周期进行延拓。

    # 采样定理

    要想使采样后的信号样本能完全代表原来的信号,就意味着要能够从 X𝑝(jω)X_𝑝(j\omega)不失真地分离X(jω)X(j\omega) 。这就要求 X𝑝(jω)X_𝑝(j\omega) 在周期性延拓时不能发生频谱的混叠,因此:

    1. x(t)x(t) 必须是带限的,设最高频率分量为 ωM\omega_M
    2. 采样间隔(周期)不能是任意的,必须保持采样频率 ωs=2πT>2ωM\omega_s=\frac{2\pi}{T}\gt 2\omega_M
    3. 采样频率也不能过大,不能导致周期延拓后频谱发生混叠(补充)

    采样频率的下限 2ωM2\omega_M 也叫做奈奎斯特率(Nyquist rate)

    在满足上述条件时,就可以通过理想低通滤波器从 X𝑝(jω)X_𝑝(j\omega) 中不失真地分离出 X(jω)X(j\omega)

    # 理想低通滤波器

    低通滤波器频域

    若要满足采样定理,低通滤波器截止频率 ωc\omega_c 必须满足:ωM<ωc<ωsωM\omega_M\lt\omega_c\lt\omega_s-\omega_M

    为补偿采样导致的频谱幅度减小,滤波器在通带内应具有 TT 倍增益。

    # Nyquist 采样定理

    对带限于最高频率 ω𝑀\omega_𝑀 的连续时间信号 x(t)x(t) ,如果以 ωs>2ω𝑀\omega_s>2\omega_𝑀 的频率进行理想采样,则 x(t)x(t) 可以唯一地由其样本 x(nT)x(nT) 来确定。

    这里的 TT 也是采样周期,对应采样频率 ωs\omega_s

    # 零阶保持采样

    零阶保持采样相当于理想采样后,再级联一个零阶保持系统,是最简单的一种恢复重建采样前原信号的方法。

    如果由零阶保持所给出的粗糙内插令人不够满意,则可以使用各种更为平滑的內插手段,其中的一些合起来统称为高阶保持 (higher order hold)

    零阶保持采样

    零阶保持采样结果

    为了能从 x0(t)x_0(t) 恢复 x(t)x(t) ,零阶保持后要再级(串,卷积)联一个系统 H𝑟(jω)H_𝑟(j\omega) ,使得:

    H0(jω)Hr(jω)=H(jω)={T,ω<ωc0,ω>ωcH_0(j\omega)H_r(j\omega)=H(j\omega)= \begin{cases} T,&|\omega|\lt\omega_c\\ 0,&|\omega|\gt\omega_c \end{cases}

    其中 ωM<ωc<ωsωM\omega_M\lt\omega_c\lt\omega_s-\omega_M ;图像如下:

    零阶保持采样最终输出

    零阶保持系统的单位冲激响应 h0(t)h_0(t) 是矩形方波信号(经过了一定的时移),利用基本 CTFT 变换对和时移特性可得到:

    H0(jω)=2ωsin(ω2T)ejω2TH_0(j\omega)=\frac{2}{\omega}\sin(\frac{\omega}{2}T)e^{-j\frac{\omega}{2}T}

    零阶保持系统频域

    H(jω)H(j\omega)H0(jω)H_0(j\omega) 可得到 H𝑟(jω)H_𝑟(j\omega)

    H𝑟(jω)=H(jω)H0(jω)=2ωH(jω)sin(ω2T)ejω2TH_𝑟(j\omega)=\frac{H(j\omega)}{H_0(j\omega)}=\frac{2}{\omega}\frac{H(j\omega)}{\sin(\frac{\omega}{2}T)}e^{j\frac{\omega}{2}T}

    Hr的模Hr的幅度

    # 内插

    # 理想内插

    也称为时域带限内插

    h(t)h(t) 为理想低通滤波器的单位冲激响应,则

    xr(t)=x𝑝(t)h(t)=n=+x(nT)δ(tnT)h(t)=n=+x(nT)h(tnT)\begin{aligned} x_r(t) &=x_𝑝(t)*h(t)\\ &=\sum_{n=-∞}^{+∞}x(nT)\delta(t-nT)*h(t)\\ &=\sum_{n=-∞}^{+∞}x(nT)h(t - nT) \end{aligned}

    这里利用了单位冲激响应的卷积特性(定义):

    x(h(g(t)))=x(h(t))δ(g(t))x(h(g(t)))=x(h(t))*\delta(g(t))

    表明:理想内插以理想低通滤波器的单位冲激响应作为内插函数。

    当理想低通滤波器的截止频率为 ωc=12ωs=πT\omega_c=\frac{1}{2}\omega_s=\frac{\pi}{T} 时有:

    \begin{aligned} h(t)=T\frac{\sin \omega_c t}{\pit}\text{Sa}(\omega_ct)=\text{Sa}(\omega_ct) \end{aligned}

    则有:

    xr(t)=n=+x(nT)Sa[ωc(tnT)]x_r(t)=\sum_{n=-∞}^{+\infin}x(nT)\cdot Sa[\omega_c(t-nT)]

    按照这个过程,就得到了原始信号 x(t)x(t) 的重建信号 xr(t)x_r(t)。只要采样过程满足采样定理,那么这个重建就完全恢复了 x(t)x(t)

    # 调制与调解

    Modulation & Demodulation

    利用傅里叶变换的相乘特性:两个信号在时域相乘,可以看成是由一个信号控制另一个信号的幅度,这就是幅度调制。其中前者为调制信号,后者为载波

    WIKI:调制调解

    # 正弦幅度调制

    正弦幅度调制等效于在频域将调制信号的频谱搬移到载频位置

    调制信号 s(t)CTFTS(jω)s(t)\stackrel{CTFT}{\longleftrightarrow}S(j\omega)

    s正弦幅度调制信号的频域

    载波信号 p(t)CTFTP(jω)p(t)\stackrel{CTFT}{\longleftrightarrow}P(j\omega)

    p余弦信号的频域

    输入乘法器,输出调制后的信号:

    正弦幅度调制

    调制后的信号 r(t)CTFTR(jω)r(t)\stackrel{CTFT}{\longleftrightarrow}R(j\omega)

    正弦幅度调制结果

    # 同步调解

    必须要求调制和解调时所使用的载波不仅要严格同频,而且要同相。因此这种解调方法称为同步解调。

    r(t)cosω0tF12πR(jω)π[δ(ωω0)+δ(ω+ω0)]=12S(jω)+14S[j(ω2ω0)]+14S[j(ω+2ω0)]\begin{aligned} r(t)cos\omega_0t\stackrel{F}{\longleftrightarrow} &\ \frac{1}{2\pi}R(j\omega)*\pi[\delta(\omega-\omega_0)+\delta(\omega+\omega_0)]\\ =&\ \frac{1}{2}S(j\omega)+\frac{1}{4}S[j(\omega-2\omega_0)]+\frac{1}{4}S[j(\omega+2\omega_0)] \end{aligned}

    同步调解结果

    \ No newline at end of file +通信系统 - 信号与系统 - 电子信息 | Balloon Party = 風船のパーティー = fuusen no party

    there are still some bugs rendering katex here...

    LTI 系统对输入信号所起的作用包括两个方面:

    • 改变输入信号各频率分量的幅度
    • 改变输入信号各频率分量的相对相位

    Y(jω)=X(jω)H(jω)Y(jω)=X(jω)H(jω)Y(jω)=X(jω)+H(jω)\begin{aligned} Y(j\omega)&=X(j\omega)H(j\omega)\\ |Y(j\omega)|&=|X(j\omega)|\cdot|H(j\omega)|\\ \angle Y(j\omega)&=\angle X(j\omega)+\angle H(j\omega) \end{aligned}

    线性与非线性相位

    信号在传输过程中,相位特性或幅度特性发生改变都可能引起信号波形的改变,即发生失真

    当相位特性仅仅是产生一个线性相移时,只引起信号在时间上的平移。如连续时间 LTI 系统,信号的时移会导致其频谱产生一个线性相移

    x(t)FX(jω)x(tt0)FX(jω)ejωt0x(t)\stackrel{F}{\longleftrightarrow}X(j\omega)\\ \dArr\\ x(t-t_0)\stackrel{F}{\longleftrightarrow}X(j\omega)e^{-j\omega t_0}

    此时信号波形不发生改变,只是发生时间上的延迟,因此未丢失信号所携带的任何信息。

    如果系统的相位特性是非线性的,由于不同频率分量所产生的时移不同,叠加起来会变成一个与原来信号不相同的信号波形。

    系统响应与输入信号满足:

    y(t)=kx(tt0)y[n]=kx[nn0]y(t)=kx(t-t_0)\ \ \ y[n]=kx[n-n_0]

    也就是:

    则可视为在传输中未发生失真

    H(jω)=k,H(jω)=ωt0|H(j\omega)|=k,\ \ \angle H(j\omega)=-\omega t_0

    在传输信号的带宽范围内

    理想滤波器

    • 非因果系统,因而是物理不可实现
    • h(t)h(t) 存在震荡

    # 采样

    # 采样的概念

    在某些离散的时间点上提取连续时间信号值的过程就是采样(sampling)

    与之相对应的,从离散时间信号等间隔地提取信号,则叫做抽取(decimation)

    一般情况下,从连续时间信号采样所得到的样本序列不能唯一地代表原来的连续时间信号,决定是否能从采样结果恢复原信号的关键在于采样定理

    # 采样的数学模型

    采样的数学模型

    在时域:

    xp(t)=x(t)p(t)x_p(t)=x(t)\cdot p(t)

    在频域:

    Xp(jω)=12πX(jω)P(jω)X_p(j\omega)=\frac{1}{2\pi}X(j\omega)*P(j\omega)

    # 冲激串采样 (理想采样)

    TT 为采样间隔的脉冲采样信号为:

    p(t)=n=+δ(tnT)p(t)=\sum_{n=-\infin}^{+\infin}\delta(t-nT)

    • p(t)p(t)采样函数
    • p(t)p(t) 的基波周期 TT采样周期
    • p(t)p(t) 的基波频率 ωs=2πT\omega_s=\frac{2\pi}{T}采样频率

    采样结果为:

    xp(t)=x(t)p(t)=n=+x(nT)δ(tnT)x_p(t)=x(t)\cdot p(t)=\sum_{n=-\infin}^{+\infin}x(nT)\delta(t-nT)

    这样的采样就是理想采样,也就是冲激串采样

    理想采样

    由 CTFT 在频域有:

    p(t)CTFTP(jω)=2πTn=+δ(ω2πTk)p(t)\stackrel{CTFT}{\longleftrightarrow}P(j\omega)=\frac{2\pi}{T}\sum_{n=-\infin}^{+\infin}\delta(\omega-\frac{2\pi}{T}k)

    从而可得到 Xp(jω)X_p(j\omega)

    Xp(jω)=12πX(jω)P(jω)=12πX(jω)2πTn=+δ(ω2πTk)=1Tn=+X(j(ω2πTk))=1Tn=+X(j(ωkωs))\begin{aligned} X_p(j\omega) &=\frac{1}{2\pi}X(j\omega)*P(j\omega)\\ &=\frac{1}{2\pi}X(j\omega)*\frac{2\pi}{T}\sum_{n=-\infin}^{+\infin}\delta(\omega-\frac{2\pi}{T}k)\\ &=\frac{1}{T}\sum_{n=-\infin}^{+\infin}X(j(\omega-\frac{2\pi}{T}k))\\ &=\frac{1}{T}\sum_{n=-\infin}^{+\infin}X(j(\omega-k\omega_s)) \end{aligned}

    可见,在时域对连续时间信号进行理想采样,就相当于在频域将连续时间信号的频谱以 ωs=2πT\omega_s=\frac{2\pi}{T} 为周期进行延拓。

    # 采样定理

    要想使采样后的信号样本能完全代表原来的信号,就意味着要能够从 X𝑝(jω)X_𝑝(j\omega)不失真地分离X(jω)X(j\omega) 。这就要求 X𝑝(jω)X_𝑝(j\omega) 在周期性延拓时不能发生频谱的混叠,因此:

    1. x(t)x(t) 必须是带限的,设最高频率分量为 ωM\omega_M
    2. 采样间隔(周期)不能是任意的,必须保持采样频率 ωs=2πT>2ωM\omega_s=\frac{2\pi}{T}\gt 2\omega_M
    3. 采样频率也不能过大,不能导致周期延拓后频谱发生混叠(补充)

    采样频率的下限 2ωM2\omega_M 也叫做奈奎斯特率(Nyquist rate)

    在满足上述条件时,就可以通过理想低通滤波器从 X𝑝(jω)X_𝑝(j\omega) 中不失真地分离出 X(jω)X(j\omega)

    # 理想低通滤波器

    低通滤波器频域

    若要满足采样定理,低通滤波器截止频率 ωc\omega_c 必须满足:ωM<ωc<ωsωM\omega_M\lt\omega_c\lt\omega_s-\omega_M

    为补偿采样导致的频谱幅度减小,滤波器在通带内应具有 TT 倍增益。

    # Nyquist 采样定理

    对带限于最高频率 ω𝑀\omega_𝑀 的连续时间信号 x(t)x(t) ,如果以 ωs>2ω𝑀\omega_s>2\omega_𝑀 的频率进行理想采样,则 x(t)x(t) 可以唯一地由其样本 x(nT)x(nT) 来确定。

    这里的 TT 也是采样周期,对应采样频率 ωs\omega_s

    # 零阶保持采样

    零阶保持采样相当于理想采样后,再级联一个零阶保持系统,是最简单的一种恢复重建采样前原信号的方法。

    如果由零阶保持所给出的粗糙内插令人不够满意,则可以使用各种更为平滑的內插手段,其中的一些合起来统称为高阶保持 (higher order hold)

    零阶保持采样

    零阶保持采样结果

    为了能从 x0(t)x_0(t) 恢复 x(t)x(t) ,零阶保持后要再级(串,卷积)联一个系统 H𝑟(jω)H_𝑟(j\omega) ,使得:

    H0(jω)Hr(jω)=H(jω)={T,ω<ωc0,ω>ωcH_0(j\omega)H_r(j\omega)=H(j\omega)= \begin{cases} T,&|\omega|\lt\omega_c\\ 0,&|\omega|\gt\omega_c \end{cases}

    其中 ωM<ωc<ωsωM\omega_M\lt\omega_c\lt\omega_s-\omega_M ;图像如下:

    零阶保持采样最终输出

    零阶保持系统的单位冲激响应 h0(t)h_0(t) 是矩形方波信号(经过了一定的时移),利用基本 CTFT 变换对和时移特性可得到:

    H0(jω)=2ωsin(ω2T)ejω2TH_0(j\omega)=\frac{2}{\omega}\sin(\frac{\omega}{2}T)e^{-j\frac{\omega}{2}T}

    零阶保持系统频域

    H(jω)H(j\omega)H0(jω)H_0(j\omega) 可得到 H𝑟(jω)H_𝑟(j\omega)

    H𝑟(jω)=H(jω)H0(jω)=2ωH(jω)sin(ω2T)ejω2TH_𝑟(j\omega)=\frac{H(j\omega)}{H_0(j\omega)}=\frac{2}{\omega}\frac{H(j\omega)}{\sin(\frac{\omega}{2}T)}e^{j\frac{\omega}{2}T}

    Hr的模Hr的幅度

    # 内插

    # 理想内插

    也称为时域带限内插

    h(t)h(t) 为理想低通滤波器的单位冲激响应,则

    xr(t)=x𝑝(t)h(t)=n=+x(nT)δ(tnT)h(t)=n=+x(nT)h(tnT)\begin{aligned} x_r(t) &=x_𝑝(t)*h(t)\\ &=\sum_{n=-∞}^{+∞}x(nT)\delta(t-nT)*h(t)\\ &=\sum_{n=-∞}^{+∞}x(nT)h(t - nT) \end{aligned}

    这里利用了单位冲激响应的卷积特性(定义):

    x(h(g(t)))=x(h(t))δ(g(t))x(h(g(t)))=x(h(t))*\delta(g(t))

    表明:理想内插以理想低通滤波器的单位冲激响应作为内插函数。

    当理想低通滤波器的截止频率为 ωc=12ωs=πT\omega_c=\frac{1}{2}\omega_s=\frac{\pi}{T} 时有:

    \begin{aligned} h(t)=T\frac{\sin \omega_c t}{\pit}\text{Sa}(\omega_ct)=\text{Sa}(\omega_ct) \end{aligned}

    则有:

    xr(t)=n=+x(nT)Sa[ωc(tnT)]x_r(t)=\sum_{n=-∞}^{+\infin}x(nT)\cdot Sa[\omega_c(t-nT)]

    按照这个过程,就得到了原始信号 x(t)x(t) 的重建信号 xr(t)x_r(t)。只要采样过程满足采样定理,那么这个重建就完全恢复了 x(t)x(t)

    # 调制与调解

    Modulation & Demodulation

    利用傅里叶变换的相乘特性:两个信号在时域相乘,可以看成是由一个信号控制另一个信号的幅度,这就是幅度调制。其中前者为调制信号,后者为载波

    WIKI:调制调解

    # 正弦幅度调制

    正弦幅度调制等效于在频域将调制信号的频谱搬移到载频位置

    调制信号 s(t)CTFTS(jω)s(t)\stackrel{CTFT}{\longleftrightarrow}S(j\omega)

    s正弦幅度调制信号的频域

    载波信号 p(t)CTFTP(jω)p(t)\stackrel{CTFT}{\longleftrightarrow}P(j\omega)

    p余弦信号的频域

    输入乘法器,输出调制后的信号:

    正弦幅度调制

    调制后的信号 r(t)CTFTR(jω)r(t)\stackrel{CTFT}{\longleftrightarrow}R(j\omega)

    正弦幅度调制结果

    # 同步调解

    必须要求调制和解调时所使用的载波不仅要严格同频,而且要同相。因此这种解调方法称为同步解调。

    r(t)cosω0tF12πR(jω)π[δ(ωω0)+δ(ω+ω0)]=12S(jω)+14S[j(ω2ω0)]+14S[j(ω+2ω0)]\begin{aligned} r(t)cos\omega_0t\stackrel{F}{\longleftrightarrow} &\ \frac{1}{2\pi}R(j\omega)*\pi[\delta(\omega-\omega_0)+\delta(\omega+\omega_0)]\\ =&\ \frac{1}{2}S(j\omega)+\frac{1}{4}S[j(\omega-2\omega_0)]+\frac{1}{4}S[j(\omega+2\omega_0)] \end{aligned}

    同步调解结果

    \ No newline at end of file diff --git a/ie/signals-and-systems/duality/index.html b/ie/signals-and-systems/duality/index.html index f728fd37..79b9245e 100644 --- a/ie/signals-and-systems/duality/index.html +++ b/ie/signals-and-systems/duality/index.html @@ -1 +1 @@ -离散 / 连续时间傅里叶系数 / 变换的对偶关系 - 信号与系统 - 电子信息 | Balloon Party = 風船のパーティー = fuusen no party

    # 归纳总结

    缩写:

    • CTFS—— 连续时间傅里叶级数
    • DTFS—— 离散时间傅里叶级数
    • CTFT—— 连续时间傅里叶变换
    • DTFT—— 离散时间傅里叶变换

    一图归纳:

    归纳

    信号在时域的特性和在频域的特性之间存在以下对应关系:

    • 时域的周期性 —— 频域的离散性
    • 时域的非周期性 —— 频域的连续性
    • 时域的离散性 —— 频域的周期性
    • 时域的连续性 —— 频域的非周期性

    # CTFT 的对偶

    x(t)CTFTX(jω)x(t)\stackrel{CTFT}{\longleftrightarrow}X(j\omega) ,则有:

    x(t)=12π+X(jω)ejωtdω2πx(t)=+X(jω)ejωtdω2πx(ω)=+X(jt)ejωtdt2πx(ω)=+X(jt)ejωtdt\begin{aligned} x(t) &=\frac{1}{2\pi} \int_{-\infin}^{+\infin} X(j\omega) e^{j\omega t}d\omega\\ 2\pi x(t) &=\int_{-\infin}^{+\infin} X(j\omega) e^{j\omega t}d\omega\\ 2\pi x(\omega) &=\int_{-\infin}^{+\infin} X(jt) e^{j\omega t}dt\\ 2\pi x(-\omega) &=\int_{-\infin}^{+\infin} X(jt) e^{-j\omega t}dt\\ \end{aligned}

    可得到对偶关系:

    x(t)CTFTX(jω)X(jt)CTFT2πx(ω)x(t)\stackrel{CTFT}{\longleftrightarrow}X(j\omega)\\\\ X(jt)\stackrel{CTFT}{\longleftrightarrow}2\pi x(-\omega)

    利用这一对偶关系,可以将 CTFT 时域的某些特性对偶到频域,或者反之。

    # DFS 的对偶

    ak=1Nn=Nx[n]ej2πNkna_k=\frac{1}{N}\sum_{n=\langle N\rangle}x[n]e^{-j\frac{2\pi}{N}kn}

    aka_k 本身也是以 NN 为周期的序列,用 n-n 变量代换 nn ,则可以将 aka_k 写为离散时间傅里叶级数(DFS)的形式:

    ak=n=N1Nx[n]ej2πNkna_k=\sum_{n=\langle N\rangle}\frac{1}{N}x[-n]\cdot e^{j\frac{2\pi}{N}kn}

    也就是(将 kknn 变量交换):

    an=k=N1Nx[k]ej2πNkna_n=\sum_{k=\langle N\rangle}\frac{1}{N}x[-k]\cdot e^{j\frac{2\pi}{N}kn}

    可以得到对偶关系;

    x[n]DFSakanDFS1Nx[k]\begin{aligned} x[n]&\stackrel{DFS}{\longleftrightarrow}a_k\\\\ a_n&\stackrel{DFS}{\longleftrightarrow}\frac{1}{N}x[-k] \end{aligned}

    利用这一对偶关系,可以将 DFS 在时域的性质推广至频域,或者反之。

    # DTFT 与 CFS 间的对偶

    x[n]x[n] 的离散时间傅里叶正变换(DTFT)X(ejω)X(e^{j\omega}) 是以 2π2\pi 为周期的连续函数。如果将其视为连续时间信号 X(ejt)X(e^{jt}) ,则可以表示为连续时间傅里叶级数(CFS)的形式:

    X(ejt)=n=+x[n]ejtn=k=+x[k]ejkt\begin{aligned} X(e^{jt}) &=\sum_{n=-\infin}^{+\infin}x[n]\cdot e^{-jtn}\\ &=\sum_{k=-\infin}^{+\infin}x[-k]\cdot e^{jkt} \end{aligned}

    于是就有对偶关系:

    x[n]DTFTX(ejω)X(ejt)DFSx[k]\begin{aligned} x[n]&\stackrel{DTFT}{\longleftrightarrow}X(e^{j\omega})\\\\ X(e^{jt})&\stackrel{DFS}{\longleftrightarrow}x[-k] \end{aligned}

    利用这一对偶关系,可以将 DTFT 的若干特性对偶到 CFS 中去,或者反之。

    \ No newline at end of file +离散 / 连续时间傅里叶系数 / 变换的对偶关系 - 信号与系统 - 电子信息 | Balloon Party = 風船のパーティー = fuusen no party

    # 归纳总结

    缩写:

    • CTFS—— 连续时间傅里叶级数
    • DTFS—— 离散时间傅里叶级数
    • CTFT—— 连续时间傅里叶变换
    • DTFT—— 离散时间傅里叶变换

    一图归纳:

    归纳

    信号在时域的特性和在频域的特性之间存在以下对应关系:

    • 时域的周期性 —— 频域的离散性
    • 时域的非周期性 —— 频域的连续性
    • 时域的离散性 —— 频域的周期性
    • 时域的连续性 —— 频域的非周期性

    # CTFT 的对偶

    x(t)CTFTX(jω)x(t)\stackrel{CTFT}{\longleftrightarrow}X(j\omega) ,则有:

    x(t)=12π+X(jω)ejωtdω2πx(t)=+X(jω)ejωtdω2πx(ω)=+X(jt)ejωtdt2πx(ω)=+X(jt)ejωtdt\begin{aligned} x(t) &=\frac{1}{2\pi} \int_{-\infin}^{+\infin} X(j\omega) e^{j\omega t}d\omega\\ 2\pi x(t) &=\int_{-\infin}^{+\infin} X(j\omega) e^{j\omega t}d\omega\\ 2\pi x(\omega) &=\int_{-\infin}^{+\infin} X(jt) e^{j\omega t}dt\\ 2\pi x(-\omega) &=\int_{-\infin}^{+\infin} X(jt) e^{-j\omega t}dt\\ \end{aligned}

    可得到对偶关系:

    x(t)CTFTX(jω)X(jt)CTFT2πx(ω)x(t)\stackrel{CTFT}{\longleftrightarrow}X(j\omega)\\\\ X(jt)\stackrel{CTFT}{\longleftrightarrow}2\pi x(-\omega)

    利用这一对偶关系,可以将 CTFT 时域的某些特性对偶到频域,或者反之。

    # DFS 的对偶

    ak=1Nn=Nx[n]ej2πNkna_k=\frac{1}{N}\sum_{n=\langle N\rangle}x[n]e^{-j\frac{2\pi}{N}kn}

    aka_k 本身也是以 NN 为周期的序列,用 n-n 变量代换 nn ,则可以将 aka_k 写为离散时间傅里叶级数(DFS)的形式:

    ak=n=N1Nx[n]ej2πNkna_k=\sum_{n=\langle N\rangle}\frac{1}{N}x[-n]\cdot e^{j\frac{2\pi}{N}kn}

    也就是(将 kknn 变量交换):

    an=k=N1Nx[k]ej2πNkna_n=\sum_{k=\langle N\rangle}\frac{1}{N}x[-k]\cdot e^{j\frac{2\pi}{N}kn}

    可以得到对偶关系;

    x[n]DFSakanDFS1Nx[k]\begin{aligned} x[n]&\stackrel{DFS}{\longleftrightarrow}a_k\\\\ a_n&\stackrel{DFS}{\longleftrightarrow}\frac{1}{N}x[-k] \end{aligned}

    利用这一对偶关系,可以将 DFS 在时域的性质推广至频域,或者反之。

    # DTFT 与 CFS 间的对偶

    x[n]x[n] 的离散时间傅里叶正变换(DTFT)X(ejω)X(e^{j\omega}) 是以 2π2\pi 为周期的连续函数。如果将其视为连续时间信号 X(ejt)X(e^{jt}) ,则可以表示为连续时间傅里叶级数(CFS)的形式:

    X(ejt)=n=+x[n]ejtn=k=+x[k]ejkt\begin{aligned} X(e^{jt}) &=\sum_{n=-\infin}^{+\infin}x[n]\cdot e^{-jtn}\\ &=\sum_{k=-\infin}^{+\infin}x[-k]\cdot e^{jkt} \end{aligned}

    于是就有对偶关系:

    x[n]DTFTX(ejω)X(ejt)DFSx[k]\begin{aligned} x[n]&\stackrel{DTFT}{\longleftrightarrow}X(e^{j\omega})\\\\ X(e^{jt})&\stackrel{DFS}{\longleftrightarrow}x[-k] \end{aligned}

    利用这一对偶关系,可以将 DTFT 的若干特性对偶到 CFS 中去,或者反之。

    \ No newline at end of file diff --git a/ie/signals-and-systems/lti/index.html b/ie/signals-and-systems/lti/index.html index 2780e7c8..59967958 100644 --- a/ie/signals-and-systems/lti/index.html +++ b/ie/signals-and-systems/lti/index.html @@ -1 +1 @@ -卷积と线性时不变系统 - 信号与系统 - 电子信息 | Balloon Party = 風船のパーティー = fuusen no party

    Linear Time Invariant 线性时不变

    基本信号应具有以下特点:

    • 本身尽可能简单,并且用它们的线性组合能够表示尽可能广泛的信号
    • LTI 系统对这种信号的响应易于计算且有利于对系统的深入理解

    两类有用的基本信号:

    • 复指数信号
    • 单位冲激信号

    本章主要考虑以单位冲激信号为基本信号进行信号与系统分析

    summary

    # 卷积和

    # 定义

    对于任意一个离散时间系统:

    离散时间系统

    设离散时间系统为 LTI 的,定义单位脉冲响应 h[n]h[n] 为以单位冲激信号 δ[n]\delta[n] 为输入时的输出:

    δ[n]h[n]\delta[n]\to h[n]

    由 LTI 时不变性有:

    δ[nk]h[nk]\delta[n-k]\to h[n-k]

    由 LTI 线性有:

    x[n]=k=+x[k]δ[nk]y[n]=k=+x[k]h[nk]x[n]=\sum_{k=-\infin}^{+\infin}x[k]\delta[n-k]\to y[n]=\sum_{k=-\infin}^{+\infin}x[k]h[n-k]

    这种求得系统响应的运算关系称为卷积和,即离散时间 LTI 系统的响应可以用卷积和表示为:

    y[n]=x[n]h[n]=k=+x[k]h[nk]y[n]=x[n]*h[n]=\sum_{k=-\infin}^{+\infin}x[k]h[n-k]

    ⭐️更一般的形式:

    x[nn0n1]=x[nn0]h[nn1]x[n-n_0-n_1]=x[n-n_0]*h[n-n_1]

    ⭐️只要得到了 LTI 系统对 δ[n]\delta[n] 的 响 应 h[n]h[n] —— 单位脉冲响应,就可以得到 LTI 系统对任何输入信号 x[n]x[n] 的响应。

    换言之,一个 LTI 系统可以完全由它的单位脉冲响应 h[n]h[n] 来表征。

    # 特殊离散 LTI 系统

    # 延时器

    h[n]=δ[nn0]h[n]=\delta[n-n_0]

    具有该单位脉冲响应的系统一般情况下存在多种可能性,但是对应的 LTI 系统只有一个:

    y[n]=x[nn0]=x[n]δ[nn0]y[n]=x[n-n_0]=x[n]*\delta[n-n_0]

    # 累加器

    h[n]=k=nδ[k]=u[n]h[n]=\sum_{k=-\infin}^{n}\delta[k]=u[n]

    对应的 LTI 系统:

    y[n]=x[n]u[n]=k=nx[k]y[n]=x[n]*u[n]=\sum_{k=-\infin}^{n}x[k]

    # 计算

    选择 nn 值并固定,求和项视为 kk 的函数,步骤如下:

    1. x[n]x[n]h[n]h[n] 变为 x[k]x[k]h[k]h[k] ,以 kk 为求和变量
    2. h[k]h[k] 翻转变为 h[k]h[-k]
    3. h[k]h[-k] 平移 nn,变为 h[nk]h[n - k]
    4. x[k]x[k]h[nk]h[n-k] 相乘;
    5. 对乘积 x[k]h[nk]x[k]h[n-k]kZk\in \mathbb{Z} 取和;
      • 一般都转化为只对非 0 区间积分。注意非 0 区间上下限!!!
    6. 改变 nn 值并重复 3–5 步,直到 nn 超过了 x[n]x[n] 非零部分定义域

    注意很多题目最后答案会加上一个 u(n)u(n) ,是因为只求定义域在 [0,+)[0,+\infin) 的部分

    # 卷积积分

    # 定义

    类似积分思想,转化为无限多的离散和的极限

    用冲激信号表示连续时间信号:

    将输入信号 x(t)x(t) 近似表示为移位、放缩后的脉冲之和

    脉冲和近似表示输入信号

    xΔ(t)=x(kΔ),kΔ<t<(k+1)Δx_{\Delta}(t)=x(k\Delta)\ ,\ k\Delta\lt t\lt(k+1)\Delta

    阶梯近似表达过程:

    设连续时间系统为 LTI 的,定义(名字待补,与卷积和中的单位脉冲响应对应) hΔ(n)h_{\Delta}(n) 为以单位冲激信号 δΔ(n)\delta_{\Delta}(n) 为输入时的输出:

    δΔ(n)hΔ(n)\delta_{\Delta}(n)\to h_{\Delta}(n)

    阶梯近似表达

    由 LTI 时不变性有:

    δΔ(nkΔ)hΔ(nkΔ)\delta_{\Delta}(n-k\Delta)\to h_{\Delta}(n-k\Delta)

    由 LTI 线性齐次性尺度调整:

    x(kΔ)δΔ(nkΔ)x(kΔ)hΔ(nkΔ)x(k\Delta)\delta_{\Delta}(n-k\Delta)\to x(k\Delta)h_{\Delta}(n-k\Delta)

    阶梯近似表达2

    由 LTI 线性叠加性有:

    k=+x(kΔ)δΔ(nkΔ)Δk=+x(kΔ)hΔ(nkΔ)ΔxΔ(t)yΔ(t)\sum_{k=-\infin}^{+\infin}x(k\Delta)\delta_{\Delta}(n-k\Delta)\Delta\to \sum_{k=-\infin}^{+\infin}x(k\Delta)h_{\Delta}(n-k\Delta)\Delta\\ x_{\Delta}(t)\to y_{\Delta}(t)

    当 Δ → 0 时 :

    • xΔ(t)x(t)x_Δ(t) → x(t)
    • yΔ(t)y(t)y_Δ(t) → y(t)
    • kΔτkΔ → \tau
    • ΔdτΔ → d\tau
    • δΔ(tkΔ)δ(tτ)\delta_Δ(t - kΔ) → \delta(t - \tau)
    • hΔ(tkΔ)h(tτ)h_Δ(t - kΔ) → h(t - \tau)
    • \sum\to\int

    从而得到:

    x(t)=+x(τ)δ(tτ)dτ=x(t)δ(t)y(t)=+x(τ)h(tτ)dτ=x(t)h(t)x(t)=\int_{-\infin}^{+\infin}x(\tau)\delta(t-\tau)d\tau=x(t)*\delta(t)\\ y(t)=\int_{-\infin}^{+\infin}x(\tau)h(t-\tau)d\tau=x(t)*h(t)

    • 其中第一个式子意义是:用位移的单位冲激信号表示任意的连续时间信号
    • 其中第一个式子意义是:用卷积积分表示连续时间 LTI 系统的响应

    阶梯近似图解过程:

    阶梯近似图解过程

    # 计算

    求卷积积分 x(t)h(t)=+x(τ)h(tτ)dτx(t) * h(t) = \int_{-∞}^{+∞}x(\tau)h(t - \tau)d\tau 的过程与卷积和计算过程相似。卷积积分的计算步骤如下:

    1. x(t)x(t)h(t)h(t) 变为 x(τ)x(\tau)h(τ)h(\tau) ,以 τ\tau 为积分变量;
    2. h(τ)h(\tau) 翻转变为 h(τ)h(-\tau)
    3. h(τ)h(-\tau) 平移 tt ,变为 h(tτ)h(t - \tau)
    4. x(τ)x(\tau)h(tτ)h(t-\tau) 相乘;
    5. 求区间 (,+)(-\infin,+\infin) 上乘积 x(τ)h(tτ)x(\tau)h(t-\tau) 下的面积;
      • 一般都转化为只对非 0 区间积分。注意非 0 区间上下限!!!
    6. 改变 tt 值并重复 3–5 步。

    # 特殊连续 LTI 系统

    # 积分器

    单位冲激信号为输入下系统的响应为:

    h(t)=u(t)h(t)=u(t)

    对任意激励信号 x(t)x(t) ,系统响应为:

    y(t)=x(t)h(t)=+x(τ)h(tτ)dτ=tx(τ)dτy(t)=x(t)*h(t)=\int_{-\infin}^{+\infin}x(\tau)h(t-\tau)d\tau = \int_{-\infin}^{t}x(\tau)d\tau

    # 卷积和与卷积积分的性质

    # 交换律

    变量替换:

    x[n]h[n]=k=+x[k]h[nk]=l=+x[nl]h[l]=h[n]x[n]x[n]*h[n]=\sum_{k=-\infin}^{+\infin}x[k]h[n-k]=\sum_{l=-\infin}^{+\infin}x[n-l]h[l]=h[n]*x[n]

    x(n)h(n)=+x(τ)h(tτ)dτ=+x(tτ)h(τ)dτ=h(t)x(t)x(n)*h(n)=\int_{-\infin}^{+\infin}x(\tau)h(t-\tau)d\tau=\int_{-\infin}^{+\infin}x(t-\tau')h(\tau')d\tau'=h(t)*x(t)

    交换律

    s[n]=u[n]h[n]=h[n]u[n]=k=nh[k]s[n]=u[n]*h[n]=h[n]*u[n]=\sum_{k=-\infin}^{n}h[k]

    利用 u[n]u[n] 是累加器的单位脉冲响应以及交换律可以得到:

    该系统的单位脉冲响应是 h[n]h[n]u[n]u[n] 单位阶跃信号作为输入,则其单位阶跃响应 s[n]s[n] 等价于 以 h[n]h[n] 作为输入时,累加器的响应

    # 分配律

    x[n](h1[n]+h2[n])=x[n]h1[n]+x[n]h2[n]x(t)(h1(t)+h2(t))=x(t)h1(t)+x(t)h2(t)x[n] * (h_1[n] + h_2[n]) = x[n] * h_1[n] + x[n] * h_2[n]\\ x(t) * (h_1(t) + h_2(t)) = x(t) * h_1(t) + x(t) * h_2(t)

    分配律

    # 结合律

    (x[n]h1[n])h2[n]=x[n](h1[n]h2[n])(x(t)h1(t))h2(t)=x(t)(h1(t)h2(t))(x[n] * h_1 [n] ) * h_2[n] = x[n] * (h_1 [n] * h_2[n])\\ (x(t) * h_1 (t) ) * h_2(t) = x(t) * (h_1(t)* h_2 (t))

    结合律

    级联的 LTI 系统其总的单位冲激 (脉冲) 响应等于各子系统单位冲激 (脉冲) 响应的卷积

    # 级联可交换

    卷积运算满足结合律和交换律,因此,系统级联的先后次序可以调换。

    x[n]h1[n]h2[n]=x[n]h2[n]h1[n]x(t)h1(t)h2(t)=x(t)h2(t)h1(t)x[n] * h_1[n] * h_2[n] = x[n] * h_2[n] * h_1[n]\\ x(t) * h_1(t) * h_2(t) = x(t) * h_2(t) * h_1(t)

    级联可交换

    级联次序可交换的前提条件:

    • 系统是 LTI 的
    • 所有涉及到的卷积运算必须收敛

    # 微分、积分及时移特性

    卷积积分满足微分、积分及时移特性。若 x(t)h(t)=y(t)x(t)*h(t)=y(t),则:

    # 微分特性

    x(t)h(t)=x(t)h(t)=y(t)x'(t) * h(t) = x(t) * h'(t) = y'(t)

    x(m)(t)h(n)(t)=y(m+n)(t)x^{(m)}(t) * h^{(n)}(t) = y^{(m+n)}(t)

    tt 求导而不是对 τ\tau 求导

    # 积分特性

    (tx(τ)dτ)h(t)=+(τ1x(τ)dτ)h(tτ1)dτ1=x(t)(th(τ)dτ)=ty(τ)dτ\begin{aligned} \big(\int_{-\infin}^{t}x(\tau)d\tau\big)*h(t)&= \int_{-\infin}^{+\infin}\big(\int_{-\infin}^{\tau_1}x(\tau)d\tau\big)h(t-\tau_1)d\tau_1\\ &=x(t)*\big(\int_{-\infin}^{t}h(\tau)d\tau\big)\\ &=\int_{-\infin}^{t}y(\tau)d\tau \end{aligned}

    # 时移特性

    x(tt0)h(tt1)=x(tt1)h(tt0)=y(tt0t1)x(t - t_0) * h(t-t_1) = x(t-t_1) * h(t - t_0) = y(t - t_0-t_1)

    # 差分、求和及时移特性

    卷积和满足差分、求和及时移特性。若 x [n] * h [n] = y [n],则:

    # 差分特性

    x[n]x[n1]h[n]=x[n](h[n]h[n1])=y[n]y[n1]x[n]- x[n- 1] * h[n]= x[n]* (h[n]- h[n - 1])= y[n] - y[n - 1]

    # 求和特性

    (k=nx[k])h[n]=x[n](k=nh[k])=k=ny[k](\sum_{k=-∞}^{n}x[k] )* h[n] = x[n] * (\sum_{k=-∞}^{n}h[k] )=\sum_{k=-∞}^{n}y[k]

    # 时移特性

    x[tt0]h[tt1]=x[tt1]h[tt0]=y[tt0t1]x[t - t_0] * h[t-t_1] = x[t-t_1]* h[t - t_0]= y[t - t_0-t_1]

    # LTI 系统的性质

    # 记忆性

    对于离散时间 LTI 系统:

    y[n]=k=+x[k]h[nk]y[n]=\sum_{k=-\infin}^{+\infin}x[k]h[n-k]

    如果系统是无记忆的,则在任何时刻 n ,y [n] 都只能和 n 时刻的输入有关,等式右边的和式中只能有 k = n 时的一项非零,即必须有 h [n - k] = 0 , k ≠ n,也即:

    h[n]=0,n0h[n] = 0 ,\ \ \ n\neq0

    所以,离散时间无记忆系统的单位脉冲响应为:

    h[n]=kδ[n]h[n]=k\delta[n]

    即为离散时间系统无记忆充要条件

    类似地,连续时间系统无记忆的充要条件为:

    h(n)=kδ[(n)h(n)=k\delta[(n)

    当 k = 1 时系统是恒等系统。

    # 可逆性

    如果 LTI 系统是可逆的,则其逆系统也是 LTI 系统。

    互逆的两个系统的满足(充要条件卷积和等于单位冲激信号

    h(t)g(t)=δ(t)h[n]g[n]=δ[n]\begin{aligned} h(t) * g(t) =\delta(t)\\ h[n] * g[n] =\delta[n] \end{aligned}

    从而有:

    x(t)h(t)g(t)=x(t)δ(t)=x(t)x[t]h[n]g[n]=x[t]δ[n]=x[t]\begin{aligned} x(t)*h(t)*g(t)=x(t)*\delta(t)=x(t)\\ x[t]*h[n]*g[n]=x[t]*\delta[n]=x[t] \end{aligned}

    可逆性

    例如:延时器是可逆的 LTI 系统,h (t) = \delta (t - t0),其逆系统是 g (t) = \delta (t + t0),显然有:

    h(t)g(t)=(tt0)δ(t+t0)=δ(t)h(t) * g(t) = (t - t_0) * \delta(t + t_0) = \delta(t)

    又如:累加器是可逆的 LTI 系统,其 h [n] = u [n],其逆系统是 g [n] = \delta [n] - \delta [n - 1],显然也有:

    h[n]g[n]=u[n](δ[n]δ[n1])=u[n]u[n1]=δ[n]\begin{aligned} h[n] * g[n] &= u[n] * (\delta[n] - \delta[n - 1])\\ &= u[n] - u[n - 1] = \delta[n] \end{aligned}

    # 因果性

    对于离散时间 LTI 系统:

    y[n]=k=+x[k]h[nk]y[n]=\sum_{k=-\infin}^{+\infin}x[k]h[n-k]

    因果性意味着上式等号右边和式中所有 k > n 的项都必须为零,即 h [n - k] = 0,k > n,也即:

    h[n]=0,n<0h[n] = 0,\ \ \ n < 0

    同理对连续时间因果系统有:

    h(t)=0,t<0h(t) = 0, \ \ \ t < 0

    即为 LTI 系统具有因果性的充要条件

    对于线性系统而言,因果性与初始松弛条件等价。所谓系统满足初始松弛条件是指该系统具有如下表现:对于任意时刻,只要该时刻之前系统的输入为 0,则该时刻之前系统的输出也必然为 0。

    (参考教材习题 1.44 (a))

    – 该等价性的前提是系统为线性的。否则不一定成立。例如 y n = 2x n + 3 是因果的,但显然不满足初始松弛条件。

    # 稳定性

    稳定性意味着有界输入必然产生有界输出。以离散时间 LTI 系统为例,设 |x [n]| 有界,| x [n - k]| ≤ A。有:

    y[n]=k=+h[k]x[nk]k=+h[k]x[nk]Ak=+h[k]|y[n]| = |\sum_{k=-∞}^{+∞}h[k]x[n - k]| ≤ \sum_{k=-∞}^{+∞} |h[k]| |x[n - k]| ≤ A \sum_{k=-∞}^{+∞}|h[k]|

    离散 LTI 系统稳定的充分必要条件为:

    n=+h[n]<+\sum_{n=-\infin}^{+\infin}|h[n]|\lt +\infin

    类似地有连续 LTI 系统稳定的充分必要条件为:

    +h(t)dt<+\int_{-\infin}^{+\infin}|h(t)|dt \lt +\infin

    # 单位阶跃响应

    在工程实际中,也常用单位阶跃响应来描述 LTI 系统。单位阶跃响应就是系统对 u (t) 或 u [n] 所产生的响应,记为 s (t) 或 s [n]。因此有:

    s[n]=u[n]h[n]=k=nh[k]s[n] = u[n] * h[n] = \sum_{k=-∞}^n h[k]

    h[n]=s[n]s[n1]h[n] = s[n] - s[n - 1]

    s(t)=u(t)h(t)=th(τ)dτs(t) = u(t) * h(t) = \int_{-\infin}^{t}h(\tau)d\tau

    h(t)=ds(t)dt=s(t)h(t) =\frac{ds(t)}{dt}= s'(t)

    # 用微分和差分方程描述的因果 LTI 系统

    在工程实际中有相当普遍的一类系统,其数学模型可以用线性常系数微分方程或线性常系数差分方程来描述。

    在时域内分析这类 LTI 系统,就是要求解线性常系数微分方程或差分方程。

    # 线性常系数微分方程

    一般情况 NMN\geq M , N 阶非其次常系数微分方程的一般形式:

    k=0Nakdky(t)dtk=k=0Mbkdky(t)dtk\sum_{k=0}^{N}a_k\frac{d^ky(t)}{dt^k}=\sum_{k=0}^{M}b_k\frac{d^ky(t)}{dt^k}

    其中 ak, bk 均为常数

    求解该微分方程(求 y (t) 的显式表达)的步骤:

    1. 求出齐次解通解)yh (t) ,也称为系统的自然响应自由响应
      1. 建立特征方程求出特征根 𝜆k
      2. 根据特征根得到齐次解形式
      3. 根据附加条件确定齐次解形式中的待定常数 Ck
    2. 特解 y𝑝(t),也称系统的受迫响应强迫响应,其形式与输入信号有关
      1. 根据输入信号形式得到特解形式
      2. 将输入信号与特解代入微分方程确定特解形式中的待定常数
    3. 得到完全解 y (t) = yh(t) + y𝑝(t)

    附加条件常给出为初始松弛条件,即 t<0t\lt 0 时若 x(t)=0x(t)=0 则有 y(t)=0y(t)=0 ,或 n<0n\lt 0 时若 x[n]=0x[n]=0 则有 y[n]=0y[n]=0

    对于线性系统,初始松弛条件与满足因果性是等价的

    解题时,可以直接认为就是有 y(0)=0y(0)=0x[0]=0x[0]=0

    # 线性常系数差分方程

    一般情况 NMN\geq M , N 阶非其次常系数微分方程的一般形式

    k=0Naky[nk]=k=0Mbkx[nk]\sum_{k=0}^{N}a_k y[n-k]=\sum_{k=0}^{M}b_k x[n-k]

    其中 ak, bk 均为常数

    还可以将其改写为 ==(正向) 递归形式 ==:

    y[n]=1a0(k=0Mbkx[nk]k=0Naky[nk])y[n]=\frac{1}{a_0}(\sum_{k=0}^{M}b_k x[n-k]-\sum_{k=0}^{N}a_k y[n-k])

    还可将差分方程改写反向递归求解形式

    y[nN]=1aN(k=0Mbkx[nk]k=0N1aky[nk])y[n-N]=\frac{1}{a_N}(\sum_{k=0}^{M}b_k x[n-k]-\sum_{k=0}^{N-1}a_k y[n-k])

    ak=0,k0a_k =0,k\neq 0 时,差分方程变为一个卷积和的形式即非递归方程

    y[n]=k=0Mbka0x[nk]=+h[n]x[nk]y[n]=\sum_{k=0}^{M}\frac{b_k}{a_0}x[n-k]=\sum_{-\infin}^{+\infin}h[n]\ x[n-k]

    其中:

    h[n]={bna0,0nM0,othersh[n]= \begin{cases} \frac{b_n}{a_0},\ \ 0\leq n\leq M \\ 0,\ \ \ \ \text{others} \end{cases}

    • 由于 h [n] 是有限长的,当 ak=0,k0a_k =0,k\neq 0非递归方程描述的系统称为有限脉冲响应( Finite Impulse Response,FIR)系统

    • 而其他情况下递归方程描述的系统结合初始松弛条件具有无限长的单位脉冲响应,因而称为无限脉冲响应(Infinite Impulse Response,IIR)系统

    • 与微分方程类似,差分方程的齐次解通解)与输入信号的形式无关,称为系统的自然响应自由响应

    • 特解具有与输入信号相同的函数形式,称为受迫响应强迫响应

    具有非零起始状态的线性常系数微分或差分方程对应增量线性系统,其响应由两部分组成:

    • 零输入响应,与输入信号无关,属于自然响应
    • 零状态响应,既与输入信号有关,也与系统特性有关,它包含了一部分受迫响应,也包含有一部分自然响应

    # 方框图表示

    # 差分方程描述系统

    对于正向递归形式的 LTI 系统差分方程,其方程中包括三种基本运算:

    • 移位(延迟)—— 对应延迟器
    • 乘系数 —— 对应数乘器
    • 相加 —— 对应加法器

    令:

    w[n]=k=0Mbkx[nk]w[n]=\sum_{k=0}^{M}b_k x[n-k]

    正向递归形式可写为:

    y[n]=1a0(w[n]k=0Naky[nk])y[n]=\frac{1}{a_0}(w[n]-\sum_{k=0}^{N}a_k y[n-k])

    差分-直接I型

    将其级联起来,就成为线性常系数差分方程描述的系统,它具有与差分方程完全相同的运算功能。可以调换级联次序,并将移位单元合并,于是得到直接 II 型

    差分-直接II型

    # 微分方程描述系统

    微分方程也包括三种基本运算:

    • 微分
    • 乘系数
    • 相加

    但微分器不仅在工程实现上有困难,而且对误差及噪声极为敏感,因此,通常使用积分器而不用微分器

    N 阶微分方程两边同时积分 N 次,即可得到对应 N 阶积分方程:

    k=0Naky(Nk)(t)=k=0Mbkx(Nk)(t)\sum_{k=0}^{N}a_k\ y_{(N-k)}(t)=\sum_{k=0}^{M}b_k\ x_{(N-k)}(t)

    对此积分方程完全按照差分方程的办法有其递归方程:

    y(t)=1a0(k=0Mbkx(Nk)(t)k=0Naky(Nk)(t))y(t)=\frac{1}{a_0}(\sum_{k=0}^{M}b_k x_{(N-k)}(t)-\sum_{k=0}^{N}a_k y_{(N-k)}(t))

    令:

    w(t)=k=0Mbkx(Nk)(t)w(t)=\sum_{k=0}^{M}b_k x_{(N-k)}(t)

    可得:

    y(t)=1a0(w(t)k=0Naky(Nk)(t))y(t)=\frac{1}{a_0}(w(t)-\sum_{k=0}^{N}a_k y_{(N-k)}(t))

    积分-直接I型

    通过交换级联次序,合并积分器可得直接 Ⅱ 型

    积分-直接II型

    # 卷积定义奇异函数

    # 单位冲激信号u0(t)u_0(t)

    δ(t)=ddtu(t)\delta(t)=\frac{d}{dt}u(t) 定义下的单位冲激。这是不严密的,因为 u(t)u(t) 在 t=0 处不连续。

    进而采用极限的观点,将 δ(t)\delta(t) 视为 δΔ(t)\delta_Δ (t)Δ0Δ → 0 时的极限。但这种定义或描述 \delta (t) 的方法在数学上仍然是不严格的,因为存在无穷多个不同的函数在 Δ0Δ → 0 时都表现为与δΔ(t)\delta_Δ (t) 有相同的特性。

    极限定义举例1

    极限定义举例2

    上述现象出现的原因在于 \delta (t) 是一个理想化的非常规函数,被称为奇异函数,通常采用卷积或积分运算下表现的特性来定义奇异函数

    从系统的角度,将 δ(t)\delta(t) 定义为恒等系统单位冲激响应,也就是通过卷积定义 δ(t)\delta(t) ,对任意 x(t)x(t)δ(t)\delta(t) 应满足:

    x(t)=x(t)δ(t)x(t)=x(t)*\delta (t)

    单位冲激函数相当于卷积运算系统的单位元

    ⭐️更一般地,应该满足:

    x(h(g(t)))=x(h(t))δ(g(t))x(h(g(t)))=x(h(t))*\delta(g(t))

    根据定义可以得出 δ(t)\delta(t) 的如下性质:

    1=1δ(t)=+δ(τ)dτ1=1*\delta (t) = \int_{-\infin}^{+\infin}\delta(\tau)d\tau

    卷积定义还有一个等价版本 —— 积分定义

    g(t)=g(t)δ(t)=+g(tτ)δ(τ)dτg(0)=+g(t)δ(τ)dτg(-t)=g(-t)*\delta(t)=\int_{-\infin}^{+\infin}g(t-\tau)\delta(\tau)d\tau\\ g(0)=\int_{-\infin}^{+\infin}g(t)\delta(\tau)d\tau

    # 偶函数

    单位冲激信号 \delta (t) 是偶函数:$$\delta (t)=\delta (-t)$$

    证明:

    +δ(t)f(t)dt=+δ(τ)f(τ)d(τ)=+δ(τ)f(τ)dτ=f(0)\begin{aligned} \int_{-\infin}^{+\infin}\delta(-t)f(t)dt &=\int_{-\infin}^{+\infin}\delta(\tau)f(-\tau)d(-\tau)\\ &=\int_{-\infin}^{+\infin}\delta(\tau)f(-\tau)d\tau\\ &=f(0) \end{aligned}

    # 时间尺度变换

    δ(at)=1aδ(t)\delta(at)=\frac{1}{|a|}\delta(t)

    验证信号下方面积:

    a>0a\gt 0 时 :

    +δ(aτ)dτ=1a+δ(aτ)d(aτ)=1a=1a\int_{-\infin}^{+\infin}\delta(a\tau)d\tau=\frac{1}{a}\int_{-\infin}^{+\infin}\delta(a\tau)d(a\tau)=\frac{1}{a}=\frac{1}{|a|}

    a<0a\lt 0 时 :

    +δ(aτ)dτ=1a+δ(aτ)d(aτ)=1a+δ(t)dt=1a+δ(t)dt=1a=1a\begin{aligned} \int_{-\infin}^{+\infin}\delta(a\tau)d\tau &= \frac{1}{a}\int_{-\infin}^{+\infin}\delta(a\tau)d(a\tau)\\ &= \frac{1}{a}\int_{+\infin}^{-\infin}\delta(t)dt \\ &= -\frac{1}{a}\int_{-\infin}^{+\infin}\delta(t)dt \\ &= -\frac{1}{a} \\ &= |\frac{1}{a}| \end{aligned}

    # 单位冲激偶u1(t)u_1(t)

    系统角度定义为理想微分器单位冲激响应,即 \delta (t) 的微分:

    u1(t)=ddtδ(t)u_1(t)=\frac{d}{dt}\delta(t)

    微分器

    单位冲激偶

    卷积定义,从 LTI 系统分析的角度有:

    x(t)u1(t)=ddtx(t)x(t)*u_1 (t)=\frac{d}{dt}x(t)

    该信号可视为短脉冲求导的理想化结果。

    短脉冲求导的理想化

    # 积分性质

    令输入 x(t)=1x(t)=1 有:

    +u1(τ)dτ=+x(tτ)u1(τ)dτ=x(t)u1(t)=ddtx(t)=0\int_{-\infin}^{+\infin}u_1(\tau)d\tau=\int_{-\infin}^{+\infin}x(t-\tau)u_1(\tau)d\tau=x(t)*u_1(t)=\frac{d}{dt}x(t)=0

    # 积分意义下的定义

    +g(τt)u1(τ)dτ=g(t)u1(t)=ddtg(t)\int_{-\infin}^{+\infin}g(\tau-t)u_1(\tau)d\tau=g(-t)*u_1(t)=-\frac{d}{dt}g(-t)

    令 t=0 有:

    g(0)=+g(τ)u1(τ)dτ-g'(0)=\int_{-\infin}^{+\infin}g(\tau)u_1(\tau)d\tau

    该积分即可作为 u1(t)u_1(t) 在积分意义下的定义

    # \delta (t) 的积分u1(t)u_{-1}(t)

    从系统的角度,定义 u1(t)u_{-1}(t)理想积分器单位冲激响应

    u1(t)=tδ(τ)dτ=u(t)u_{-1}(t)=\int_{-\infin}^{t}\delta(\tau)d\tau=u(t)

    运算定义为:

    x(t)u1(t)=tx(τ)dτx(t)*u_{-1}(t)=\int_{-\infin}^{t}x(\tau)d\tau

    # 单位斜坡函数u2(t)u_{-2}(t)

    系统的角度,可以定义为:(卷积)

    u2(t)=u1(t)u1(t)=tu(t)dt=tu(t)u_{-2}(t)=u_{-1}(t)*u_{-1}(t)=\int_{-\infin}^{t}u(t)dt=tu(t)

    运算定义为二重积分:

    x(t)u2(t)=x(t)u2(t)=x(t)u1(t)u1(t)=(tu(σ)dσ)u1(t)=t(τu(σ)dσ)dτ=tτu(σ)dσdτ\begin{aligned} x(t)*u_{-2}(t) &= x(t)*u_{-2}(t)\\ &= x(t)*u_{-1}(t)*u_{-1}(t) \\ &= \big(\int_{-\infin}^{t}u(\sigma)d\sigma\large)*u_{-1}(t) \\ &= \int_{-\infin}^{t}\big(\int_{-\infin}^{\tau}u(\sigma)d\sigma\big)d\tau \\ &= \int_{-\infin}^{t}\int_{-\infin}^{\tau}u(\sigma)\ d\sigma d\tau \end{aligned}

    # 卷积定义推广

    例如 u2(t)u_2(t) 可以定义为:u2(t)=u1(t)u1(t)u_2(t)=u_1(t)*u_1(t) :

    x(t)u2(t)=d2dt2x(t)=x(t)u1(t)u1(t)x(t)*u_2(t)=\frac{d^2}{dt^2}x(t)=x(t)*u_1(t)*u_1(t)

    以此类推:

    u2(t)=u1(t)u1(t)u2(t)=u1(t)u1(t)u1(t)......uk(t)=u1(t)u1(t)...u1(t)k个\begin{aligned} u_2(t) &= u_1(t)*u_1(t) \\ u_2(t) &= u_1(t)*u_1(t)*u_1(t)\\ &......\\ u_k(t)\over &{=\over} {u_1(t)*u_1(t)*...*u_1(t)\over \text{k个}} \end{aligned}

    单位阶跃函数的高阶积分推广:

    u3(t)=u1(t)u1(t)u1(t)=12t2u(t)u4(t)=u1(t)u1(t)u1(t)u1(t)=16t3u(t)......uk(t)=u1(t)u1(t)...u1(t)k个=tk1(k1)!u(t)\begin{aligned} u_{-3}(t) &= u_{-1}(t)*u_{-1}(t)*u_{-1}(t)=\frac{1}{2}t^2 u(t) \\ u_{-4}(t) &= u_{-1}(t)*u_{-1}(t)*u_{-1}(t)*u_{-1}(t)=\frac{1}{6}t^3 u(t)\\ &......\\ u_{-k}(t)\over &{=\over} {u_{-1}(t)*u_{-1}(t)*...*u_{-1}(t)\over \text{k个}}{=\over}{\frac{t^{k-1}}{(k-1)!}u(t)\over} \end{aligned}

    实际上, \delta (t) 的各次积分已经是常规函数了

    \ No newline at end of file +卷积と线性时不变系统 - 信号与系统 - 电子信息 | Balloon Party = 風船のパーティー = fuusen no party

    Linear Time Invariant 线性时不变

    基本信号应具有以下特点:

    • 本身尽可能简单,并且用它们的线性组合能够表示尽可能广泛的信号
    • LTI 系统对这种信号的响应易于计算且有利于对系统的深入理解

    两类有用的基本信号:

    • 复指数信号
    • 单位冲激信号

    本章主要考虑以单位冲激信号为基本信号进行信号与系统分析

    summary

    # 卷积和

    # 定义

    对于任意一个离散时间系统:

    离散时间系统

    设离散时间系统为 LTI 的,定义单位脉冲响应 h[n]h[n] 为以单位冲激信号 δ[n]\delta[n] 为输入时的输出:

    δ[n]h[n]\delta[n]\to h[n]

    由 LTI 时不变性有:

    δ[nk]h[nk]\delta[n-k]\to h[n-k]

    由 LTI 线性有:

    x[n]=k=+x[k]δ[nk]y[n]=k=+x[k]h[nk]x[n]=\sum_{k=-\infin}^{+\infin}x[k]\delta[n-k]\to y[n]=\sum_{k=-\infin}^{+\infin}x[k]h[n-k]

    这种求得系统响应的运算关系称为卷积和,即离散时间 LTI 系统的响应可以用卷积和表示为:

    y[n]=x[n]h[n]=k=+x[k]h[nk]y[n]=x[n]*h[n]=\sum_{k=-\infin}^{+\infin}x[k]h[n-k]

    ⭐️更一般的形式:

    x[nn0n1]=x[nn0]h[nn1]x[n-n_0-n_1]=x[n-n_0]*h[n-n_1]

    ⭐️只要得到了 LTI 系统对 δ[n]\delta[n] 的 响 应 h[n]h[n] —— 单位脉冲响应,就可以得到 LTI 系统对任何输入信号 x[n]x[n] 的响应。

    换言之,一个 LTI 系统可以完全由它的单位脉冲响应 h[n]h[n] 来表征。

    # 特殊离散 LTI 系统

    # 延时器

    h[n]=δ[nn0]h[n]=\delta[n-n_0]

    具有该单位脉冲响应的系统一般情况下存在多种可能性,但是对应的 LTI 系统只有一个:

    y[n]=x[nn0]=x[n]δ[nn0]y[n]=x[n-n_0]=x[n]*\delta[n-n_0]

    # 累加器

    h[n]=k=nδ[k]=u[n]h[n]=\sum_{k=-\infin}^{n}\delta[k]=u[n]

    对应的 LTI 系统:

    y[n]=x[n]u[n]=k=nx[k]y[n]=x[n]*u[n]=\sum_{k=-\infin}^{n}x[k]

    # 计算

    选择 nn 值并固定,求和项视为 kk 的函数,步骤如下:

    1. x[n]x[n]h[n]h[n] 变为 x[k]x[k]h[k]h[k] ,以 kk 为求和变量
    2. h[k]h[k] 翻转变为 h[k]h[-k]
    3. h[k]h[-k] 平移 nn,变为 h[nk]h[n - k]
    4. x[k]x[k]h[nk]h[n-k] 相乘;
    5. 对乘积 x[k]h[nk]x[k]h[n-k]kZk\in \mathbb{Z} 取和;
      • 一般都转化为只对非 0 区间积分。注意非 0 区间上下限!!!
    6. 改变 nn 值并重复 3–5 步,直到 nn 超过了 x[n]x[n] 非零部分定义域

    注意很多题目最后答案会加上一个 u(n)u(n) ,是因为只求定义域在 [0,+)[0,+\infin) 的部分

    # 卷积积分

    # 定义

    类似积分思想,转化为无限多的离散和的极限

    用冲激信号表示连续时间信号:

    将输入信号 x(t)x(t) 近似表示为移位、放缩后的脉冲之和

    脉冲和近似表示输入信号

    xΔ(t)=x(kΔ),kΔ<t<(k+1)Δx_{\Delta}(t)=x(k\Delta)\ ,\ k\Delta\lt t\lt(k+1)\Delta

    阶梯近似表达过程:

    设连续时间系统为 LTI 的,定义(名字待补,与卷积和中的单位脉冲响应对应) hΔ(n)h_{\Delta}(n) 为以单位冲激信号 δΔ(n)\delta_{\Delta}(n) 为输入时的输出:

    δΔ(n)hΔ(n)\delta_{\Delta}(n)\to h_{\Delta}(n)

    阶梯近似表达

    由 LTI 时不变性有:

    δΔ(nkΔ)hΔ(nkΔ)\delta_{\Delta}(n-k\Delta)\to h_{\Delta}(n-k\Delta)

    由 LTI 线性齐次性尺度调整:

    x(kΔ)δΔ(nkΔ)x(kΔ)hΔ(nkΔ)x(k\Delta)\delta_{\Delta}(n-k\Delta)\to x(k\Delta)h_{\Delta}(n-k\Delta)

    阶梯近似表达2

    由 LTI 线性叠加性有:

    k=+x(kΔ)δΔ(nkΔ)Δk=+x(kΔ)hΔ(nkΔ)ΔxΔ(t)yΔ(t)\sum_{k=-\infin}^{+\infin}x(k\Delta)\delta_{\Delta}(n-k\Delta)\Delta\to \sum_{k=-\infin}^{+\infin}x(k\Delta)h_{\Delta}(n-k\Delta)\Delta\\ x_{\Delta}(t)\to y_{\Delta}(t)

    当 Δ → 0 时 :

    • xΔ(t)x(t)x_Δ(t) → x(t)
    • yΔ(t)y(t)y_Δ(t) → y(t)
    • kΔτkΔ → \tau
    • ΔdτΔ → d\tau
    • δΔ(tkΔ)δ(tτ)\delta_Δ(t - kΔ) → \delta(t - \tau)
    • hΔ(tkΔ)h(tτ)h_Δ(t - kΔ) → h(t - \tau)
    • \sum\to\int

    从而得到:

    x(t)=+x(τ)δ(tτ)dτ=x(t)δ(t)y(t)=+x(τ)h(tτ)dτ=x(t)h(t)x(t)=\int_{-\infin}^{+\infin}x(\tau)\delta(t-\tau)d\tau=x(t)*\delta(t)\\ y(t)=\int_{-\infin}^{+\infin}x(\tau)h(t-\tau)d\tau=x(t)*h(t)

    • 其中第一个式子意义是:用位移的单位冲激信号表示任意的连续时间信号
    • 其中第一个式子意义是:用卷积积分表示连续时间 LTI 系统的响应

    阶梯近似图解过程:

    阶梯近似图解过程

    # 计算

    求卷积积分 x(t)h(t)=+x(τ)h(tτ)dτx(t) * h(t) = \int_{-∞}^{+∞}x(\tau)h(t - \tau)d\tau 的过程与卷积和计算过程相似。卷积积分的计算步骤如下:

    1. x(t)x(t)h(t)h(t) 变为 x(τ)x(\tau)h(τ)h(\tau) ,以 τ\tau 为积分变量;
    2. h(τ)h(\tau) 翻转变为 h(τ)h(-\tau)
    3. h(τ)h(-\tau) 平移 tt ,变为 h(tτ)h(t - \tau)
    4. x(τ)x(\tau)h(tτ)h(t-\tau) 相乘;
    5. 求区间 (,+)(-\infin,+\infin) 上乘积 x(τ)h(tτ)x(\tau)h(t-\tau) 下的面积;
      • 一般都转化为只对非 0 区间积分。注意非 0 区间上下限!!!
    6. 改变 tt 值并重复 3–5 步。

    # 特殊连续 LTI 系统

    # 积分器

    单位冲激信号为输入下系统的响应为:

    h(t)=u(t)h(t)=u(t)

    对任意激励信号 x(t)x(t) ,系统响应为:

    y(t)=x(t)h(t)=+x(τ)h(tτ)dτ=tx(τ)dτy(t)=x(t)*h(t)=\int_{-\infin}^{+\infin}x(\tau)h(t-\tau)d\tau = \int_{-\infin}^{t}x(\tau)d\tau

    # 卷积和与卷积积分的性质

    # 交换律

    变量替换:

    x[n]h[n]=k=+x[k]h[nk]=l=+x[nl]h[l]=h[n]x[n]x[n]*h[n]=\sum_{k=-\infin}^{+\infin}x[k]h[n-k]=\sum_{l=-\infin}^{+\infin}x[n-l]h[l]=h[n]*x[n]

    x(n)h(n)=+x(τ)h(tτ)dτ=+x(tτ)h(τ)dτ=h(t)x(t)x(n)*h(n)=\int_{-\infin}^{+\infin}x(\tau)h(t-\tau)d\tau=\int_{-\infin}^{+\infin}x(t-\tau')h(\tau')d\tau'=h(t)*x(t)

    交换律

    s[n]=u[n]h[n]=h[n]u[n]=k=nh[k]s[n]=u[n]*h[n]=h[n]*u[n]=\sum_{k=-\infin}^{n}h[k]

    利用 u[n]u[n] 是累加器的单位脉冲响应以及交换律可以得到:

    该系统的单位脉冲响应是 h[n]h[n]u[n]u[n] 单位阶跃信号作为输入,则其单位阶跃响应 s[n]s[n] 等价于 以 h[n]h[n] 作为输入时,累加器的响应

    # 分配律

    x[n](h1[n]+h2[n])=x[n]h1[n]+x[n]h2[n]x(t)(h1(t)+h2(t))=x(t)h1(t)+x(t)h2(t)x[n] * (h_1[n] + h_2[n]) = x[n] * h_1[n] + x[n] * h_2[n]\\ x(t) * (h_1(t) + h_2(t)) = x(t) * h_1(t) + x(t) * h_2(t)

    分配律

    # 结合律

    (x[n]h1[n])h2[n]=x[n](h1[n]h2[n])(x(t)h1(t))h2(t)=x(t)(h1(t)h2(t))(x[n] * h_1 [n] ) * h_2[n] = x[n] * (h_1 [n] * h_2[n])\\ (x(t) * h_1 (t) ) * h_2(t) = x(t) * (h_1(t)* h_2 (t))

    结合律

    级联的 LTI 系统其总的单位冲激 (脉冲) 响应等于各子系统单位冲激 (脉冲) 响应的卷积

    # 级联可交换

    卷积运算满足结合律和交换律,因此,系统级联的先后次序可以调换。

    x[n]h1[n]h2[n]=x[n]h2[n]h1[n]x(t)h1(t)h2(t)=x(t)h2(t)h1(t)x[n] * h_1[n] * h_2[n] = x[n] * h_2[n] * h_1[n]\\ x(t) * h_1(t) * h_2(t) = x(t) * h_2(t) * h_1(t)

    级联可交换

    级联次序可交换的前提条件:

    • 系统是 LTI 的
    • 所有涉及到的卷积运算必须收敛

    # 微分、积分及时移特性

    卷积积分满足微分、积分及时移特性。若 x(t)h(t)=y(t)x(t)*h(t)=y(t),则:

    # 微分特性

    x(t)h(t)=x(t)h(t)=y(t)x'(t) * h(t) = x(t) * h'(t) = y'(t)

    x(m)(t)h(n)(t)=y(m+n)(t)x^{(m)}(t) * h^{(n)}(t) = y^{(m+n)}(t)

    tt 求导而不是对 τ\tau 求导

    # 积分特性

    (tx(τ)dτ)h(t)=+(τ1x(τ)dτ)h(tτ1)dτ1=x(t)(th(τ)dτ)=ty(τ)dτ\begin{aligned} \big(\int_{-\infin}^{t}x(\tau)d\tau\big)*h(t)&= \int_{-\infin}^{+\infin}\big(\int_{-\infin}^{\tau_1}x(\tau)d\tau\big)h(t-\tau_1)d\tau_1\\ &=x(t)*\big(\int_{-\infin}^{t}h(\tau)d\tau\big)\\ &=\int_{-\infin}^{t}y(\tau)d\tau \end{aligned}

    # 时移特性

    x(tt0)h(tt1)=x(tt1)h(tt0)=y(tt0t1)x(t - t_0) * h(t-t_1) = x(t-t_1) * h(t - t_0) = y(t - t_0-t_1)

    # 差分、求和及时移特性

    卷积和满足差分、求和及时移特性。若 x [n] * h [n] = y [n],则:

    # 差分特性

    x[n]x[n1]h[n]=x[n](h[n]h[n1])=y[n]y[n1]x[n]- x[n- 1] * h[n]= x[n]* (h[n]- h[n - 1])= y[n] - y[n - 1]

    # 求和特性

    (k=nx[k])h[n]=x[n](k=nh[k])=k=ny[k](\sum_{k=-∞}^{n}x[k] )* h[n] = x[n] * (\sum_{k=-∞}^{n}h[k] )=\sum_{k=-∞}^{n}y[k]

    # 时移特性

    x[tt0]h[tt1]=x[tt1]h[tt0]=y[tt0t1]x[t - t_0] * h[t-t_1] = x[t-t_1]* h[t - t_0]= y[t - t_0-t_1]

    # LTI 系统的性质

    # 记忆性

    对于离散时间 LTI 系统:

    y[n]=k=+x[k]h[nk]y[n]=\sum_{k=-\infin}^{+\infin}x[k]h[n-k]

    如果系统是无记忆的,则在任何时刻 n ,y [n] 都只能和 n 时刻的输入有关,等式右边的和式中只能有 k = n 时的一项非零,即必须有 h [n - k] = 0 , k ≠ n,也即:

    h[n]=0,n0h[n] = 0 ,\ \ \ n\neq0

    所以,离散时间无记忆系统的单位脉冲响应为:

    h[n]=kδ[n]h[n]=k\delta[n]

    即为离散时间系统无记忆充要条件

    类似地,连续时间系统无记忆的充要条件为:

    h(n)=kδ[(n)h(n)=k\delta[(n)

    当 k = 1 时系统是恒等系统。

    # 可逆性

    如果 LTI 系统是可逆的,则其逆系统也是 LTI 系统。

    互逆的两个系统的满足(充要条件卷积和等于单位冲激信号

    h(t)g(t)=δ(t)h[n]g[n]=δ[n]\begin{aligned} h(t) * g(t) =\delta(t)\\ h[n] * g[n] =\delta[n] \end{aligned}

    从而有:

    x(t)h(t)g(t)=x(t)δ(t)=x(t)x[t]h[n]g[n]=x[t]δ[n]=x[t]\begin{aligned} x(t)*h(t)*g(t)=x(t)*\delta(t)=x(t)\\ x[t]*h[n]*g[n]=x[t]*\delta[n]=x[t] \end{aligned}

    可逆性

    例如:延时器是可逆的 LTI 系统,h (t) = \delta (t - t0),其逆系统是 g (t) = \delta (t + t0),显然有:

    h(t)g(t)=(tt0)δ(t+t0)=δ(t)h(t) * g(t) = (t - t_0) * \delta(t + t_0) = \delta(t)

    又如:累加器是可逆的 LTI 系统,其 h [n] = u [n],其逆系统是 g [n] = \delta [n] - \delta [n - 1],显然也有:

    h[n]g[n]=u[n](δ[n]δ[n1])=u[n]u[n1]=δ[n]\begin{aligned} h[n] * g[n] &= u[n] * (\delta[n] - \delta[n - 1])\\ &= u[n] - u[n - 1] = \delta[n] \end{aligned}

    # 因果性

    对于离散时间 LTI 系统:

    y[n]=k=+x[k]h[nk]y[n]=\sum_{k=-\infin}^{+\infin}x[k]h[n-k]

    因果性意味着上式等号右边和式中所有 k > n 的项都必须为零,即 h [n - k] = 0,k > n,也即:

    h[n]=0,n<0h[n] = 0,\ \ \ n < 0

    同理对连续时间因果系统有:

    h(t)=0,t<0h(t) = 0, \ \ \ t < 0

    即为 LTI 系统具有因果性的充要条件

    对于线性系统而言,因果性与初始松弛条件等价。所谓系统满足初始松弛条件是指该系统具有如下表现:对于任意时刻,只要该时刻之前系统的输入为 0,则该时刻之前系统的输出也必然为 0。

    (参考教材习题 1.44 (a))

    – 该等价性的前提是系统为线性的。否则不一定成立。例如 y n = 2x n + 3 是因果的,但显然不满足初始松弛条件。

    # 稳定性

    稳定性意味着有界输入必然产生有界输出。以离散时间 LTI 系统为例,设 |x [n]| 有界,| x [n - k]| ≤ A。有:

    y[n]=k=+h[k]x[nk]k=+h[k]x[nk]Ak=+h[k]|y[n]| = |\sum_{k=-∞}^{+∞}h[k]x[n - k]| ≤ \sum_{k=-∞}^{+∞} |h[k]| |x[n - k]| ≤ A \sum_{k=-∞}^{+∞}|h[k]|

    离散 LTI 系统稳定的充分必要条件为:

    n=+h[n]<+\sum_{n=-\infin}^{+\infin}|h[n]|\lt +\infin

    类似地有连续 LTI 系统稳定的充分必要条件为:

    +h(t)dt<+\int_{-\infin}^{+\infin}|h(t)|dt \lt +\infin

    # 单位阶跃响应

    在工程实际中,也常用单位阶跃响应来描述 LTI 系统。单位阶跃响应就是系统对 u (t) 或 u [n] 所产生的响应,记为 s (t) 或 s [n]。因此有:

    s[n]=u[n]h[n]=k=nh[k]s[n] = u[n] * h[n] = \sum_{k=-∞}^n h[k]

    h[n]=s[n]s[n1]h[n] = s[n] - s[n - 1]

    s(t)=u(t)h(t)=th(τ)dτs(t) = u(t) * h(t) = \int_{-\infin}^{t}h(\tau)d\tau

    h(t)=ds(t)dt=s(t)h(t) =\frac{ds(t)}{dt}= s'(t)

    # 用微分和差分方程描述的因果 LTI 系统

    在工程实际中有相当普遍的一类系统,其数学模型可以用线性常系数微分方程或线性常系数差分方程来描述。

    在时域内分析这类 LTI 系统,就是要求解线性常系数微分方程或差分方程。

    # 线性常系数微分方程

    一般情况 NMN\geq M , N 阶非其次常系数微分方程的一般形式:

    k=0Nakdky(t)dtk=k=0Mbkdky(t)dtk\sum_{k=0}^{N}a_k\frac{d^ky(t)}{dt^k}=\sum_{k=0}^{M}b_k\frac{d^ky(t)}{dt^k}

    其中 ak, bk 均为常数

    求解该微分方程(求 y (t) 的显式表达)的步骤:

    1. 求出齐次解通解)yh (t) ,也称为系统的自然响应自由响应
      1. 建立特征方程求出特征根 𝜆k
      2. 根据特征根得到齐次解形式
      3. 根据附加条件确定齐次解形式中的待定常数 Ck
    2. 特解 y𝑝(t),也称系统的受迫响应强迫响应,其形式与输入信号有关
      1. 根据输入信号形式得到特解形式
      2. 将输入信号与特解代入微分方程确定特解形式中的待定常数
    3. 得到完全解 y (t) = yh(t) + y𝑝(t)

    附加条件常给出为初始松弛条件,即 t<0t\lt 0 时若 x(t)=0x(t)=0 则有 y(t)=0y(t)=0 ,或 n<0n\lt 0 时若 x[n]=0x[n]=0 则有 y[n]=0y[n]=0

    对于线性系统,初始松弛条件与满足因果性是等价的

    解题时,可以直接认为就是有 y(0)=0y(0)=0x[0]=0x[0]=0

    # 线性常系数差分方程

    一般情况 NMN\geq M , N 阶非其次常系数微分方程的一般形式

    k=0Naky[nk]=k=0Mbkx[nk]\sum_{k=0}^{N}a_k y[n-k]=\sum_{k=0}^{M}b_k x[n-k]

    其中 ak, bk 均为常数

    还可以将其改写为 ==(正向) 递归形式 ==:

    y[n]=1a0(k=0Mbkx[nk]k=0Naky[nk])y[n]=\frac{1}{a_0}(\sum_{k=0}^{M}b_k x[n-k]-\sum_{k=0}^{N}a_k y[n-k])

    还可将差分方程改写反向递归求解形式

    y[nN]=1aN(k=0Mbkx[nk]k=0N1aky[nk])y[n-N]=\frac{1}{a_N}(\sum_{k=0}^{M}b_k x[n-k]-\sum_{k=0}^{N-1}a_k y[n-k])

    ak=0,k0a_k =0,k\neq 0 时,差分方程变为一个卷积和的形式即非递归方程

    y[n]=k=0Mbka0x[nk]=+h[n]x[nk]y[n]=\sum_{k=0}^{M}\frac{b_k}{a_0}x[n-k]=\sum_{-\infin}^{+\infin}h[n]\ x[n-k]

    其中:

    h[n]={bna0,0nM0,othersh[n]= \begin{cases} \frac{b_n}{a_0},\ \ 0\leq n\leq M \\ 0,\ \ \ \ \text{others} \end{cases}

    • 由于 h [n] 是有限长的,当 ak=0,k0a_k =0,k\neq 0非递归方程描述的系统称为有限脉冲响应( Finite Impulse Response,FIR)系统

    • 而其他情况下递归方程描述的系统结合初始松弛条件具有无限长的单位脉冲响应,因而称为无限脉冲响应(Infinite Impulse Response,IIR)系统

    • 与微分方程类似,差分方程的齐次解通解)与输入信号的形式无关,称为系统的自然响应自由响应

    • 特解具有与输入信号相同的函数形式,称为受迫响应强迫响应

    具有非零起始状态的线性常系数微分或差分方程对应增量线性系统,其响应由两部分组成:

    • 零输入响应,与输入信号无关,属于自然响应
    • 零状态响应,既与输入信号有关,也与系统特性有关,它包含了一部分受迫响应,也包含有一部分自然响应

    # 方框图表示

    # 差分方程描述系统

    对于正向递归形式的 LTI 系统差分方程,其方程中包括三种基本运算:

    • 移位(延迟)—— 对应延迟器
    • 乘系数 —— 对应数乘器
    • 相加 —— 对应加法器

    令:

    w[n]=k=0Mbkx[nk]w[n]=\sum_{k=0}^{M}b_k x[n-k]

    正向递归形式可写为:

    y[n]=1a0(w[n]k=0Naky[nk])y[n]=\frac{1}{a_0}(w[n]-\sum_{k=0}^{N}a_k y[n-k])

    差分-直接I型

    将其级联起来,就成为线性常系数差分方程描述的系统,它具有与差分方程完全相同的运算功能。可以调换级联次序,并将移位单元合并,于是得到直接 II 型

    差分-直接II型

    # 微分方程描述系统

    微分方程也包括三种基本运算:

    • 微分
    • 乘系数
    • 相加

    但微分器不仅在工程实现上有困难,而且对误差及噪声极为敏感,因此,通常使用积分器而不用微分器

    N 阶微分方程两边同时积分 N 次,即可得到对应 N 阶积分方程:

    k=0Naky(Nk)(t)=k=0Mbkx(Nk)(t)\sum_{k=0}^{N}a_k\ y_{(N-k)}(t)=\sum_{k=0}^{M}b_k\ x_{(N-k)}(t)

    对此积分方程完全按照差分方程的办法有其递归方程:

    y(t)=1a0(k=0Mbkx(Nk)(t)k=0Naky(Nk)(t))y(t)=\frac{1}{a_0}(\sum_{k=0}^{M}b_k x_{(N-k)}(t)-\sum_{k=0}^{N}a_k y_{(N-k)}(t))

    令:

    w(t)=k=0Mbkx(Nk)(t)w(t)=\sum_{k=0}^{M}b_k x_{(N-k)}(t)

    可得:

    y(t)=1a0(w(t)k=0Naky(Nk)(t))y(t)=\frac{1}{a_0}(w(t)-\sum_{k=0}^{N}a_k y_{(N-k)}(t))

    积分-直接I型

    通过交换级联次序,合并积分器可得直接 Ⅱ 型

    积分-直接II型

    # 卷积定义奇异函数

    # 单位冲激信号u0(t)u_0(t)

    δ(t)=ddtu(t)\delta(t)=\frac{d}{dt}u(t) 定义下的单位冲激。这是不严密的,因为 u(t)u(t) 在 t=0 处不连续。

    进而采用极限的观点,将 δ(t)\delta(t) 视为 δΔ(t)\delta_Δ (t)Δ0Δ → 0 时的极限。但这种定义或描述 \delta (t) 的方法在数学上仍然是不严格的,因为存在无穷多个不同的函数在 Δ0Δ → 0 时都表现为与δΔ(t)\delta_Δ (t) 有相同的特性。

    极限定义举例1

    极限定义举例2

    上述现象出现的原因在于 \delta (t) 是一个理想化的非常规函数,被称为奇异函数,通常采用卷积或积分运算下表现的特性来定义奇异函数

    从系统的角度,将 δ(t)\delta(t) 定义为恒等系统单位冲激响应,也就是通过卷积定义 δ(t)\delta(t) ,对任意 x(t)x(t)δ(t)\delta(t) 应满足:

    x(t)=x(t)δ(t)x(t)=x(t)*\delta (t)

    单位冲激函数相当于卷积运算系统的单位元

    ⭐️更一般地,应该满足:

    x(h(g(t)))=x(h(t))δ(g(t))x(h(g(t)))=x(h(t))*\delta(g(t))

    根据定义可以得出 δ(t)\delta(t) 的如下性质:

    1=1δ(t)=+δ(τ)dτ1=1*\delta (t) = \int_{-\infin}^{+\infin}\delta(\tau)d\tau

    卷积定义还有一个等价版本 —— 积分定义

    g(t)=g(t)δ(t)=+g(tτ)δ(τ)dτg(0)=+g(t)δ(τ)dτg(t)=g(t)*\delta(t)=\int_{-\infin}^{+\infin}g(t-\tau)\delta(\tau)d\tau\\ g(0)=\int_{-\infin}^{+\infin}g(t)\delta(\tau)d\tau

    # 偶函数

    单位冲激信号 δ(t)\delta(t) 是偶函数:

    δ(t)=δ(t)\delta(t)=\delta(-t)

    证明:

    +δ(t)f(t)dt=+δ(τ)f(τ)d(τ)=+δ(τ)f(τ)dτ=f(0)\begin{aligned} \int_{-\infin}^{+\infin}\delta(-t)f(t)dt &=\int_{-\infin}^{+\infin}\delta(\tau)f(-\tau)d(-\tau)\\ &=\int_{-\infin}^{+\infin}\delta(\tau)f(-\tau)d\tau\\ &=f(0) \end{aligned}

    # 时间尺度变换

    δ(at)=1aδ(t)\delta(at)=\frac{1}{|a|}\delta(t)

    验证信号下方面积:

    a>0a\gt 0 时 :

    +δ(aτ)dτ=1a+δ(aτ)d(aτ)=1a=1a\int_{-\infin}^{+\infin}\delta(a\tau)d\tau=\frac{1}{a}\int_{-\infin}^{+\infin}\delta(a\tau)d(a\tau)=\frac{1}{a}=\frac{1}{|a|}

    a<0a\lt 0 时 :

    +δ(aτ)dτ=1a+δ(aτ)d(aτ)=1a+δ(t)dt=1a+δ(t)dt=1a=1a\begin{aligned} \int_{-\infin}^{+\infin}\delta(a\tau)d\tau &= \frac{1}{a}\int_{-\infin}^{+\infin}\delta(a\tau)d(a\tau)\\ &= \frac{1}{a}\int_{+\infin}^{-\infin}\delta(t)dt \\ &= -\frac{1}{a}\int_{-\infin}^{+\infin}\delta(t)dt \\ &= -\frac{1}{a} \\ &= |\frac{1}{a}| \end{aligned}

    # 单位冲激偶u1(t)u_1(t)

    系统角度定义为理想微分器单位冲激响应,即 δ(t)\delta(t) 的微分:

    u1(t)=ddtδ(t)u_1(t)=\frac{d}{dt}\delta(t)

    微分器

    单位冲激偶

    卷积定义,从 LTI 系统分析的角度有:

    x(t)u1(t)=ddtx(t)x(t)*u_1 (t)=\frac{d}{dt}x(t)

    该信号可视为短脉冲求导的理想化结果。

    短脉冲求导的理想化

    # 积分性质

    令输入 x(t)=1x(t)=1 有:

    +u1(τ)dτ=+x(tτ)u1(τ)dτ=x(t)u1(t)=ddtx(t)=0\int_{-\infin}^{+\infin}u_1(\tau)d\tau=\int_{-\infin}^{+\infin}x(t-\tau)u_1(\tau)d\tau=x(t)*u_1(t)=\frac{d}{dt}x(t)=0

    # 积分意义下的定义

    +g(τt)u1(τ)dτ=g(t)u1(t)=ddtg(t)\int_{-\infin}^{+\infin}g(\tau-t)u_1(\tau)d\tau=g(-t)*u_1(t)=-\frac{d}{dt}g(-t)

    t=0t=0 有:

    g(0)=+g(τ)u1(τ)dτ-g'(0)=\int_{-\infin}^{+\infin}g(\tau)u_1(\tau)d\tau

    该积分即可作为 u1(t)u_1(t) 在积分意义下的定义

    # δ(t)\delta(t) 的积分u1(t)u_{-1}(t)

    从系统的角度,定义 u1(t)u_{-1}(t)理想积分器单位冲激响应

    u1(t)=tδ(τ)dτ=u(t)u_{-1}(t)=\int_{-\infin}^{t}\delta(\tau)d\tau=u(t)

    运算定义为:

    x(t)u1(t)=tx(τ)dτx(t)*u_{-1}(t)=\int_{-\infin}^{t}x(\tau)d\tau

    # 单位斜坡函数u2(t)u_{-2}(t)

    系统的角度,可以定义为:(卷积)

    u2(t)=u1(t)u1(t)=tu(t)dt=tu(t)u_{-2}(t)=u_{-1}(t)*u_{-1}(t)=\int_{-\infin}^{t}u(t)dt=tu(t)

    运算定义为二重积分:

    x(t)u2(t)=x(t)u2(t)=x(t)u1(t)u1(t)=(tu(σ)dσ)u1(t)=t(τu(σ)dσ)dτ=tτu(σ)dσdτ\begin{aligned} x(t)*u_{-2}(t) &= x(t)*u_{-2}(t)\\ &= x(t)*u_{-1}(t)*u_{-1}(t) \\ &= \big(\int_{-\infin}^{t}u(\sigma)d\sigma\large)*u_{-1}(t) \\ &= \int_{-\infin}^{t}\big(\int_{-\infin}^{\tau}u(\sigma)d\sigma\big)d\tau \\ &= \int_{-\infin}^{t}\int_{-\infin}^{\tau}u(\sigma)\ d\sigma d\tau \end{aligned}

    # 卷积定义推广

    例如 u2(t)u_2(t) 可以定义为:u2(t)=u1(t)u1(t)u_2(t)=u_1(t)*u_1(t) :

    x(t)u2(t)=d2dt2x(t)=x(t)u1(t)u1(t)x(t)*u_2(t)=\frac{d^2}{dt^2}x(t)=x(t)*u_1(t)*u_1(t)

    以此类推:

    u2(t)=u1(t)u1(t)u2(t)=u1(t)u1(t)u1(t)......uk(t)=u1(t)u1(t)...u1(t)k个\begin{aligned} u_2(t) &= u_1(t)*u_1(t) \\ u_2(t) &= u_1(t)*u_1(t)*u_1(t)\\ &......\\ u_k(t)\over &{=\over} {u_1(t)*u_1(t)*...*u_1(t)\over \text{k个}} \end{aligned}

    单位阶跃函数的高阶积分推广:

    u3(t)=u1(t)u1(t)u1(t)=12t2u(t)u4(t)=u1(t)u1(t)u1(t)u1(t)=16t3u(t)......uk(t)=u1(t)u1(t)...u1(t)k个=tk1(k1)!u(t)\begin{aligned} u_{-3}(t) &= u_{-1}(t)*u_{-1}(t)*u_{-1}(t)=\frac{1}{2}t^2 u(t) \\ u_{-4}(t) &= u_{-1}(t)*u_{-1}(t)*u_{-1}(t)*u_{-1}(t)=\frac{1}{6}t^3 u(t)\\ &......\\ u_{-k}(t)\over &{=\over} {u_{-1}(t)*u_{-1}(t)*...*u_{-1}(t)\over \text{k个}}{=\over}{\frac{t^{k-1}}{(k-1)!}u(t)\over} \end{aligned}

    实际上, \delta (t) 的各次积分已经是常规函数了

    \ No newline at end of file diff --git a/ie/signals-and-systems/signals-concepts/index.html b/ie/signals-and-systems/signals-concepts/index.html index a1fbf7f6..1fcfc27d 100644 --- a/ie/signals-and-systems/signals-concepts/index.html +++ b/ie/signals-and-systems/signals-concepts/index.html @@ -1,4 +1,4 @@ -信号基本概念 - 信号与系统 - 电子信息 | Balloon Party = 風船のパーティー = fuusen no party

    # 能量 / 功率 信号

    能量功率信号.png

    # 能量信号

    能量为一有界值,即 0 < E < + ∞,此时 P = 0

    有时求 P 不好求,但是 E 好求,可以证明 E 为有界的,从而得到 P = 0

    # 功率信号

    功率为一有界值,即 0 < P < + ∞ ,此时 E = ∞

    有时求 E 不好求,但是 P 好求,可以证明 P 为有界的,从而得到 E = ∞

    # 信号的分解

    # 偶部与奇部

    f(t)=fe(t)+fo(t)f(t)=f_e(t)+f_o(t)

    偶部:

    fe(t)=12[f(t)+f(t)]f_e(t)=\frac{1}{2}[f(t)+f(-t)]

    奇部:

    fo(t)=12[f(t)f(t)]f_o(t)=\frac{1}{2}[f(t)-f(-t)]

    # 实部与虚部

    针对复函数而言

    f(t)=Re{f(t)}+Im{f(t)}f(t)=Re\{f(t)\}+Im\{f(t)\}

    实部:

    Re{f(t)}Re\{f(t)\}

    虚部:

    Im{f(t)}Im\{f(t)\}

    # 连续 / 离散 (时间) 信号

    区分 模拟 / 数字 信号,模拟 / 数字 信号的划分是依据 “值域” 是否连续,而 连续 / 离散 时间信号的划分是依据定义域是否连续

    连续时间信号 包含 模拟信号 和 数字 (量化) 信号

    # 周期信号定义

    • 连续周期信号 : f(t+mT)=f(t)f(t+mT)=f(t) , T 为周期 m = 0,±1,±2,±3...
    • 离散周期信号 : f(k+mN)=f(k)f(k+mN) = f(k) , N 为周期 m = 0,±1,±2,±3...

    T N 趋于⽆无穷时,周期信号变为⾮非周期信号。

    # 连续时间信号

    定义域连续

    在一定时间范围内,除若⼲不连续点之外,对任意时刻函数都有确定的函数值。 “连续” 的含义指定义域连续,⽽连续信号中可以含有不连续点。

    # 普通连续信号

    普通连续信号是指函数的定义域和值域没有不连续点 (即跳变点) 的信号。

    # (实) 指数信号

    f(t)=Keαtf (t) = Ke^{αt}

    指数函数求导、积分后仍为指数函数。

    # 正弦信号

    利用正弦函数与余弦函数表示的信号统称为正弦信号,可表示为

    f(t)=Acos(ωtθ)f(t)=Acos(\omega t-\theta)

    正弦信号求导、积分后仍然为正弦信号。

    指数信号可以转化为正弦信号:

    ejωt=cosωt+jsinωtejωt=cosωtjsinωte^{j\omega t}=cos\ \omega t+j\ sin\ \omega t \\ e^{-j\omega t}=cos\ \omega t-j\ sin\ \omega t

    指数信号可以转化为正弦信号:

    sinωt=12j(ejωtejωt)cosωt=12(ejωt+ejωt)sin\ \omega t = \frac{1}{2j}( e^{j\omega t} - e^{-j\omega t})\\ cos\ \omega t = \frac{1}{2}( e^{j\omega t} + e^{-j\omega t})

    # 复指数信号

    f(t)=Kest=Ke(σ+jω)t=Keσt(cosωt+jsinωt)f (t) = Ke^{st} = Ke^{(σ+j\omega)t} = Ke^{σt}(cos\ ωt + j\ sin\ ωt),

    其中 s=σ+jω0s=σ+j\omega_0 为复数

    复指数信号中,σ\sigma 决定增长衰减,ω0\omega_0 决定震荡快慢;

    σ\sigma 的大小ω0\omega_0 的大小最终结果
    σ=1\sigma=1ω00\omega_0\neq0实部虚部等幅震荡
    σ>1\sigma>1ω00\omega_0\neq0实部虚部增⻓震荡
    σ<1\sigma<1ω00\omega_0\neq0实部虚部衰减震荡
    \sigma≠0ω0=0\omega_0=0实部虚部为指数信号
    σ=0\sigma=0ω0=0\omega_0=0为直流信号
    # 连续复指数信号的周期性

    复指数信号是周期信号当且仅当 σ\sigma = 1 且 ω00\omega_0\neq 0

    基波周期:T=2πω0T=|\frac{2\pi}{\omega_0}|

    基波频率:ω0\omega_0

    # 成谐波关系的连续复指数信号集

    {φk(t)}={ejkω0t}k=0,±1,±2......\{\varphi_k(t) \}=\{ e^{jk\omega_0 t} \}\\ k=0,\pm1,\pm2......

    一般假设 ω0>0\omega_0 \gt 0 ,该集合中的每个信号都是周期的,它们的基波频率分别为 kω0k\omega_0 ,都是 ω0\omega_0 的整数倍,因而称它们是成谐波关系的。

    各次谐波的基波周期分别为 Tk=2πkω0T_k =\frac{2\pi}{|k\omega_0|} ,它们的公共周期是 T_0=\frac{2\pi}

    • k=0k=0 称为直流分量
    • k=±1k=\pm1 称为基波分量
    • k=±2k=\pm2 称为二次谐波分量
    • ... 等等...
    # 一般连续复指数信号

    f(t)=Kest=Kejθe(σ+jω0)t=Keσtej(ω0+θ)f (t) = Ke^{st} = |K|e^{j\theta}e^{(σ+j\omega_0)t} = Ke^{σt}e^{j(\omega_0+\theta)},

    其中 s=σ+jω0s=σ+j\omega_0K=KejθK=|K|e^{j\theta} 为复数

    该信号可看成是振幅按实指数信号规律变化的周期性复指数信号。它的实部与虚部都是振幅呈实指数规律变化的正弦振荡。

    # 抽样信号

    Sa(t)=sinttlimtSa(t)=0;limt0Sa(t)=10+Sa(t)dt=π2;+Sa(t)dt=πSa(t)=\frac{sin\ t}{t} \\ \lim_{t\to \infin}Sa(t)=0 \ \ ;\ \ \lim_{t\to 0}Sa(t)=1 \\ \int_{0}^{+\infin}Sa(t)\ d\ t=\frac{\pi}{2}\ \ ;\ \ \int_{-\infin}^{+\infin}Sa(t)\ d\ t=\pi

    # 高斯函数 “钟形信号”

    即正态分布 PDF

    φ(t)=1σ2πe(tμ)22σ2\varphi(t) = \frac{1}{\sigma\sqrt{2\pi}}e^{-\frac{(t-\mu)^2}{2\sigma^2}}

    # 能量 / 功率 信号

    能量功率信号.png

    # 能量信号

    能量为一有界值,即 0 < E < + ∞,此时 P = 0

    有时求 P 不好求,但是 E 好求,可以证明 E 为有界的,从而得到 P = 0

    # 功率信号

    功率为一有界值,即 0 < P < + ∞ ,此时 E = ∞

    有时求 E 不好求,但是 P 好求,可以证明 P 为有界的,从而得到 E = ∞

    # 信号的分解

    # 偶部与奇部

    f(t)=fe(t)+fo(t)f(t)=f_e(t)+f_o(t)

    偶部:

    fe(t)=12[f(t)+f(t)]f_e(t)=\frac{1}{2}[f(t)+f(-t)]

    奇部:

    fo(t)=12[f(t)f(t)]f_o(t)=\frac{1}{2}[f(t)-f(-t)]

    # 实部与虚部

    针对复函数而言

    f(t)=Re{f(t)}+Im{f(t)}f(t)=Re\{f(t)\}+Im\{f(t)\}

    实部:

    Re{f(t)}Re\{f(t)\}

    虚部:

    Im{f(t)}Im\{f(t)\}

    # 连续 / 离散 (时间) 信号

    区分 模拟 / 数字 信号,模拟 / 数字 信号的划分是依据 “值域” 是否连续,而 连续 / 离散 时间信号的划分是依据定义域是否连续

    连续时间信号 包含 模拟信号 和 数字 (量化) 信号

    # 周期信号定义

    • 连续周期信号 : f(t+mT)=f(t)f(t+mT)=f(t) , T 为周期 m = 0,±1,±2,±3...
    • 离散周期信号 : f(k+mN)=f(k)f(k+mN) = f(k) , N 为周期 m = 0,±1,±2,±3...

    T N 趋于⽆无穷时,周期信号变为⾮非周期信号。

    # 连续时间信号

    定义域连续

    在一定时间范围内,除若⼲不连续点之外,对任意时刻函数都有确定的函数值。 “连续” 的含义指定义域连续,⽽连续信号中可以含有不连续点。

    # 普通连续信号

    普通连续信号是指函数的定义域和值域没有不连续点 (即跳变点) 的信号。

    # (实) 指数信号

    f(t)=Keαtf (t) = Ke^{αt}

    指数函数求导、积分后仍为指数函数。

    # 正弦信号

    利用正弦函数与余弦函数表示的信号统称为正弦信号,可表示为

    f(t)=Acos(ωtθ)f(t)=Acos(\omega t-\theta)

    正弦信号求导、积分后仍然为正弦信号。

    指数信号可以转化为正弦信号:

    ejωt=cosωt+jsinωtejωt=cosωtjsinωte^{j\omega t}=cos\ \omega t+j\ sin\ \omega t \\ e^{-j\omega t}=cos\ \omega t-j\ sin\ \omega t

    指数信号可以转化为正弦信号:

    sinωt=12j(ejωtejωt)cosωt=12(ejωt+ejωt)sin\ \omega t = \frac{1}{2j}( e^{j\omega t} - e^{-j\omega t})\\ cos\ \omega t = \frac{1}{2}( e^{j\omega t} + e^{-j\omega t})

    # 复指数信号

    f(t)=Kest=Ke(σ+jω)t=Keσt(cosωt+jsinωt)f (t) = Ke^{st} = Ke^{(σ+j\omega)t} = Ke^{σt}(cos\ ωt + j\ sin\ ωt),

    其中 s=σ+jω0s=σ+j\omega_0 为复数

    复指数信号中,σ\sigma 决定增长衰减,ω0\omega_0 决定震荡快慢;

    σ\sigma 的大小ω0\omega_0 的大小最终结果
    σ=1\sigma=1ω00\omega_0\neq0实部虚部等幅震荡
    σ>1\sigma>1ω00\omega_0\neq0实部虚部增⻓震荡
    σ<1\sigma<1ω00\omega_0\neq0实部虚部衰减震荡
    \sigma≠0ω0=0\omega_0=0实部虚部为指数信号
    σ=0\sigma=0ω0=0\omega_0=0为直流信号
    # 连续复指数信号的周期性

    复指数信号是周期信号当且仅当 σ\sigma = 1 且 ω00\omega_0\neq 0

    基波周期:T=2πω0T=|\frac{2\pi}{\omega_0}|

    基波频率:ω0\omega_0

    # 成谐波关系的连续复指数信号集

    {φk(t)}={ejkω0t}k=0,±1,±2......\{\varphi_k(t) \}=\{ e^{jk\omega_0 t} \}\\ k=0,\pm1,\pm2......

    一般假设 ω0>0\omega_0 \gt 0 ,该集合中的每个信号都是周期的,它们的基波频率分别为 kω0k\omega_0 ,都是 ω0\omega_0 的整数倍,因而称它们是成谐波关系的。

    各次谐波的基波周期分别为 Tk=2πkω0T_k =\frac{2\pi}{|k\omega_0|} ,它们的公共周期是 T_0=\frac{2\pi}

    • k=0k=0 称为直流分量
    • k=±1k=\pm1 称为基波分量
    • k=±2k=\pm2 称为二次谐波分量
    • ... 等等...
    # 一般连续复指数信号

    f(t)=Kest=Kejθe(σ+jω0)t=Keσtej(ω0+θ)f (t) = Ke^{st} = |K|e^{j\theta}e^{(σ+j\omega_0)t} = Ke^{σt}e^{j(\omega_0+\theta)},

    其中 s=σ+jω0s=σ+j\omega_0K=KejθK=|K|e^{j\theta} 为复数

    该信号可看成是振幅按实指数信号规律变化的周期性复指数信号。它的实部与虚部都是振幅呈实指数规律变化的正弦振荡。

    # 抽样信号

    Sa(t)=sinttlimtSa(t)=0;limt0Sa(t)=10+Sa(t)dt=π2;+Sa(t)dt=πSa(t)=\frac{sin\ t}{t} \\ \lim_{t\to \infin}Sa(t)=0 \ \ ;\ \ \lim_{t\to 0}Sa(t)=1 \\ \int_{0}^{+\infin}Sa(t)\ d\ t=\frac{\pi}{2}\ \ ;\ \ \int_{-\infin}^{+\infin}Sa(t)\ d\ t=\pi

    # 高斯函数 “钟形信号”

    即正态分布 PDF

    φ(t)=1σ2πe(tμ)22σ2\varphi(t) = \frac{1}{\sigma\sqrt{2\pi}}e^{-\frac{(t-\mu)^2}{2\sigma^2}}

    # 奇异信号

    奇异信号是指函数本身或其导数与积分不连续点 (即跳变点) 的信号。

    # 单位斜坡信号 (ramp)

    r(t)={0t<0tt0\begin{aligned} r(t) = \begin{cases} 0 & t \lt 0 \\ t & t \geq 0 \end{cases} \end{aligned}

    # 符号函数 (signal)

    sig(t)={1t<00t=01t<0\begin{aligned} sig(t) = \begin{cases} 1 & t \lt 0 \\ 0 & t = 0 \\ -1 & t \lt 0 \end{cases} \end{aligned}

    # 单位阶跃信号

    u(t)=ε(t)={1t>0undifinedt=00t<0\begin{aligned} u(t)=\varepsilon(t) = \begin{cases} 1 & t \gt 0 \\ \text{undifined} & t = 0 \\ 0 & t \lt 0 \end{cases} \end{aligned}

    t = 0 处函数无定义 (部分教材规定 t = 0 时取 12\frac{1}{2})

    有的教材用 u(t)u(t) 也有的教材用 ε(t)\varepsilon(t)

    # 单位延迟阶跃信号

    ε(tt0)={1t>t0undifinedt=t00t<t0\begin{aligned} \varepsilon(t-t_0) = \begin{cases} 1 & t \gt t_0 \\ \text{undifined} & t = t_0 \\ 0 & t \lt t_0 \end{cases} \end{aligned}

    # 物理意义
    • ε(t) 表示信号 f (t) = 1 在 t = 0 时刻接⼊入系统。
    • f (t) ε(t) 表示信号 f (t) 在 t = 0 时刻接⼊入系统。
    • f (t) ε(t - t0 ) 表示信号 f (t) 在 t = t0 时刻接⼊入系统。
    # 与斜坡信号的关系

    r(t)=tε(t)dr(t)dt=ε(t)tε(τ)dτ=r(t)r(t)=t\ \varepsilon(t)\\ \frac{d\ r(t)}{d\ t}=\varepsilon(t) \\ \int_{-\infin}^{t}\varepsilon(\tau)\ d\ \tau=r(t)

    # (单位) 冲激信号

    # 物理定义

    δ(t)=limτ01τ[ε(t+τ2)ε(tτ2)]δ(tt0)=limτt01τ[ε(t+τ2t0)ε(tτ2t0)]\delta(t)=\lim_{\tau\to 0}\frac{1}{\tau}[\varepsilon(t+\frac{\tau}{2})-\varepsilon(t-\frac{\tau}{2})] \\ \delta(t-t_0)=\lim_{\tau\to t_0}\frac{1}{\tau}[\varepsilon(t+\frac{\tau}{2}-t_0)-\varepsilon(t-\frac{\tau}{2}-t_0)]

    # 狄拉克定义

    δ(t)={+δ(t)dt=1δ(t)=0,t0\delta (t)= \begin{cases} \int_{-\infin}^{+\infin}\delta (t)\ d\ t=1 \\ \delta (t)=0,t\neq 0 \end{cases}

    δ(tt0)={+δ(tt0)dt=1δ(t)=0,tt0\delta (t-t_0)= \begin{cases} \int_{-\infin}^{+\infin}\delta (t-t_0)\ d\ t=1 \\ \delta (t)=0,t\neq t_0 \end{cases}

    # 广义函数定义

    +δ(t)φ(t)dt=φ(0)+δ(tt0)φ(t)dt=φ(t0)\int_{-\infin}^{+\infin}\delta (t)\ \varphi(t)\ d\ t =\varphi (0) \\ \int_{-\infin}^{+\infin}\delta (t-t_0)\ \varphi(t)\ d\ t =\varphi (t_0)

    # 取样性

    即广义函数定义,注意在有限区间同样有取样性

    # 尺度变化特性

    δ(at)=1aδ(t)δ(att0)=1aδ(tt0a)+f(t)δ(att0)=1af(t0a)a0\delta (a\ t)=\frac{1}{|a|}\delta (t) \\ \delta (a\ t-t_0)=\frac{1}{|a|}\delta (t-\frac{t_0}{a}) \\ \int_{-\infin}^{+\infin}f(t)\ \delta (a\ t-t_0)=\frac{1}{|a|}f(\frac{t_0}{a}) \\ a\neq 0

    # 与普通函数相乘

    f(t)δ(t)=f(0)δ(t)f(t)δ(tt0)=f(t0)δ(tt0)f(t)\ \delta(t)=f(0)\ \delta(t) \\ f(t)\ \delta(t-t_0)=f(t_0)\ \delta(t-t_0)

    # 与阶跃信号的关系

    ddtε(t)=δ(t)\frac{d}{d\ t}\varepsilon(t)=\delta(t)

    若信号的函数值有跳变,则信号在跳变点处的导数为冲激信号,其中冲击强度为信号在跳变点的跳跃值

    # (单位) 冲激偶信号

    # 定义

    单位冲激信号的 “奇函数版本”,且是单位冲激信号的导函数

    δ(t)=ddtδ(t)\delta '(t)=\frac{d}{d\ t}\delta (t)

    单位冲激信号是偶函数;单位冲激偶信号是奇函数!!

    二者通过积分和求导可相互转化

    # 与普通函数相乘

    f(t)δ(t)=f(0)δ(t)f(0)δ(t)f(t)δ(tt0)=f(0)δ(tt0)f(0)δ(tt0)f(t)\delta'(t)=f(0)\delta'(t)-f'(0)\delta(t) \\ f(t)\delta'(t-t_0)=f(0)\delta'(t-t_0)-f'(0)\delta(t-t_0)

    # 筛选特性

    +f(t)δ(t)dt=f(0)+f(t)δ(tt0)dt=f(t0)\int_{-\infin}^{+\infin}f(t)\delta'(t)d\ t=-f'(0) \\ \int_{-\infin}^{+\infin}f(t)\delta'(t-t_0)d\ t=-f'(t_0)

    # 尺度变化特性

    δ(at)=1aaδ(t)a0\delta'(a\ t)=\frac{1}{a|a|}\delta'(t) \\ a\neq 0

    # 离散时间信号

    定义域离散

    仅在一些离散时刻才有意义,⽽而在其他时刻⽆无定义。离散时间⼀一般为均匀间隔,“离散” 是指信号的定义域是离散的。

    # 基本离散信号

    # 单位序列

    单位序列列 δ(k) 亦称单位样值序列列,类似于连续时间信号中的 δ(t)

    δ(k)={1k=00k0\delta (k)= \begin{cases} 1 & k=0 \\ 0 & k\neq 0 \end{cases}

    δ(ki)={1k=i0ki\delta (k-i)= \begin{cases} 1 & k=i \\ 0 & k\neq i \end{cases}

    # 单位阶跃序列

    单位阶跃序列 ε(k) 类似于连续时间信号中的单位阶跃函数 ε(t)

    ε(k)={1k00k<0\varepsilon (k)= \begin{cases} 1 & k \geq 0 \\ 0 & k \lt 0 \end{cases}

    ε(ki)={1ki0k<i\varepsilon (k-i)= \begin{cases} 1 & k \geq i \\ 0 & k \lt i \end{cases}

    # 单位序列 δ(k) 转化为单位阶跃序列 ε(k)

    ε(k)=j=0+δ(kj)\varepsilon (k)=\sum_{j=0}^{+\infin}\delta(k-j)

    # 单位阶跃序列 ε(k) 转化为单位序列 δ(k)

    δ(k)=ε(k)ε(k1)\delta(k)=\varepsilon(k)-\varepsilon(k-1)

    # 单边指数序列

    f(k)=αkε(k),kzf (k) = \alpha^{k}\varepsilon(k)\ ,\ \ k \in z

    当 |α| > 1 时,序列列发散,|α| < 1 时,序列列收敛;

    α > 0 时,序列列恒正, α < 0 时,序列列正负交错。

    # 正弦序列

    f(k)=cosω0k,kzf (k) = cos\ \omega_0\ k\ ,\ \ k \in z

    ω0\omega_0 为数字角频率,其与连续正弦信号的⻆频率不同,详见数字信号处理

    不是所有的正弦序列都是周期序列,因为正弦序列为连续正弦信号的抽样

    # 离散正弦序列的周期性

    正弦序列是周期序列当且仅当

    kz,ω02kπ\exist k\in z\ \ ,\ \ \omega_0 | 2k\pi

    # 复指数序列

    f(k)=e(α+jω0)k=eαkejω0k=eαk(cosω0k+jsinω0k)kzf (k) = e^{(\alpha +j\omega_0)k} = e^{\alpha k}e^{j\omega_0 k}=e^{\alpha k}(cos\ \omega_0k +j\ sin\ \omega_0k)\\ k \in z

    α 的大小ω0 的大小最终结果
    α = 0ω00\omega_0\neq 0实部虚部均等幅正弦序列
    α > 0ω00\omega_0\neq 0实部虚部均增幅正弦序列
    α < 0ω00\omega_0\neq 0实部虚部均减幅正弦序列
    α ≠ 0ω0=0\omega_0 = 0为指数序列
    α = 0ω0=0\omega_0 = 0为常数序列
    # 离散复指数序列的周期性

    $\alpha = 0 \omega_0\neq 0$ 时实部虚部均等幅正弦序列,对应连续复指数信号是周期信号,离散复指数序列才有可能是周期序列,因为离散复指数序列为连续复指数信号的抽样

    复指数序列是周期序列当且仅当

    N,mz2πω0=Nmejω0N=1ejω0n=ejω0nejω0N=ejω0(n+N)x[k]=x[k+N]\exist N,m\in z\\ \frac{2\pi}{\omega_0}=\frac{N}{m} \\ \lrArr e^{j\omega_0N} = 1\\ \lrArr e^{j\omega_0n} = e^{j\omega_0n}\cdot e^{j\omega_0N} = e^{j\omega_0(n+N)} \\ \lrArr x[k]=x[k+N]

    NNmm 互质(无公因子)时:

    基波周期:N=2πω0mN =\frac{2π}{\omega_0}m

    基波频率为:\omega=\frac{2π}{N}=\frac{\omega_0}

    # 成谐波关系的离散复指数信号集

    {Φk[n]}={ej2πNkn}k=0,±1,±2......\{\varPhi_k[n] \}=\{ e^{j\frac{2\pi}{N}kn} \}\\ k=0,\pm1,\pm2......

    该信号集中的每一个信号都是以 NN 为公共周期的,频率是 2π 的整数倍。

    与连续时间情况不同,该信号集中的信号并不都是彼此独立的。显然有:

    Φk+N[n]=Φk[n]\varPhi_{k+N}[n] = \varPhi_k[n]

    该信号集中只有 N 个信号是独立的。即当 k 取相连的 NN 个整数时所对应的各个谐波才是彼此独立的。

    根据集合定义,那么实际上这个信号集的元素个数是 NN 个。

    这里 “相互独立” 的意思是互不相等

    # 信号的基本运算

    # 平移与反转

    连续信号和离散信号的运算一致

    f(t) \to f(t+t_0) & 沿横轴向左平移\\ f(t) \to f(t-t_0) & 沿横轴向右平移\\ f(t) \to f(-t) & 沿纵轴镜像翻转

    # 尺度变换

    对于离散信号,⼀般不做尺度变换,因为可能丢失序列信息,只对连续信号尺度变换

    f(t)f(at)f(t) \to f(a\ t)

    横轴变为原来的 1a\frac{1}{a}

    # 连续信号的求导与积分

    # 离散信号的差分与求和

    好离散信号的向 (一阶) 差分

    Δf(k)=f(k+1)f(k)\Delta f(k)=f(k+1)-f(k)

    好离散信号的向 (一阶) 差分

    f(k)=f(k)f(k1)\nabla f(k)=f(k)-f(k-1)

    # 信号的运算的综合运⽤

    平移、反折、压缩等各种运算均对独⽴、单一的变量 t 计算,⽽不是对 atat + b 整体计算

    对于 f (t) → f(at + b) 先平移,后尺度 (含反折),即 f (t) → f(t + b) → f(at + b)

    对于 f (at + b) → f(t) 先尺度 (含反折),后平移,即 f (at + b) → f(t + b) → f(t)

    \ No newline at end of file +M834 80h400000v40h-400000z"/>1e2σ2(tμ)2

    # 奇异信号

    奇异信号是指函数本身或其导数与积分不连续点 (即跳变点) 的信号。

    # 单位斜坡信号 (ramp)

    r(t)={0t<0tt0\begin{aligned} r(t) = \begin{cases} 0 & t \lt 0 \\ t & t \geq 0 \end{cases} \end{aligned}

    # 符号函数 (signal)

    sig(t)={1t<00t=01t<0\begin{aligned} sig(t) = \begin{cases} 1 & t \lt 0 \\ 0 & t = 0 \\ -1 & t \lt 0 \end{cases} \end{aligned}

    # 单位阶跃信号

    u(t)=ε(t)={1t>0undifinedt=00t<0\begin{aligned} u(t)=\varepsilon(t) = \begin{cases} 1 & t \gt 0 \\ \text{undifined} & t = 0 \\ 0 & t \lt 0 \end{cases} \end{aligned}

    t = 0 处函数无定义 (部分教材规定 t = 0 时取 12\frac{1}{2})

    有的教材用 u(t)u(t) 也有的教材用 ε(t)\varepsilon(t)

    # 单位延迟阶跃信号

    ε(tt0)={1t>t0undifinedt=t00t<t0\begin{aligned} \varepsilon(t-t_0) = \begin{cases} 1 & t \gt t_0 \\ \text{undifined} & t = t_0 \\ 0 & t \lt t_0 \end{cases} \end{aligned}

    # 物理意义
    • ε(t) 表示信号 f (t) = 1 在 t = 0 时刻接⼊入系统。
    • f (t) ε(t) 表示信号 f (t) 在 t = 0 时刻接⼊入系统。
    • f (t) ε(t - t0 ) 表示信号 f (t) 在 t = t0 时刻接⼊入系统。
    # 与斜坡信号的关系

    r(t)=tε(t)dr(t)dt=ε(t)tε(τ)dτ=r(t)r(t)=t\ \varepsilon(t)\\ \frac{d\ r(t)}{d\ t}=\varepsilon(t) \\ \int_{-\infin}^{t}\varepsilon(\tau)\ d\ \tau=r(t)

    # (单位) 冲激信号

    # 物理定义

    δ(t)=limτ01τ[ε(t+τ2)ε(tτ2)]δ(tt0)=limτt01τ[ε(t+τ2t0)ε(tτ2t0)]\delta(t)=\lim_{\tau\to 0}\frac{1}{\tau}[\varepsilon(t+\frac{\tau}{2})-\varepsilon(t-\frac{\tau}{2})] \\ \delta(t-t_0)=\lim_{\tau\to t_0}\frac{1}{\tau}[\varepsilon(t+\frac{\tau}{2}-t_0)-\varepsilon(t-\frac{\tau}{2}-t_0)]

    # 狄拉克定义

    δ(t)={+δ(t)dt=1δ(t)=0,t0\delta (t)= \begin{cases} \int_{-\infin}^{+\infin}\delta (t)\ d\ t=1 \\ \delta (t)=0,t\neq 0 \end{cases}

    δ(tt0)={+δ(tt0)dt=1δ(t)=0,tt0\delta (t-t_0)= \begin{cases} \int_{-\infin}^{+\infin}\delta (t-t_0)\ d\ t=1 \\ \delta (t)=0,t\neq t_0 \end{cases}

    # 广义函数定义

    +δ(t)φ(t)dt=φ(0)+δ(tt0)φ(t)dt=φ(t0)\int_{-\infin}^{+\infin}\delta (t)\ \varphi(t)\ d\ t =\varphi (0) \\ \int_{-\infin}^{+\infin}\delta (t-t_0)\ \varphi(t)\ d\ t =\varphi (t_0)

    # 取样性

    即广义函数定义,注意在有限区间同样有取样性

    # 尺度变化特性

    δ(at)=1aδ(t)δ(att0)=1aδ(tt0a)+f(t)δ(att0)=1af(t0a)a0\delta (a\ t)=\frac{1}{|a|}\delta (t) \\ \delta (a\ t-t_0)=\frac{1}{|a|}\delta (t-\frac{t_0}{a}) \\ \int_{-\infin}^{+\infin}f(t)\ \delta (a\ t-t_0)=\frac{1}{|a|}f(\frac{t_0}{a}) \\ a\neq 0

    # 与普通函数相乘

    f(t)δ(t)=f(0)δ(t)f(t)δ(tt0)=f(t0)δ(tt0)f(t)\ \delta(t)=f(0)\ \delta(t) \\ f(t)\ \delta(t-t_0)=f(t_0)\ \delta(t-t_0)

    # 与阶跃信号的关系

    ddtε(t)=δ(t)\frac{d}{d\ t}\varepsilon(t)=\delta(t)

    若信号的函数值有跳变,则信号在跳变点处的导数为冲激信号,其中冲击强度为信号在跳变点的跳跃值

    # (单位) 冲激偶信号

    # 定义

    单位冲激信号的 “奇函数版本”,且是单位冲激信号的导函数

    δ(t)=ddtδ(t)\delta '(t)=\frac{d}{d\ t}\delta (t)

    单位冲激信号是偶函数;单位冲激偶信号是奇函数!!

    二者通过积分和求导可相互转化

    # 与普通函数相乘

    f(t)δ(t)=f(0)δ(t)f(0)δ(t)f(t)δ(tt0)=f(0)δ(tt0)f(0)δ(tt0)f(t)\delta'(t)=f(0)\delta'(t)-f'(0)\delta(t) \\ f(t)\delta'(t-t_0)=f(0)\delta'(t-t_0)-f'(0)\delta(t-t_0)

    # 筛选特性

    +f(t)δ(t)dt=f(0)+f(t)δ(tt0)dt=f(t0)\int_{-\infin}^{+\infin}f(t)\delta'(t)d\ t=-f'(0) \\ \int_{-\infin}^{+\infin}f(t)\delta'(t-t_0)d\ t=-f'(t_0)

    # 尺度变化特性

    δ(at)=1aaδ(t)a0\delta'(a\ t)=\frac{1}{a|a|}\delta'(t) \\ a\neq 0

    # 离散时间信号

    定义域离散

    仅在一些离散时刻才有意义,⽽而在其他时刻⽆无定义。离散时间⼀一般为均匀间隔,“离散” 是指信号的定义域是离散的。

    # 基本离散信号

    # 单位序列

    单位序列列 δ(k) 亦称单位样值序列列,类似于连续时间信号中的 δ(t)

    δ(k)={1k=00k0\delta (k)= \begin{cases} 1 & k=0 \\ 0 & k\neq 0 \end{cases}

    δ(ki)={1k=i0ki\delta (k-i)= \begin{cases} 1 & k=i \\ 0 & k\neq i \end{cases}

    # 单位阶跃序列

    单位阶跃序列 ε(k) 类似于连续时间信号中的单位阶跃函数 ε(t)

    ε(k)={1k00k<0\varepsilon (k)= \begin{cases} 1 & k \geq 0 \\ 0 & k \lt 0 \end{cases}

    ε(ki)={1ki0k<i\varepsilon (k-i)= \begin{cases} 1 & k \geq i \\ 0 & k \lt i \end{cases}

    # 单位序列 δ(k) 转化为单位阶跃序列 ε(k)

    ε(k)=j=0+δ(kj)\varepsilon (k)=\sum_{j=0}^{+\infin}\delta(k-j)

    # 单位阶跃序列 ε(k) 转化为单位序列 δ(k)

    δ(k)=ε(k)ε(k1)\delta(k)=\varepsilon(k)-\varepsilon(k-1)

    # 单边指数序列

    f(k)=αkε(k),kzf (k) = \alpha^{k}\varepsilon(k)\ ,\ \ k \in z

    当 |α| > 1 时,序列列发散,|α| < 1 时,序列列收敛;

    α > 0 时,序列列恒正, α < 0 时,序列列正负交错。

    # 正弦序列

    f(k)=cosω0k,kzf (k) = cos\ \omega_0\ k\ ,\ \ k \in z

    ω0\omega_0 为数字角频率,其与连续正弦信号的⻆频率不同,详见数字信号处理

    不是所有的正弦序列都是周期序列,因为正弦序列为连续正弦信号的抽样

    # 离散正弦序列的周期性

    正弦序列是周期序列当且仅当

    kz,ω02kπ\exist k\in z\ \ ,\ \ \omega_0 | 2k\pi

    # 复指数序列

    f(k)=e(α+jω0)k=eαkejω0k=eαk(cosω0k+jsinω0k)kzf (k) = e^{(\alpha +j\omega_0)k} = e^{\alpha k}e^{j\omega_0 k}=e^{\alpha k}(cos\ \omega_0k +j\ sin\ \omega_0k)\\ k \in z

    α 的大小ω0 的大小最终结果
    α = 0ω00\omega_0\neq 0实部虚部均等幅正弦序列
    α > 0ω00\omega_0\neq 0实部虚部均增幅正弦序列
    α < 0ω00\omega_0\neq 0实部虚部均减幅正弦序列
    α ≠ 0ω0=0\omega_0 = 0为指数序列
    α = 0ω0=0\omega_0 = 0为常数序列
    # 离散复指数序列的周期性

    $\alpha = 0 \omega_0\neq 0$ 时实部虚部均等幅正弦序列,对应连续复指数信号是周期信号,离散复指数序列才有可能是周期序列,因为离散复指数序列为连续复指数信号的抽样

    复指数序列是周期序列当且仅当

    N,mz2πω0=Nmejω0N=1ejω0n=ejω0nejω0N=ejω0(n+N)x[k]=x[k+N]\exist N,m\in z\\ \frac{2\pi}{\omega_0}=\frac{N}{m} \\ \lrArr e^{j\omega_0N} = 1\\ \lrArr e^{j\omega_0n} = e^{j\omega_0n}\cdot e^{j\omega_0N} = e^{j\omega_0(n+N)} \\ \lrArr x[k]=x[k+N]

    NNmm 互质(无公因子)时:

    基波周期:N=2πω0mN =\frac{2π}{\omega_0}m

    基波频率为:\omega=\frac{2π}{N}=\frac{\omega_0}

    # 成谐波关系的离散复指数信号集

    {Φk[n]}={ej2πNkn}k=0,±1,±2......\{\varPhi_k[n] \}=\{ e^{j\frac{2\pi}{N}kn} \}\\ k=0,\pm1,\pm2......

    该信号集中的每一个信号都是以 NN 为公共周期的,频率是 2π 的整数倍。

    与连续时间情况不同,该信号集中的信号并不都是彼此独立的。显然有:

    Φk+N[n]=Φk[n]\varPhi_{k+N}[n] = \varPhi_k[n]

    该信号集中只有 N 个信号是独立的。即当 k 取相连的 NN 个整数时所对应的各个谐波才是彼此独立的。

    根据集合定义,那么实际上这个信号集的元素个数是 NN 个。

    这里 “相互独立” 的意思是互不相等

    # 信号的基本运算

    # 平移与反转

    连续信号和离散信号的运算一致

    f(t) \to f(t+t_0) & 沿横轴向左平移\\ f(t) \to f(t-t_0) & 沿横轴向右平移\\ f(t) \to f(-t) & 沿纵轴镜像翻转

    # 尺度变换

    对于离散信号,⼀般不做尺度变换,因为可能丢失序列信息,只对连续信号尺度变换

    f(t)f(at)f(t) \to f(a\ t)

    横轴变为原来的 1a\frac{1}{a}

    # 连续信号的求导与积分

    # 离散信号的差分与求和

    好离散信号的向 (一阶) 差分

    Δf(k)=f(k+1)f(k)\Delta f(k)=f(k+1)-f(k)

    好离散信号的向 (一阶) 差分

    f(k)=f(k)f(k1)\nabla f(k)=f(k)-f(k-1)

    # 信号的运算的综合运⽤

    平移、反折、压缩等各种运算均对独⽴、单一的变量 t 计算,⽽不是对 atat + b 整体计算

    对于 f (t) → f(at + b) 先平移,后尺度 (含反折),即 f (t) → f(t + b) → f(at + b)

    对于 f (at + b) → f(t) 先尺度 (含反折),后平移,即 f (at + b) → f(t + b) → f(t)

    \ No newline at end of file diff --git a/ie/signals-and-systems/systems-concepts/index.html b/ie/signals-and-systems/systems-concepts/index.html index bd9011f2..3fcfdf0c 100644 --- a/ie/signals-and-systems/systems-concepts/index.html +++ b/ie/signals-and-systems/systems-concepts/index.html @@ -1 +1 @@ -系统基本概念 - 信号与系统 - 电子信息 | Balloon Party = 風船のパーティー = fuusen no party

    # 系统的描述

    # 连续 / 离散 时间系统

    # 连续时间系统

    输入信号与输出响应都是连续时间信号的系统。

    连续时间系统

    eg: 机械系统、电路系统

    # 离散时间系统

    输入信号与输出响应都是离散时间信号的系统。

    离散时间系统

    eg: 人口问题

    # 两种描述方法

    # 输入输出描述法 (微分方程、差分⽅程)

    着眼于系统输入与输出的关系,适⽤用于单输入 — 单输出系统。

    # 状态变量描述法 (状态⽅程、输出方程)

    即可描述输入与输出的关系,还可以描述系统的内部状态 既可⽤用于单输入 — 单输出系统,⼜可⽤于多输⼊ — 多输出系统。

    # 系统的互联

    现实中的系统是各式各样的,其复杂程度也大相径庭。但许多系统都可以分解为若干个简单系统的组合。这一思想对系统分析系统综合十分重要。

    • 可以通过对简单系统(子系统)的分析并通过子系统互联而达到分析复杂系统的目的。
    • 也可以通过将若干个简单子系统互联起来而实现一个相对复杂的系统。

    # 级联

    级联

    # 并联

    并联

    # 反馈联结

    反馈联结

    # 系统的组成

    ⼀个系统由若⼲个基本单元组成, 其中的基本组成单元包括:

    # 加法器

    加法器

    # 数乘器

    数乘器

    # 积分器

    积分器

    # 单位延时器

    单位延时器

    # 延时器

    延时器

    # 系统的基本性质

    # 记忆性

    在任何时刻,系统的输出都只与当前时刻的输入有关,而与该时刻以外的输入无关,则称该系统是无记忆系统,也叫即时系统。否则就是记忆系统,也叫动态系统

    注意一个特例 —— 恒等系统:任何时刻系统的输出响应与输入信号都相同,即有 y(t) = x(t)y[n] = x[n]

    包含积分器、微分器的系统就是记忆系统

    # 可逆性

    如果一个系统对任何不同的输入都能产生不同的输出,即输入与输出是一一对应,则称该系统是可逆系统。反之,称为不可逆系统

    一个可逆系统与另一个系统级联后构成一个恒等系统,则称后者是前者的逆系统

    逆系统

    # 因果性

    如果系统在任何时刻的输出只与当前时刻及此前的输入有关,而和此后的输入无关,就称该系统是因果的。否则就是非因果的。

    一般说来,实时的物理系统都是因果的,因为时间是单向变化的,结果发生于原因之后。

    但对非实时处理信号的离散时间系统,或信号的自变量并不具有时间概念的情况,因果性并不一定成为系统能否物理实现的先决条件。

    • 因果系统:响应后于激励
    • 非因果系统:响应先于激励

    由定义显然可见:非记忆性系统一定是因果的;非因果系统一定是记忆性的。

    # 稳定性

    如果一个系统对任意有界输入都产生有界输出,则该系统是稳定系统。否则,就是不稳定系统

    即当 e(.)<|e(.)|\lt\infin时yzs(.)<|y_{zs}(.)|\lt\infin

    工程实际中通常希望所设计的系统是稳定的。因此稳定性对系统设计来说是一个重要的考虑因素。

    # 时不变性

    时不变性质只对零状态响应 yzs(.)y_{zs}( . ) 讨论

    e(.)e(.)x(.)x(.) 表示输入信号,或者说激励

    yzs(.)y_{zs}(.) 表示零状态响应,zs 即 zero status 零状态

    yzi(.)y_{zi}(.) 表示零输入响应,zi 即 zero input 零输入

    全响应 = 零状态响应 + 零输入响应:y(.)=yzx(.)+yzi(.)y(.)=y_{zx}(.)+y_{zi}(.)

    当输入信号有一个时移时,系统的输出响应也产生相同时移,此外无其它变化,则称该系统是时不变的。否则就是时变的。

    若对任意 x(t)y(t)x(t) → y(t)x(tt0)y(tt0)x(t - t0) → y(t - t0) ,则系统是时不变的。

    ⭐️ 检验一个系统时不变性的步骤(以连续时间为例,离散时间可类推)

    1. 令输入为 x1(t)x_1(t) ,根据系统描述确定对应输出 y1(t)y_1(t) ;
    2. 将输入变为 x2(t)=x1(tt0)x_2(t) = x_1(t - t_0) ,再根据系统描述确定输出 y2(t)y_2(t) ;
    3. 根据自变量变换,检验 y2(t)y_2(t) 是否等于 y1(tt0)y_1(t - t_0)

    # 线性 linear

    # 定义

    连续时间线性系统定义

    对任意 x1(t)y1(t)x1(t) → y1(t)x2(t)y2(t)x2(t) → y2(t) ,总有 ax1(t)+bx2(t)ay1(t)+by2(t)ax_1(t) + bx_2(t) → ay_1(t) + by_2(t) ,其中 a 与 b 是任意常数。

    连续线性时间系统

    离散时间线性系统同理。

    实际上线性等价于以下两个性质之和:

    x1(t)+x2(t)y1(t)+y2(t)x_1(t) + x_2(t) → y_1(t) + y_2(t)(可加性 / 叠加性)

    ax1(t)ay1(t)ax_1(t) → ay_1(t)(齐次性 / 均匀性)

    # 零输入 — 零输出特性

    根据线性系统的齐次性,可得出:线性系统当输入为零(即没有输入)时,系统的输出响应也必然为零(即没有输出响应)。这就是所谓线性系统的零输入 — 零输出特性

    # 非线性系统举例

    eg1 : 满足可加性,但不满足齐次性

    y(t)=Re{x(t)}y(t) = Re\{x(t)\}

    eg2 : 满足齐次性但不满足可加性

    y(t)=[x(t)]2x(t)y(t) = \frac{[x'(t)]^2}{x(t)}

    # 线性系统的分解特性

    如果一个系统是线性的,则只要我们能够把输入信号分解成若干个简单信号的线性组合,且知道该系统对每一个简单信号的响应,就可以很方便地通过线性组合而得到系统对原输入信号的响应。即:

    {x(t)=kakxk(t)xk(t)yk(t)y(t)=kakyk(t)\begin{cases} x(t)=\sum_k a_k x_k (t)\\ x_k(t)\to y_k(t) \end{cases} \rArr y(t)=\sum_k a_k y_k (t)

    这一思想是信号与系统分析理论和方法的基础

    # 特殊系统

    # 增量线性系统

    在工程实际中,有一类系统并不满足线性系统的要求。但是这类系统的输出响应的增量与输入信号的增量之间满足线性特性。这类系统称为增量线性系统

    eg:考虑系统 y(t)=x(t)+2y(t) = x(t) + 2 ,由于:

    x1(t)y1(t)=x1(t)+2x2(t)y2(t)=x2(t)+2x_1(t) → y_1(t) = x_1(t) + 2\\ x_2(t) → y_2(t) = x_2(t) + 2

    该系统既不满足齐次性,也不满足可加性,但当考查输入的增量与输出的增量之间的关系时,有

    Δy(t)=y1(t)y2(t)=x1(t)x2(t)=Δx(t)Δy(t) = y_1(t) - y_2(t) = x_1(t) - x_2(t) = Δx(t)

    可见输入的增量与输出的增量之间是满足线性关系的,它是一个增量线性系统。

    增量线性系统可以等效为一个线性系统再加上一部分与输入无关的响应。

    线性增量系统

    增量线性系统的响应包括:

    • 零状态响应:当 y0(t)=0y_0(t) = 0 时,y(t)=y1(t)y(t) = y_1(t) 。此时系统处于零初始状态,故将 y1(t)y_1(t) 称为系统的零状态响应。
    • 零输入响应:当 x(t)=0x(t) = 0 时,有 y1(t)=0y_1(t) = 0y(t)=y0(t)y(t) = y_0(t) ,因此将 y0(t)y_0(t) 称为系统的零输入响应。

    # 线性时不变系统(LTI)

    线性 + 时不变性 = LTI

    基本事实:如果知道线性时不变系统对某些输入信号的响应,则可以知道该系统对许多其它输入信号的响应。

    线性时不变(LTI)系统是本课程主要关注的系统

    # 线性性质

    # 根据系统框图描述建立微分方程

    # 对于连续系统

    步骤如下

    1. 首先判断其阶数,阶数 = 积分器个数
    2. 设信号传递 ** 最后一个积分器输出为 x(t)x(t) ** ,则各积分器的输入输出均可确定
    3. 对每个加法器列⽅程:输入和 = 输出和
    4. 联立方程消去中间变量 x(t)x(t) ,得到关于输出信号 y(t) 的微分方程

    # 对于离散系统

    步骤如下

    1. 首先判断其阶数,阶数 = 延时单元个数
    2. 设信号传递第一个延时单元输入为 x(k) ,则各延时单元的输入输出均可确定
    3. 对每个加法器列⽅程:输入和 = 输出和
    4. 联立方程消去中间变量 x(t)x(t) ,得到关于输出信号 y(t) 的微分方程
    \ No newline at end of file +系统基本概念 - 信号与系统 - 电子信息 | Balloon Party = 風船のパーティー = fuusen no party

    # 系统的描述

    # 连续 / 离散 时间系统

    # 连续时间系统

    输入信号与输出响应都是连续时间信号的系统。

    连续时间系统

    eg: 机械系统、电路系统

    # 离散时间系统

    输入信号与输出响应都是离散时间信号的系统。

    离散时间系统

    eg: 人口问题

    # 两种描述方法

    # 输入输出描述法 (微分方程、差分⽅程)

    着眼于系统输入与输出的关系,适⽤用于单输入 — 单输出系统。

    # 状态变量描述法 (状态⽅程、输出方程)

    即可描述输入与输出的关系,还可以描述系统的内部状态 既可⽤用于单输入 — 单输出系统,⼜可⽤于多输⼊ — 多输出系统。

    # 系统的互联

    现实中的系统是各式各样的,其复杂程度也大相径庭。但许多系统都可以分解为若干个简单系统的组合。这一思想对系统分析系统综合十分重要。

    • 可以通过对简单系统(子系统)的分析并通过子系统互联而达到分析复杂系统的目的。
    • 也可以通过将若干个简单子系统互联起来而实现一个相对复杂的系统。

    # 级联

    级联

    # 并联

    并联

    # 反馈联结

    反馈联结

    # 系统的组成

    ⼀个系统由若⼲个基本单元组成, 其中的基本组成单元包括:

    # 加法器

    加法器

    # 数乘器

    数乘器

    # 积分器

    积分器

    # 单位延时器

    单位延时器

    # 延时器

    延时器

    # 系统的基本性质

    # 记忆性

    在任何时刻,系统的输出都只与当前时刻的输入有关,而与该时刻以外的输入无关,则称该系统是无记忆系统,也叫即时系统。否则就是记忆系统,也叫动态系统

    注意一个特例 —— 恒等系统:任何时刻系统的输出响应与输入信号都相同,即有 y(t) = x(t)y[n] = x[n]

    包含积分器、微分器的系统就是记忆系统

    # 可逆性

    如果一个系统对任何不同的输入都能产生不同的输出,即输入与输出是一一对应,则称该系统是可逆系统。反之,称为不可逆系统

    一个可逆系统与另一个系统级联后构成一个恒等系统,则称后者是前者的逆系统

    逆系统

    # 因果性

    如果系统在任何时刻的输出只与当前时刻及此前的输入有关,而和此后的输入无关,就称该系统是因果的。否则就是非因果的。

    一般说来,实时的物理系统都是因果的,因为时间是单向变化的,结果发生于原因之后。

    但对非实时处理信号的离散时间系统,或信号的自变量并不具有时间概念的情况,因果性并不一定成为系统能否物理实现的先决条件。

    • 因果系统:响应后于激励
    • 非因果系统:响应先于激励

    由定义显然可见:非记忆性系统一定是因果的;非因果系统一定是记忆性的。

    # 稳定性

    如果一个系统对任意有界输入都产生有界输出,则该系统是稳定系统。否则,就是不稳定系统

    即当 e(.)<|e(.)|\lt\infin时yzs(.)<|y_{zs}(.)|\lt\infin

    工程实际中通常希望所设计的系统是稳定的。因此稳定性对系统设计来说是一个重要的考虑因素。

    # 时不变性

    时不变性质只对零状态响应 yzs(.)y_{zs}( . ) 讨论

    e(.)e(.)x(.)x(.) 表示输入信号,或者说激励

    yzs(.)y_{zs}(.) 表示零状态响应,zs 即 zero status 零状态

    yzi(.)y_{zi}(.) 表示零输入响应,zi 即 zero input 零输入

    全响应 = 零状态响应 + 零输入响应:y(.)=yzx(.)+yzi(.)y(.)=y_{zx}(.)+y_{zi}(.)

    当输入信号有一个时移时,系统的输出响应也产生相同时移,此外无其它变化,则称该系统是时不变的。否则就是时变的。

    若对任意 x(t)y(t)x(t) → y(t)x(tt0)y(tt0)x(t - t0) → y(t - t0) ,则系统是时不变的。

    ⭐️ 检验一个系统时不变性的步骤(以连续时间为例,离散时间可类推)

    1. 令输入为 x1(t)x_1(t) ,根据系统描述确定对应输出 y1(t)y_1(t) ;
    2. 将输入变为 x2(t)=x1(tt0)x_2(t) = x_1(t - t_0) ,再根据系统描述确定输出 y2(t)y_2(t) ;
    3. 根据自变量变换,检验 y2(t)y_2(t) 是否等于 y1(tt0)y_1(t - t_0)

    # 线性 linear

    # 定义

    连续时间线性系统定义

    对任意 x1(t)y1(t)x1(t) → y1(t)x2(t)y2(t)x2(t) → y2(t) ,总有 ax1(t)+bx2(t)ay1(t)+by2(t)ax_1(t) + bx_2(t) → ay_1(t) + by_2(t) ,其中 a 与 b 是任意常数。

    连续线性时间系统

    离散时间线性系统同理。

    实际上线性等价于以下两个性质之和:

    x1(t)+x2(t)y1(t)+y2(t)x_1(t) + x_2(t) → y_1(t) + y_2(t)(可加性 / 叠加性)

    ax1(t)ay1(t)ax_1(t) → ay_1(t)(齐次性 / 均匀性)

    # 零输入 — 零输出特性

    根据线性系统的齐次性,可得出:线性系统当输入为零(即没有输入)时,系统的输出响应也必然为零(即没有输出响应)。这就是所谓线性系统的零输入 — 零输出特性

    # 非线性系统举例

    eg1 : 满足可加性,但不满足齐次性

    y(t)=Re{x(t)}y(t) = Re\{x(t)\}

    eg2 : 满足齐次性但不满足可加性

    y(t)=[x(t)]2x(t)y(t) = \frac{[x'(t)]^2}{x(t)}

    # 线性系统的分解特性

    如果一个系统是线性的,则只要我们能够把输入信号分解成若干个简单信号的线性组合,且知道该系统对每一个简单信号的响应,就可以很方便地通过线性组合而得到系统对原输入信号的响应。即:

    {x(t)=kakxk(t)xk(t)yk(t)y(t)=kakyk(t)\begin{cases} x(t)=\sum_k a_k x_k (t)\\ x_k(t)\to y_k(t) \end{cases} \rArr y(t)=\sum_k a_k y_k (t)

    这一思想是信号与系统分析理论和方法的基础

    # 特殊系统

    # 增量线性系统

    在工程实际中,有一类系统并不满足线性系统的要求。但是这类系统的输出响应的增量与输入信号的增量之间满足线性特性。这类系统称为增量线性系统

    eg:考虑系统 y(t)=x(t)+2y(t) = x(t) + 2 ,由于:

    x1(t)y1(t)=x1(t)+2x2(t)y2(t)=x2(t)+2x_1(t) → y_1(t) = x_1(t) + 2\\ x_2(t) → y_2(t) = x_2(t) + 2

    该系统既不满足齐次性,也不满足可加性,但当考查输入的增量与输出的增量之间的关系时,有

    Δy(t)=y1(t)y2(t)=x1(t)x2(t)=Δx(t)Δy(t) = y_1(t) - y_2(t) = x_1(t) - x_2(t) = Δx(t)

    可见输入的增量与输出的增量之间是满足线性关系的,它是一个增量线性系统。

    增量线性系统可以等效为一个线性系统再加上一部分与输入无关的响应。

    线性增量系统

    增量线性系统的响应包括:

    • 零状态响应:当 y0(t)=0y_0(t) = 0 时,y(t)=y1(t)y(t) = y_1(t) 。此时系统处于零初始状态,故将 y1(t)y_1(t) 称为系统的零状态响应。
    • 零输入响应:当 x(t)=0x(t) = 0 时,有 y1(t)=0y_1(t) = 0y(t)=y0(t)y(t) = y_0(t) ,因此将 y0(t)y_0(t) 称为系统的零输入响应。

    # 线性时不变系统(LTI)

    线性 + 时不变性 = LTI

    基本事实:如果知道线性时不变系统对某些输入信号的响应,则可以知道该系统对许多其它输入信号的响应。

    线性时不变(LTI)系统是本课程主要关注的系统

    # 线性性质

    # 根据系统框图描述建立微分方程

    # 对于连续系统

    步骤如下

    1. 首先判断其阶数,阶数 = 积分器个数
    2. 设信号传递 ** 最后一个积分器输出为 x(t)x(t) ** ,则各积分器的输入输出均可确定
    3. 对每个加法器列⽅程:输入和 = 输出和
    4. 联立方程消去中间变量 x(t)x(t) ,得到关于输出信号 y(t) 的微分方程

    # 对于离散系统

    步骤如下

    1. 首先判断其阶数,阶数 = 延时单元个数
    2. 设信号传递第一个延时单元输入为 x(k) ,则各延时单元的输入输出均可确定
    3. 对每个加法器列⽅程:输入和 = 输出和
    4. 联立方程消去中间变量 x(t)x(t) ,得到关于输出信号 y(t) 的微分方程
    \ No newline at end of file diff --git a/ie/signals-and-systems/zT/index.html b/ie/signals-and-systems/zT/index.html index 390f9f2e..70025906 100644 --- a/ie/signals-and-systems/zT/index.html +++ b/ie/signals-and-systems/zT/index.html @@ -1,4 +1,4 @@ -z 变换 - 信号与系统 - 电子信息 | Balloon Party = 風船のパーティー = fuusen no party

    # 公式表

    # 基本变换对

    常用 (双边) z 变换对:

    常用z变换对

    # 二阶系统

    二阶系统

    # 性质表

    双边 z 变换的性质:

    z变换性质

    单边 z 变换的性质:

    单边z变换性质

    # 双边 z 变换

    # 定义

    X(z)=n=+x[n]znX(z)=\sum_{n=-\infin}^{+\infin}x[n]z^{-n}

    称为 x[n]x[n] 的 z 变换,后面简记为 zT,其中 z=rejωz=re^{j\omega} 为复数。

    r=1r=1z=ejωz=e^{j\omega} 位于单位圆上,则为 DTFT(离散时间傅里叶变换),也就是说 DTFT 是双边 z 变换在 r=1r=1 或者说是在单位圆上的特例,即 X(z)r=1=X(ejω)X(z)|_{r=1}=X(e^{j\omega})

    # zT 与 DTFT 的关系

    X(z)=X(rejω)=n=+x[n]rnejωn=n=+[x[n]rn]ejωn=F{x[n]rn}\begin{aligned} X(z)=X(re^{j\omega})&=\sum_{n=-\infin}^{+\infin}x[n]r^{-n}e^{-j\omega n}\\ &=\sum_{n=-\infin}^{+\infin}\big[x[n]r^{-n}\big]e^{-j\omega n}\\ &=\mathscr{F}\{x[n]r^{-n}\} \end{aligned}

    所以 x[n]x[n] 的 zT 就是 x[n]rnx[n]r^{-n} 的 DTFT 。

    # 零极点图

    X(z)X(z) 是有理函数:

    X(z)=N(z)D(z)=Mi(sβi)i(sαi)X(z)=\frac{N(z)}{D(z)}=M\frac{\prod_{i}(s-\beta_i)}{\prod_{i}(s-\alpha_i)}

    • 分子多项式的根称为零点,画为圆圈 ⭕️
    • 分母多项式的根称为极点,画为交叉 ❌

    X(z)X(z) 的全部零点极点表示在 ss 平面上,就构成了零极点图

    零极点图上标出收敛域,可以唯一确定一个 X(z)X(z) ,最多与真实的 X(z)X(z) 相差一个常数因子 𝑀𝑀

    # ROC

    ⚠️不同的信号可能会有完全相同的拉普拉斯变换表达式,只是它们的收敛域不同。拉普拉斯变换的表达式只有连同相应的收敛域,才能和信号建立一一对应的关系。

    并非 zz 平面上的任何复数都能使 z 变换 X(z)X(z) 收敛,也不是任何信号的 z 变换都存在。

    和拉普拉斯变换一样,使 z 变换收敛的复数 zz 的集合,称为 z 变换 (zT) 的收敛域ROC,Region of Convergence)

    x[n]x[n] 可以写为若干部分的线性组合,各个部分分别进行 zT 有各自的收敛域,则 X(z)X(z) 的收敛域是各个收敛域的交集

    • ROC 总是 z 平面上以原点为中心的环形区域
    • ROC 的边界总是与 X(z)X(z) 的分母的根 (极点) 相对应
    • (因为)有理 z 变换在其 ROC 内无任何极点

    ⭐️按照 ROC 在 zz 平面上的分布可以分为:

    • 右边序列的 ROC 是某个圆的外部,但可能不包括 z=z=\infin

      • x[n]x[n] 左边界 <0\lt0 时,z=|z|=\infin 不在 ROC 内
      • 当且仅当 z=z=\infin 在 ROC 内时,x[n]x[n]n>0n\gt0 时总为 00x[n]x[n]因果序列
    • 左边序列的 ROC 是某个圆的内部,但可能不包括 z=0z=0

      • x[n]x[n] 右边界 >0\gt0 时,z=0|z|=0 不在 ROC 内
      • 当且仅当 z=0z=0 在 ROC 内时,x[n]x[n]n<0n\lt0 时总为 00x[n]x[n]反因果序列
    • 双边序列的 ROC 如果存在,一定是一个环形区域

    • 有限长序列的 ROC 是整个 z 平面

      • x[n]x[n] 左边界 <0\lt0 时,z=|z|=\infin 不在 ROC 内
      • x[n]x[n] 右边界 >0\gt0 时,z=0|z|=0 不在 ROC 内

    # 有限长序列

    x[n]x[n] 是有限长序列,定义于 [N1,N2][N_1,N_2] ,则有其 z 变换:

    X(z)=n=N1N2x[n]znX(z)=\sum_{n=N_1}^{N_2}x[n]z^{-n}

    此时既然是有限项求和,当 z 不等于零或无穷大时,和式中的每一项都是有限的,那么显然整个 z 平面上都是收敛的。

    如果 N1<0N_1\lt0,那么 x[n]x[n]n<0n<0 一定会有非零值,和式中包括 z 的正幂次项。当 z|z|\to\infin 时,涉及 z 的正幂次的那些项就成为无界的,因此收敛域不包括 z=|z|=\infin

    如果 N2>0N_2\gt0,那么 x[n]x[n]n>0n>0 一定会有非零值,和式中包括 z 的负幂次项。当 z0|z|\to0 时,涉及 z 的负幂次的那些项就成为无界的,因此收敛域不包括 z=0|z|=0

    # 右边序列

    x[n]x[n] 是右边序列,定义于 [N1,+)[N_1,+\infin) ,则有其 z 变换:

    X(z)=n=N1+x[n]znX(z)=\sum_{n=N_1}^{+\infin}x[n]z^{-n}

    z=r0ROC|z|=r_0\in\text{ROC} ,则有 X(z)X(z) 收敛:

    n=N1+x[n]zn=n=N1+x[n]r0n<\sum_{n=N_1}^{+\infin}|x[n]z^{-n}|=\sum_{n=N_1}^{+\infin}|x[n]r_0^{-n}|\lt\infin

    对任意 r1>r0r_1\gt r_0 有:

    n=N1+x[n]r1n=n=N1+x[n]r0n(r0r1)nn=N1+x[n]r0n(r0r1)N1<\sum_{n=N_1}^{+\infin}|x[n]r_1^{-n}|=\sum_{n=N_1}^{+\infin}|x[n]r_0^{-n}|\cdot(\frac{r_0}{r_1})^n\leq\sum_{n=N_1}^{+\infin}|x[n]r_0^{-n}|\cdot(\frac{r_0}{r_1})^{N_1}\lt\infin

    如果 N1<0N_1\lt0,那么 x[n]x[n]n<0n<0 一定会有非零值,和式中包括 z 的正幂次项。当 z|z|\to\infin 时,涉及 z 的正幂次的那些项就成为无界的,因此收敛域不包括 z=|z|=\infin

    于是就可得:

    • X(z)X(z) 为有理函数,则 ROC 必位于最外极点之外
    • N1<0N_1\lt0 时,z=|z|=\infin 不在 ROC 内
    • 当且仅当 z=z=\infin 在 ROC 内时,x[n]x[n] 为因果序列

    # 左边序列

    x[n]x[n] 是左边序列,定义于 (,N1](-\infin,N_1] ,则有其 z 变换:

    X(z)=n=N1x[n]znX(z)=\sum_{n=-\infin}^{N_1}x[n]z^{-n}

    z=r0ROC|z|=r_0\in\text{ROC} ,则有 X(z)X(z) 收敛:

    n=N1x[n]zn=n=N1x[n]r0n<\sum_{n=-\infin}^{N_1}|x[n]z^{-n}|=\sum_{n=-\infin}^{N_1}|x[n]r_0^{-n}|\lt\infin

    对任意 r1<r0r_1\lt r_0 有:

    n=N1x[n]r1n=n=N1x[n]r0n(r0r1)nn=N1x[n]r0n(r0r1)N1<\sum_{n=-\infin}^{N_1}|x[n]r_1^{-n}|=\sum_{n=-\infin}^{N_1}|x[n]r_0^{-n}|\cdot(\frac{r_0}{r_1})^n\leq\sum_{n=-\infin}^{N_1}|x[n]r_0^{-n}|\cdot(\frac{r_0}{r_1})^{N_1}\lt\infin

    如果 N1>0N_1\gt0,那么 x[n]x[n]n>0n>0 一定会有非零值,和式中包括 z 的负幂次项。当 z0|z|\to0 时,涉及 z 的负幂次的那些项就成为无界的,因此收敛域不包括 z=0|z|=0

    于是就可得:

    • X(z)X(z) 为有理函数,则 ROC 必位于最内极点之内内
    • N1>0N_1\gt0 时,z=0|z|=0 不在 ROC 内
    • 当且仅当 z=0z=0 在 ROC 内时,x[n]x[n] 为反因果序列

    # 双边序列

    x[n]x[n] 是双边序列,定义于 (,+)(-\infin,+\infin) ,则有其 z 变换:

    X(z)=n=+x[n]znX(z)=\sum_{n=\infin}^{+\infin}x[n]z^{-n}

    和有限长序列相反,此时收敛域不包括 z=|z|=\infin 也不包括 z=0|z|=0。此时的收敛域是 z 平面上的环形区域

    zN=aNz^N=a^N 的解为:z=aej2πNkz=ae^{j\frac{2\pi}{N}k}。均匀分布在半径为 aa 的圆上

    # z 反变换

    # 定义

    利用 zT 和 DTFT 的关系、DTFT 的反变换可以得到 zT 的反变换。

    (zT 和 DTFT 的关系:)

    X(z)=n=+x[n]rnejωnX(rejω)=n=+[x[n]rn]ejωn=F{x[n]rn}\begin{aligned} X(z)&=\sum_{n=-\infin}^{+\infin}x[n]r^{-n}e^{-j\omega n}\\ X(re^{j\omega})&=\sum_{n=-\infin}^{+\infin}\big[x[n]r^{-n}\big]e^{-j\omega n}\\ &=\mathscr{F}\{x[n]r^{-n}\} \end{aligned}

    进而由 DTFT 的反变换有:

    x[n]rn=12π2πX(rejω)ejωndωx[n]=12π2πX(rejω)rnejωndωx[n]=12π2πX(z)zndω\begin{aligned} x[n]r^{-n}&=\frac{1}{2\pi}\int_{2\pi}X(re^{j\omega})e^{j\omega n}d\omega\\ x[n]&=\frac{1}{2\pi}\int_{2\pi}X(re^{j\omega})r^ne^{j\omega n}d\omega\\ x[n]&=\frac{1}{2\pi}\int_{2\pi}X(z)z^{n}d\omega \end{aligned}

    z=rejωz=re^{j\omega} 两侧同时微分得到:dz=jrejωdω=jzdωdz=jre^{j\omega}d\omega=jzd\omega ;同时,当 ω\omega02π0\to2\pi 时,zz 沿着 ROC 内半径为 rr 的圆积分一周 ,所以可得到 z 反变换为

    x[n]=12πjX(z)zn1dzx[n]=\frac{1}{2\pi j}\oint X(z)z^{n-1}dz

    式中 \oint 记为在半径为 rr,以原点为中心的封闭圆上沿逆时针方向环绕 一周的积分。rr 的值可选为使 X(z)X(z) 收敛的任 何值也就是使 z=r|z|=r 的积分围线位于收敛域内的任何值。

    # 部分分式展开法

    1. X(z)X(z) 展开为部分分式
    2. 根据 X(z)X(z) 的 ROC ,确定每一项的 ROC
    3. 利用常用信号变换对与拉普拉斯变换性质,对每一项进行反变换

    # 幂级数展开法

    X(z)X(z) 的定义,将其展开为幂级数,有

    X(z)=+x[n]zn++x[2]z2+x[1]z+x[0]+x[1]z1+x[2]z2++x[n]zn+\begin{aligned} X(z)=&\cdots+x[-n]z^n+\cdots+x[-2]z^{2}+x[-1]z\\ &+x[0]+\\ &x[1]z^{-1}+x[2]z^{-2}+\cdots+x[n]z^{-n}+\cdots \end{aligned}

    展开式中 znz^{-n} 项的系数即为 x[n]x[n]

    泰勒级数展开法适合用来求解非有理函数形式 X(z)X(z) 的反变换。

    ⭐️当 X(z)X(z) 是有理函数时,可以通过长除的方法将其展开为幂级数:(但可能不易获得闭式表达)

    • 右边序列的展开式中应包含无限个 zz 的负幂项,所以要按降幂长除
    • 左边序列的展开式中应包含无限个 zz 的正幂项,要按升幂长除
    • 双边序列,先要将其分成对应信号的右边和左边的两部分,再分别按上述原则长除。

    # zT 的几何求值

    其方法与拉普拉斯变换时完全类似

    考查动点在单位圆上移动一周时,各极点矢量和零点矢量的长度与幅角变化的情况,即可反映频率特性。

    # 一般情况

    对有理函数形式的 X(z)X(z)

    X(z)=N(z)D(z)=Mi(sβi)i(sαi)X(z)=\frac{N(z)}{D(z)}=M\frac{\prod_i(s-\beta_i)}{\prod_i(s-\alpha_i)}

    可得:

    X(z)=MisβiisαiX(z)=i(sβi)i(sαi)\begin{aligned} |X(z)|&=|M|\frac{\prod_i|\vec{s}-\vec{\beta_i}|}{\prod_i|\vec{s}-\vec{\alpha_i}|}\\ \angle X(z)&=\sum_i\angle(\vec{s}-\vec{\beta_i})-\sum_i\angle(\vec{s}-\vec{\alpha_i}) \end{aligned}

    # 公式表

    # 基本变换对

    常用 (双边) z 变换对:

    常用z变换对

    # 二阶系统

    二阶系统

    # 性质表

    双边 z 变换的性质:

    z变换性质

    单边 z 变换的性质:

    单边z变换性质

    # 双边 z 变换

    # 定义

    X(z)=n=+x[n]znX(z)=\sum_{n=-\infin}^{+\infin}x[n]z^{-n}

    称为 x[n]x[n] 的 z 变换,后面简记为 zT,其中 z=rejωz=re^{j\omega} 为复数。

    r=1r=1z=ejωz=e^{j\omega} 位于单位圆上,则为 DTFT(离散时间傅里叶变换),也就是说 DTFT 是双边 z 变换在 r=1r=1 或者说是在单位圆上的特例,即 X(z)r=1=X(ejω)X(z)|_{r=1}=X(e^{j\omega})

    # zT 与 DTFT 的关系

    X(z)=X(rejω)=n=+x[n]rnejωn=n=+[x[n]rn]ejωn=F{x[n]rn}\begin{aligned} X(z)=X(re^{j\omega})&=\sum_{n=-\infin}^{+\infin}x[n]r^{-n}e^{-j\omega n}\\ &=\sum_{n=-\infin}^{+\infin}\big[x[n]r^{-n}\big]e^{-j\omega n}\\ &=\mathscr{F}\{x[n]r^{-n}\} \end{aligned}

    所以 x[n]x[n] 的 zT 就是 x[n]rnx[n]r^{-n} 的 DTFT 。

    # 零极点图

    X(z)X(z) 是有理函数:

    X(z)=N(z)D(z)=Mi(sβi)i(sαi)X(z)=\frac{N(z)}{D(z)}=M\frac{\prod_{i}(s-\beta_i)}{\prod_{i}(s-\alpha_i)}

    • 分子多项式的根称为零点,画为圆圈 ⭕️
    • 分母多项式的根称为极点,画为交叉 ❌

    X(z)X(z) 的全部零点极点表示在 ss 平面上,就构成了零极点图

    零极点图上标出收敛域,可以唯一确定一个 X(z)X(z) ,最多与真实的 X(z)X(z) 相差一个常数因子 𝑀𝑀

    # ROC

    ⚠️不同的信号可能会有完全相同的拉普拉斯变换表达式,只是它们的收敛域不同。拉普拉斯变换的表达式只有连同相应的收敛域,才能和信号建立一一对应的关系。

    并非 zz 平面上的任何复数都能使 z 变换 X(z)X(z) 收敛,也不是任何信号的 z 变换都存在。

    和拉普拉斯变换一样,使 z 变换收敛的复数 zz 的集合,称为 z 变换 (zT) 的收敛域ROC,Region of Convergence)

    x[n]x[n] 可以写为若干部分的线性组合,各个部分分别进行 zT 有各自的收敛域,则 X(z)X(z) 的收敛域是各个收敛域的交集

    • ROC 总是 z 平面上以原点为中心的环形区域
    • ROC 的边界总是与 X(z)X(z) 的分母的根 (极点) 相对应
    • (因为)有理 z 变换在其 ROC 内无任何极点

    ⭐️按照 ROC 在 zz 平面上的分布可以分为:

    • 右边序列的 ROC 是某个圆的外部,但可能不包括 z=z=\infin

      • x[n]x[n] 左边界 <0\lt0 时,z=|z|=\infin 不在 ROC 内
      • 当且仅当 z=z=\infin 在 ROC 内时,x[n]x[n]n>0n\gt0 时总为 00x[n]x[n]因果序列
    • 左边序列的 ROC 是某个圆的内部,但可能不包括 z=0z=0

      • x[n]x[n] 右边界 >0\gt0 时,z=0|z|=0 不在 ROC 内
      • 当且仅当 z=0z=0 在 ROC 内时,x[n]x[n]n<0n\lt0 时总为 00x[n]x[n]反因果序列
    • 双边序列的 ROC 如果存在,一定是一个环形区域

    • 有限长序列的 ROC 是整个 z 平面

      • x[n]x[n] 左边界 <0\lt0 时,z=|z|=\infin 不在 ROC 内
      • x[n]x[n] 右边界 >0\gt0 时,z=0|z|=0 不在 ROC 内

    # 有限长序列

    x[n]x[n] 是有限长序列,定义于 [N1,N2][N_1,N_2] ,则有其 z 变换:

    X(z)=n=N1N2x[n]znX(z)=\sum_{n=N_1}^{N_2}x[n]z^{-n}

    此时既然是有限项求和,当 z 不等于零或无穷大时,和式中的每一项都是有限的,那么显然整个 z 平面上都是收敛的。

    如果 N1<0N_1\lt0,那么 x[n]x[n]n<0n<0 一定会有非零值,和式中包括 z 的正幂次项。当 z|z|\to\infin 时,涉及 z 的正幂次的那些项就成为无界的,因此收敛域不包括 z=|z|=\infin

    如果 N2>0N_2\gt0,那么 x[n]x[n]n>0n>0 一定会有非零值,和式中包括 z 的负幂次项。当 z0|z|\to0 时,涉及 z 的负幂次的那些项就成为无界的,因此收敛域不包括 z=0|z|=0

    # 右边序列

    x[n]x[n] 是右边序列,定义于 [N1,+)[N_1,+\infin) ,则有其 z 变换:

    X(z)=n=N1+x[n]znX(z)=\sum_{n=N_1}^{+\infin}x[n]z^{-n}

    z=r0ROC|z|=r_0\in\text{ROC} ,则有 X(z)X(z) 收敛:

    n=N1+x[n]zn=n=N1+x[n]r0n<\sum_{n=N_1}^{+\infin}|x[n]z^{-n}|=\sum_{n=N_1}^{+\infin}|x[n]r_0^{-n}|\lt\infin

    对任意 r1>r0r_1\gt r_0 有:

    n=N1+x[n]r1n=n=N1+x[n]r0n(r0r1)nn=N1+x[n]r0n(r0r1)N1<\sum_{n=N_1}^{+\infin}|x[n]r_1^{-n}|=\sum_{n=N_1}^{+\infin}|x[n]r_0^{-n}|\cdot(\frac{r_0}{r_1})^n\leq\sum_{n=N_1}^{+\infin}|x[n]r_0^{-n}|\cdot(\frac{r_0}{r_1})^{N_1}\lt\infin

    如果 N1<0N_1\lt0,那么 x[n]x[n]n<0n<0 一定会有非零值,和式中包括 z 的正幂次项。当 z|z|\to\infin 时,涉及 z 的正幂次的那些项就成为无界的,因此收敛域不包括 z=|z|=\infin

    于是就可得:

    • X(z)X(z) 为有理函数,则 ROC 必位于最外极点之外
    • N1<0N_1\lt0 时,z=|z|=\infin 不在 ROC 内
    • 当且仅当 z=z=\infin 在 ROC 内时,x[n]x[n] 为因果序列

    # 左边序列

    x[n]x[n] 是左边序列,定义于 (,N1](-\infin,N_1] ,则有其 z 变换:

    X(z)=n=N1x[n]znX(z)=\sum_{n=-\infin}^{N_1}x[n]z^{-n}

    z=r0ROC|z|=r_0\in\text{ROC} ,则有 X(z)X(z) 收敛:

    n=N1x[n]zn=n=N1x[n]r0n<\sum_{n=-\infin}^{N_1}|x[n]z^{-n}|=\sum_{n=-\infin}^{N_1}|x[n]r_0^{-n}|\lt\infin

    对任意 r1<r0r_1\lt r_0 有:

    n=N1x[n]r1n=n=N1x[n]r0n(r0r1)nn=N1x[n]r0n(r0r1)N1<\sum_{n=-\infin}^{N_1}|x[n]r_1^{-n}|=\sum_{n=-\infin}^{N_1}|x[n]r_0^{-n}|\cdot(\frac{r_0}{r_1})^n\leq\sum_{n=-\infin}^{N_1}|x[n]r_0^{-n}|\cdot(\frac{r_0}{r_1})^{N_1}\lt\infin

    如果 N1>0N_1\gt0,那么 x[n]x[n]n>0n>0 一定会有非零值,和式中包括 z 的负幂次项。当 z0|z|\to0 时,涉及 z 的负幂次的那些项就成为无界的,因此收敛域不包括 z=0|z|=0

    于是就可得:

    • X(z)X(z) 为有理函数,则 ROC 必位于最内极点之内内
    • N1>0N_1\gt0 时,z=0|z|=0 不在 ROC 内
    • 当且仅当 z=0z=0 在 ROC 内时,x[n]x[n] 为反因果序列

    # 双边序列

    x[n]x[n] 是双边序列,定义于 (,+)(-\infin,+\infin) ,则有其 z 变换:

    X(z)=n=+x[n]znX(z)=\sum_{n=\infin}^{+\infin}x[n]z^{-n}

    和有限长序列相反,此时收敛域不包括 z=|z|=\infin 也不包括 z=0|z|=0。此时的收敛域是 z 平面上的环形区域

    zN=aNz^N=a^N 的解为:z=aej2πNkz=ae^{j\frac{2\pi}{N}k}。均匀分布在半径为 aa 的圆上

    # z 反变换

    # 定义

    利用 zT 和 DTFT 的关系、DTFT 的反变换可以得到 zT 的反变换。

    (zT 和 DTFT 的关系:)

    X(z)=n=+x[n]rnejωnX(rejω)=n=+[x[n]rn]ejωn=F{x[n]rn}\begin{aligned} X(z)&=\sum_{n=-\infin}^{+\infin}x[n]r^{-n}e^{-j\omega n}\\ X(re^{j\omega})&=\sum_{n=-\infin}^{+\infin}\big[x[n]r^{-n}\big]e^{-j\omega n}\\ &=\mathscr{F}\{x[n]r^{-n}\} \end{aligned}

    进而由 DTFT 的反变换有:

    x[n]rn=12π2πX(rejω)ejωndωx[n]=12π2πX(rejω)rnejωndωx[n]=12π2πX(z)zndω\begin{aligned} x[n]r^{-n}&=\frac{1}{2\pi}\int_{2\pi}X(re^{j\omega})e^{j\omega n}d\omega\\ x[n]&=\frac{1}{2\pi}\int_{2\pi}X(re^{j\omega})r^ne^{j\omega n}d\omega\\ x[n]&=\frac{1}{2\pi}\int_{2\pi}X(z)z^{n}d\omega \end{aligned}

    z=rejωz=re^{j\omega} 两侧同时微分得到:dz=jrejωdω=jzdωdz=jre^{j\omega}d\omega=jzd\omega ;同时,当 ω\omega02π0\to2\pi 时,zz 沿着 ROC 内半径为 rr 的圆积分一周 ,所以可得到 z 反变换为

    x[n]=12πjX(z)zn1dzx[n]=\frac{1}{2\pi j}\oint X(z)z^{n-1}dz

    式中 \oint 记为在半径为 rr,以原点为中心的封闭圆上沿逆时针方向环绕 一周的积分。rr 的值可选为使 X(z)X(z) 收敛的任 何值也就是使 z=r|z|=r 的积分围线位于收敛域内的任何值。

    # 部分分式展开法

    1. X(z)X(z) 展开为部分分式
    2. 根据 X(z)X(z) 的 ROC ,确定每一项的 ROC
    3. 利用常用信号变换对与拉普拉斯变换性质,对每一项进行反变换

    # 幂级数展开法

    X(z)X(z) 的定义,将其展开为幂级数,有

    X(z)=+x[n]zn++x[2]z2+x[1]z+x[0]+x[1]z1+x[2]z2++x[n]zn+\begin{aligned} X(z)=&\cdots+x[-n]z^n+\cdots+x[-2]z^{2}+x[-1]z\\ &+x[0]+\\ &x[1]z^{-1}+x[2]z^{-2}+\cdots+x[n]z^{-n}+\cdots \end{aligned}

    展开式中 znz^{-n} 项的系数即为 x[n]x[n]

    泰勒级数展开法适合用来求解非有理函数形式 X(z)X(z) 的反变换。

    ⭐️当 X(z)X(z) 是有理函数时,可以通过长除的方法将其展开为幂级数:(但可能不易获得闭式表达)

    • 右边序列的展开式中应包含无限个 zz 的负幂项,所以要按降幂长除
    • 左边序列的展开式中应包含无限个 zz 的正幂项,要按升幂长除
    • 双边序列,先要将其分成对应信号的右边和左边的两部分,再分别按上述原则长除。

    # zT 的几何求值

    其方法与拉普拉斯变换时完全类似

    考查动点在单位圆上移动一周时,各极点矢量和零点矢量的长度与幅角变化的情况,即可反映频率特性。

    # 一般情况

    对有理函数形式的 X(z)X(z)

    X(z)=N(z)D(z)=Mi(sβi)i(sαi)X(z)=\frac{N(z)}{D(z)}=M\frac{\prod_i(s-\beta_i)}{\prod_i(s-\alpha_i)}

    可得:

    X(z)=MisβiisαiX(z)=i(sβi)i(sαi)\begin{aligned} |X(z)|&=|M|\frac{\prod_i|\vec{s}-\vec{\beta_i}|}{\prod_i|\vec{s}-\vec{\alpha_i}|}\\ \angle X(z)&=\sum_i\angle(\vec{s}-\vec{\beta_i})-\sum_i\angle(\vec{s}-\vec{\alpha_i}) \end{aligned}

    从所有零点向 ss 点作零点矢量,从所有极点向 ss 点作极点矢量:

    • 所有零点矢量的长度之积 (不存在则为 1) 除以所有极点矢量的长度之积即为 X(z)|X(z)|
    • 所有零点矢量的幅角之和 (不存在则为 0) 减去所有极点矢量的幅角之和即为 \angX(z);

    ss 取为单位元上的点时,即为傅里叶变换的几何求值。考查 ss 在单位圆上移动时所有零、极点矢量的长度和幅角的变化,即可得出 X(ejω)X(e^{j\omega}) 的幅频特性和相频特性

    ⭐️ ROC 包括单位圆,则说明 x[n]x[n] 对应的傅里叶变换存在⚠️

    # zT 性质

    与 s 变换类似,在讨论 z 变换的许多性质时我们都要考虑其 ROC 的变化。

    # 线性

    x1[n]zTX1(z),ROC:R1x2[n]zTX2(z),ROC:R2ax1[n]+bx2[n]zTaX1(z)+bX2(z)x_1[n]\stackrel{zT}{\longleftrightarrow}X_1(z),\ \ \text{ROC}:R_1\\ x_2[n]\stackrel{zT}{\longleftrightarrow}X_2(z),\ \ \text{ROC}:R_2\\ \dArr\\ ax_1[n]+bx_2[n]\stackrel{zT}{\longleftrightarrow}aX_1(z)+bX_2(z)

    后者的 ROC 至少R1R2R_1\cap R_2 ,否则若 R1R2=R_1\cap R_2=\varnothing ,则 ax1[n]+bx2[n]ax_1[n]+bx_2[n] 的 LT 不存在。

    ROC 也有可能扩大:X1(z)X_1(z)X2(z)X_2(z) 线性组合时,若发生了零极点相抵消的现象,且当被抵消的极点恰好在 ROC 的边界上时,就会使 ROC 扩大。

    # 时移特性

    x[n]zTX(z),ROC:Rx[nn0]zTX(z)zn0x[n]\stackrel{zT}{\longleftrightarrow}X(z),\ \ \text{ROC}:R\\ \dArr\\ x[n-n_0]\stackrel{zT}{\longleftrightarrow}X(z)z^{-n_0}

    后者的 ROC 为 RR ,但是在 z=0z=0z=z=∞ 可能会有增删,这是由于 x[n]x[n] 的平移可能改变其在 xx 轴正半轴、负半轴的取值情况,信号时移可能会改变其因果性。

    # z 域尺度变换

    x[n]zTX[n],ROC:Rx[n]z0nzTX(zz0),ROC:z0Rx[n]\stackrel{zT}{\longleftrightarrow}X[n],\ \ \text{ROC}:R\\ \dArr\\ x[n]z_0^{n}\stackrel{zT}{\longleftrightarrow}X\big(\frac{z}{z_0}\big),\ \ \text{ROC}:|z_0|R

    ⚠️这里 z0R|z_0|R 的意思是将 RR 的边界缩放为 z0z_0 倍⚠️,也就是整体的模缩放为 z0z_0 倍。实际上 ROC 还会有一个 ω0\omega_0 的角度偏移(z0=z0ejω0z_0=|z_0|e^{j\omega_0}),所以零点和极点位置发生缩放和旋转。

    特别地,当 z0=ejω0z_0=e^{j\omega_0} 时,有 z0R=R|z_0|R=R,只有旋转没有缩放。在公式表中这一项被单独列出。

    连续时间的时域尺度变换的概念不能直接推广到离散时间中,因为离散时间变量仅仅定义在整数值上。相对应的,z 变换有时域反转、时域扩展这两个性质

    # 时域反转

    若信号在时域尺度变换,其 LT 的 ROC 在 ss 平面上作相反的尺度变换。

    x[n]zTX(z),ROC:Rx[n]zTX(z1),ROC:1Rx[n]\stackrel{zT}{\longleftrightarrow}X(z),\ \ \text{ROC}:R\\ \dArr\\ x[-n]\stackrel{zT}{\longleftrightarrow}X(z^{-1}),\ \ \text{ROC}:\frac{1}{R}

    ⚠️这里 1R\frac{1}{R} 的意思是 RR 收敛域边界倒置⚠️,零点、极点也将变为倒数。

    # 时域扩展

    定义在原有序列 x[n]x[n] 的各连续值之间插入 k1k-1 个零值序列:

    x(k)[n]={x[nk],nk的整数倍0,n不是k的整数倍x_{(k)}[n]=\begin{cases} x[\frac{n}{k}],&n\text{是}k\text{的整数倍}\\ 0,&n\text{不是}k\text{的整数倍} \end{cases}

    于是有:

    x[n]zTX(z),ROC:Rx(k)[n]zTX(zk),ROC:R1kx[n]\stackrel{zT}{\longleftrightarrow}X(z),\ \ \text{ROC}:R\\ \dArr\\ x_{(k)}[n]\stackrel{zT}{\longleftrightarrow}X(z^{k}),\ \ \text{ROC}:R^{\frac{1}{k}}

    ⚠️这里 1R\frac{1}{R} 的意思是 RR 收敛域边界倒置⚠️,零点、极点也将变为倒数。

    # 共轭对称性

    x[n]zTX(z),ROC:Rx[n]zTX(z),ROC:Rx[n]\stackrel{zT}{\longleftrightarrow}X(z),\ \ \text{ROC}:R\\ \dArr\\ x^*[n]\stackrel{zT}{\longleftrightarrow}X^*(z^*),\ \ \text{ROC}:R

    这里可以结合 X(z)=X(s)|X(z)|=|X(s^*)| 理解,模相同意味着二者总是零点极点相同......

    特别地,当 x[n]x[n]实信号时,有:

    x[n]=x[n]zTX(z)=X(z)=X(z)=X(z)x^*[n]=x[n]\stackrel{zT}{\longleftrightarrow} X^*(z^*)=X(z)=X^*(z)=X(z^*)

    这里 X(z)=X(z)X^*(z^*)=X(z) 等价于 X(z)=X(z)X^*(z)=X(z^*)

    因此,如果 x[n]x[n] 是实信号,且 X(z)X(z)z0z_0 有极点 (或零点),则 X(z)X(z) 一定在 z0z_0^* 也有极点 (或零点)。即实信号的 z 变换其零、极点必共轭成对出现

    # 卷积性质

    x1[n]zTX1(z),ROC:R1x2[n]zTX2(z),ROC:R2x1[n]x2[n]zTX1(z)X2(z)x_1[n]\stackrel{zT}{\longleftrightarrow}X_1(z),\ \ \text{ROC}:R_1\\ x_2[n]\stackrel{zT}{\longleftrightarrow}X_2(z),\ \ \text{ROC}:R_2\\ \dArr\\ x_1[n]*x_2[n]\stackrel{zT}{\longleftrightarrow}X_1(z)X_2(z)

    类似线性性质中,后者的 ROC 包括 R1R2R_1\cap R_2 ,否则若 R1R2=R_1\cap R_2=\varnothing ,则 x1[n]x2[n]x_1[n]*x_2[n] 的 LT 不存在。

    ROC 也有可能扩大:X1(z)X_1(z)X2(z)X_2(z) 相乘时,若发生了零极点相抵消的现象,且当被抵消的极点恰好在 ROC 的边界上时,就会使 ROC 扩大。

    # 时域差分

    y[n]=x[n]x[n1]=(δ[n]δ[n1])x[n]y[n]=x[n]-x[n-1]=\big(\delta[n]-\delta[n-1]\big)*x[n]

    y[n]y[n]x[n]x[n] 的的一次差分。可以认为是离散时间情况下的 “微分”。

    x[n]zTX(z),ROC:Rx[n]x[n1]zT(1z1)X(z)x[n]\stackrel{zT}{\longleftrightarrow}X(z),\ \ \text{ROC}:R\\ \dArr\\ x[n]-x[n-1]\stackrel{zT}{\longleftrightarrow}(1-z^{-1})X(z)

    后者的 ROC 至少是 RRz>0|z|>0 的相交。

    # z 域微分

    x[n]zTX(z),ROC:Rnx[n]zTzddzX(z),ROC:Rx[n]\stackrel{zT}{\longleftrightarrow}X(z),\ \ \text{ROC}:R\\ \dArr\\ nx[n]\stackrel{zT}{\longleftrightarrow}-z\frac{d}{dz}X(z),\ \ \text{ROC}:R\\

    # 时域累加

    w[n]=k=nx[k]=u[n]x[n]w[n]=\sum_{k=-\infin}^{n}x[k]=u[n]*x[n]

    w[n]w[n]x[n]x[n] 的的累加 / 求和。可以认为是离散时间情况下的 “积分”。

    x[n]zTX(z),ROC:Rk=nx[k]zT11z1X(z)x[n]\stackrel{zT}{\longleftrightarrow}X(z),\ \ \text{ROC}:R\\ \dArr\\ \sum_{k=-\infin}^{n}x[k]\stackrel{zT}{\longleftrightarrow}\frac{1}{1-z^{-1}}X(z)

    后者的 ROC 至少是 RRz>1|z|>1 的相交,也就是包括 RR 在单位圆外侧的部分。这是因为:

    k=nx[k]=x[n]u[n]\sum_{k=-\infin}^{n}x[k]=x[n]*u[n]

    # 初值与终值定理

    # 初值定理

    如果 x(t)x(t) 是因果信号,也就是 n<0n<0x[n]=0x[n]=0,则有初值定理

    x[0]=limzX(z)x[0]=\lim_{z\to\infin}X(z)

    或者非因果信号可以用单边 z 变换表示:

    x[0]=limzχ(z)x[0]=\lim_{z\to\infin}\chi(z)


    证明

    X(z)X(z) 按照定义式展开有:

    X(z)=x(0)+x(1)z1+x(2)z2++x(n)zn+\begin{aligned} X(z)=x(0)+x(1)z^{-1}+x(2)z^{-2}+\cdots+x(n)z^{-n}+\cdots \end{aligned}

    显然就有 x[0]=limzX(z)x[0]=\lim_{z\to\infin}X(z)

    # 终值定理

    如果 x[n]x[n] 是因果信号,X(z)X(z) 除了在 z=1z=1 可以有单阶极点外,其余极点均在单位圆内

    上述条件的目的在于保证终值存在

    则有终值定理

    limnx[n]=limz1(z1)X(z)\lim_{n\to\infin}x[n]=\lim_{z\to1}(z-1)X(z)


    证明

    limz1(z1)X(z)=limz1n=1(x[n+1]x[n])zn=limmn=1m(x[n+1]x[n])=limmn=1m(x[0]x[1]+x[1]x[0]++x[m+1]x[m])=limmx[m+1]=limnx[n]\begin{aligned} \lim_{z\to1}(z-1)X(z) &=\lim_{z\to1}\sum_{n=-1}^{\infin}\big(x[n+1]-x[n]\big)z^{-n}\\ &=\lim_{m\to\infin}\sum_{n=-1}^{m}\big(x[n+1]-x[n]\big)\\ &=\lim_{m\to\infin}\sum_{n=-1}^{m}\big(x[0]-x[-1]+x[1]-x[0]+\cdots+x[m+1]-x[m]\big)\\ &=\lim_{m\to\infin}x[m+1]\\ &=\lim_{n\to\infin}x[n] \end{aligned}

    # zT 分析 LTI 系统

    以卷积特性为基础,可建立 LTI 系统的 z 变换分析方法,即:

    Y(z)=X(z)H(z)Y(z)=X(z)\cdot H(z)

    设离散时间 LTI 系统的单位脉冲响应为 h[n]h[n],则 h[n]h[n] 的 z 变换 H(z)H(z) 称为系统函数或转移函数,它描述了一个离散时间 LTI 系统并体现其系统特性。其中 H(z)H(z)h[n]h[n] 的 z 变换,称为系统函数转移函数传递函数

    # 特征函数

    x[n]=ez0tx[n]=e^{z_0t} 时,响应为:

    y[n]=H(z0)ez0ty[n]=H(z_0)e^{z_0t}

    # 因果性

    • 系统是因果的当且仅当:如果 n<0n<0h[n]=0h[n]=0,则 H(z)H(z) 的 ROC 是最外部极点的外部,并且包括 z=z=\infin
    • 系统是反因果的当且仅当:如果 n>0n>0h[n]=0h[n]=0,则 H(z)H(z) 的 ROC 是最内部极点的内部,并且包括 z=0z=0

    对于 H(z)H(z) 表达式已知,判断因果性、反因果性与否可以直接看 limz0H(z)\lim_{z\to0}H(z)limzH(z)\lim_{z\to\infin}H(z) 是否收敛存在。

    # 稳定性

    LTI 系统若稳定,即 n=h[n]<\sum_{n=-\infin}^{\infin}|h[n]|\lt\infin,也即 h[n]h[n] 的 DTFT 存在,则 H(z)H(z) 的 ROC 必包括单位圆;反之亦然。所以 LTI 系统稳定的充要条件是 ROC 包含单位圆。

    对于因果稳定的 LTI 系统,其系统函数 H(z)H(z) 如果存在极点,则全部极点必须位于单位圆内。

    # 线性常微分方程

    * 线性常系数微分方程

    如果由线性常系数微分方程描述的系统满足初始松弛条件,则系统是因果 LTI 的,其 H(z)H(z) 的 ROC 必是最外侧极点的外部。

    对于由线性常系数微分方程描述的 LTI 系统:

    k=0Naky[nk]=k=0Mbkx[nk]\sum_{k=0}^{N}a_k\cdot y[n-k]=\sum_{k=0}^{M}b_k\cdot x[n-k]

    两边进行 z 变换,可得:

    k=0NakzkY(z)=k=0MbkzkX(z)Y(z)k=0Nakzk=X(z)k=0Mbkzk\sum_{k=0}^{N}a_k\cdot z^{-k}\cdot Y(z)=\sum_{k=0}^{M}b_k\cdot z^{-k}\cdot X(z)\\ Y(z)\sum_{k=0}^{N}a_k\cdot z^{-k}=X(z)\sum_{k=0}^{M}b_k\cdot z^{-k}

    所以:

    H(z)=Y(z)X(z)=k=0Mbkzkk=0NakzkH(z)={Y(z)\over X(z)}={\sum_{k=0}^{M}b_{k}z^{-k}\over\sum_{k=0}^{N}a_{k}z^{-k}}

    可得到 H(z)H(z) 是一个有理函数。系统的单位冲激响应 h[n]h[n] 可由 H(z)H(z) 反变换获得。

    # 因果 LTI 系统的方框图表示

    • 级联
    • 并联
    • 反馈联接

    # 直接型表示

    # 级联型表示

    H(z)H(z) 因式分解,在无重阶极点时可得:

    H(z)=k=0Mbkzkk=0Nakzk=b0a0k=1M(1+μkz1)k=1N(1+ηkzk)H(z)={\sum_{k=0}^{M}b_{k}z^{-k}\over\sum_{k=0}^{N}a_{k}z^{-k}} =\frac{b_0}{a_0}{\sum_{k=1}^{M}(1+\mu_k z^{-1})\over\sum_{k=1}^{N}(1+\eta_kz^{-k})}

    M=NM=N 为偶数时,可得:

    H(z)=b0a0k=1N21+β1kz1+β2kz21+α1kz1+α2kz2=b0a0k=1N2Hk(z)H(z)=\frac{b_0}{a_0}\sum_{k=1}^{\frac{N}{2}}{1+\beta_{1k} z^{-1}+\beta_{2k}z^{-2}\over1+\alpha_{1k}z^{-1}+\alpha_{2k}z^{-2}} =\frac{b_0}{a_0}\sum_{k=1}^{\frac{N}{2}}H_k(z)

    其中 Hk(z)H_k(z) 是二阶子系统。系统级联表示为 N2\frac{N}{2} 个二阶子系统的级联如下:

    级联表示

    # 并联型表示

    H(z)H(z) 展开为部分分式,在无重阶极点时可得:

    H(z)=b0a0+k=1NAk1+ηkz1H(z)=\frac{b_0}{a_0}+\sum_{k=1}^{N}\frac{A_k}{1+\eta_kz^{-1}}

    M=NM=N 为偶数时,可得:

    H(z)=b0a0+k=1N2r0k+r1kz11+α1kz1+α2kz2=b0a0+k=1N2Hk(z)H(z)=\frac{b_0}{a_0}+\sum_{k=1}^{\frac{N}{2}}{r_{0k}+r_{1k} z^{-1}\over1+\alpha_{1k}z^{-1}+\alpha_{2k}z^{-2}} =\frac{b_0}{a_0}+\sum_{k=1}^{\frac{N}{2}}H_k'(z)

    其中 Hk(z)H_k'(z) 是二阶子系统。系统级联表示为 N2\frac{N}{2} 个二阶子系统的并联如下:

    并联表示

    # 单边 z 变换

    Unilateral z Transform

    单边 z 变换是仅考虑大于 1-1 之后部分信号的双边 z 变换,也就是 x[n]u[n]x[n]\cdot u[n] 的双边 z 变换:

    χ(z)=n=0+x[n]zn\chi(z)=\sum_{n=0}^{+\infin}x[n]z^{-n}

    单边 z 变换的反变换与同 ROC 的双边 z 变换的反变换相同:

    x[n]=12πjX(z)zn1dz,n0x[n]=\frac{1}{2\pi j}\oint X(z)z^{n-1}dz,\ \ n\geq0

    X(z)X(z)χ(z)\chi(z) 不同,是因为 x[n]x[n]n<0n<0 的部分对 X(z)X(z) 有作用,而对 χ(z)\chi(z) 没有任何作用所致。

    由于单边拉普拉斯变换不存在类似双边拉普拉斯变换中与 ROC 相关的多义性,一般不再强调其 ROC ,任何单边 z 变换的 ROC 一定是最外侧极点的外部。

    因果信号的双边 z 变换和单边 z 变换完全相同。⭐️从而可以利用双边 LT 的基本变换对。

    # 性质

    由于单边拉普拉斯变换是特殊的双边拉普拉斯变换,因此其大部分性质与双边拉普拉斯变换相同,主要的不同是时移特性,分为:

    • 时移特性
      • 时延
      • 时间超前

    # 时延

    x[n]UzTχ(z)x[n1]UzTz1χ(z)+x[1]x[n2]UzTz2χ(z)+x[1]z1+x[2]......x[n]\stackrel{UzT}{\longleftrightarrow}\chi(z)\\ \dArr\\ \begin{aligned} x[n-1]&\stackrel{UzT}{\longleftrightarrow}z^{-1}\chi(z)+x[-1]\\ x[n-2]&\stackrel{UzT}{\longleftrightarrow}z^{-2}\chi(z)+x[-1]z^{-1}+x[-2]\\ &...... \end{aligned}

    # 时间超前

    x[n]UzTχ(z)x[n+1]UzTz(χ(z)x[0])x[n]\stackrel{UzT}{\longleftrightarrow}\chi(z)\\ \dArr\\ x[n+1]\stackrel{UzT}{\longleftrightarrow}z\big(\chi(z)-x[0]\big)

    # 差分方程

    利用单边 z 变换和时延性质来解具有非零初始条件的线性常系数差分方程,单边 z 变换在将线性常系数差分方程变换为 z 域代数方程时,可以自动将方程的初始条件引入,因而在解决增量线性系统问题时特别有用。

    往往假定系统为因果系统。该条件下得到的结果在进行单边 z 反变换的时候可以确定为右边信号

    以二阶为例(最高二阶差分),将初始条件带入后,整理得到如下形式:

    Aγ(z)=By[1]+Cy[2]+Dχ(z)γ(z)=BAy[1]+CAy[2]+DAχ(z)\begin{aligned} A\gamma(z)&=By[-1]+Cy[-2]+D\chi(z)\\ \gamma(z)&=\frac{B}{A}y[-1]+\frac{C}{A}y[-2]+\frac{D}{A}\chi(z) \end{aligned}

    其中 DAχ(z)\frac{D}{A}\chi(z) 对应的单边 z 反变换为零状态响应BAy[1]+CAy[2]\frac{B}{A}y[-1]+\frac{C}{A}y[-2] 对应的单边 z 反变换为零输入响应

    γ(z)=k111z1+\begin{aligned} \gamma(z)=k_1\frac{1}{1-z^{-1}}+\cdots \end{aligned}

    其中 k111z1k_1\frac{1}{1-z^{-1}}强迫响应k1k_1 为常数,也就是对应的单边 z 反变换为 k1u[n]k_1u[n] 形式的才是强迫响应;其他都是自然响应

    \ No newline at end of file +c-16-25.333-24-45-24-59z"/>)

    从所有零点向 ss 点作零点矢量,从所有极点向 ss 点作极点矢量:

    • 所有零点矢量的长度之积 (不存在则为 1) 除以所有极点矢量的长度之积即为 X(z)|X(z)|
    • 所有零点矢量的幅角之和 (不存在则为 0) 减去所有极点矢量的幅角之和即为 \angX(z);

    ss 取为单位元上的点时,即为傅里叶变换的几何求值。考查 ss 在单位圆上移动时所有零、极点矢量的长度和幅角的变化,即可得出 X(ejω)X(e^{j\omega}) 的幅频特性和相频特性

    ⭐️ ROC 包括单位圆,则说明 x[n]x[n] 对应的傅里叶变换存在⚠️

    # zT 性质

    与 s 变换类似,在讨论 z 变换的许多性质时我们都要考虑其 ROC 的变化。

    # 线性

    x1[n]zTX1(z),ROC:R1x2[n]zTX2(z),ROC:R2ax1[n]+bx2[n]zTaX1(z)+bX2(z)x_1[n]\stackrel{zT}{\longleftrightarrow}X_1(z),\ \ \text{ROC}:R_1\\ x_2[n]\stackrel{zT}{\longleftrightarrow}X_2(z),\ \ \text{ROC}:R_2\\ \dArr\\ ax_1[n]+bx_2[n]\stackrel{zT}{\longleftrightarrow}aX_1(z)+bX_2(z)

    后者的 ROC 至少R1R2R_1\cap R_2 ,否则若 R1R2=R_1\cap R_2=\varnothing ,则 ax1[n]+bx2[n]ax_1[n]+bx_2[n] 的 LT 不存在。

    ROC 也有可能扩大:X1(z)X_1(z)X2(z)X_2(z) 线性组合时,若发生了零极点相抵消的现象,且当被抵消的极点恰好在 ROC 的边界上时,就会使 ROC 扩大。

    # 时移特性

    x[n]zTX(z),ROC:Rx[nn0]zTX(z)zn0x[n]\stackrel{zT}{\longleftrightarrow}X(z),\ \ \text{ROC}:R\\ \dArr\\ x[n-n_0]\stackrel{zT}{\longleftrightarrow}X(z)z^{-n_0}

    后者的 ROC 为 RR ,但是在 z=0z=0z=z=∞ 可能会有增删,这是由于 x[n]x[n] 的平移可能改变其在 xx 轴正半轴、负半轴的取值情况,信号时移可能会改变其因果性。

    # z 域尺度变换

    x[n]zTX[n],ROC:Rx[n]z0nzTX(zz0),ROC:z0Rx[n]\stackrel{zT}{\longleftrightarrow}X[n],\ \ \text{ROC}:R\\ \dArr\\ x[n]z_0^{n}\stackrel{zT}{\longleftrightarrow}X\big(\frac{z}{z_0}\big),\ \ \text{ROC}:|z_0|R

    ⚠️这里 z0R|z_0|R 的意思是将 RR 的边界缩放为 z0z_0 倍⚠️,也就是整体的模缩放为 z0z_0 倍。实际上 ROC 还会有一个 ω0\omega_0 的角度偏移(z0=z0ejω0z_0=|z_0|e^{j\omega_0}),所以零点和极点位置发生缩放和旋转。

    特别地,当 z0=ejω0z_0=e^{j\omega_0} 时,有 z0R=R|z_0|R=R,只有旋转没有缩放。在公式表中这一项被单独列出。

    连续时间的时域尺度变换的概念不能直接推广到离散时间中,因为离散时间变量仅仅定义在整数值上。相对应的,z 变换有时域反转、时域扩展这两个性质

    # 时域反转

    若信号在时域尺度变换,其 LT 的 ROC 在 ss 平面上作相反的尺度变换。

    x[n]zTX(z),ROC:Rx[n]zTX(z1),ROC:1Rx[n]\stackrel{zT}{\longleftrightarrow}X(z),\ \ \text{ROC}:R\\ \dArr\\ x[-n]\stackrel{zT}{\longleftrightarrow}X(z^{-1}),\ \ \text{ROC}:\frac{1}{R}

    ⚠️这里 1R\frac{1}{R} 的意思是 RR 收敛域边界倒置⚠️,零点、极点也将变为倒数。

    # 时域扩展

    定义在原有序列 x[n]x[n] 的各连续值之间插入 k1k-1 个零值序列:

    x(k)[n]={x[nk],nk的整数倍0,n不是k的整数倍x_{(k)}[n]=\begin{cases} x[\frac{n}{k}],&n\text{是}k\text{的整数倍}\\ 0,&n\text{不是}k\text{的整数倍} \end{cases}

    于是有:

    x[n]zTX(z),ROC:Rx(k)[n]zTX(zk),ROC:R1kx[n]\stackrel{zT}{\longleftrightarrow}X(z),\ \ \text{ROC}:R\\ \dArr\\ x_{(k)}[n]\stackrel{zT}{\longleftrightarrow}X(z^{k}),\ \ \text{ROC}:R^{\frac{1}{k}}

    ⚠️这里 1R\frac{1}{R} 的意思是 RR 收敛域边界倒置⚠️,零点、极点也将变为倒数。

    # 共轭对称性

    x[n]zTX(z),ROC:Rx[n]zTX(z),ROC:Rx[n]\stackrel{zT}{\longleftrightarrow}X(z),\ \ \text{ROC}:R\\ \dArr\\ x^*[n]\stackrel{zT}{\longleftrightarrow}X^*(z^*),\ \ \text{ROC}:R

    这里可以结合 X(z)=X(s)|X(z)|=|X(s^*)| 理解,模相同意味着二者总是零点极点相同......

    特别地,当 x[n]x[n]实信号时,有:

    x[n]=x[n]zTX(z)=X(z)=X(z)=X(z)x^*[n]=x[n]\stackrel{zT}{\longleftrightarrow} X^*(z^*)=X(z)=X^*(z)=X(z^*)

    这里 X(z)=X(z)X^*(z^*)=X(z) 等价于 X(z)=X(z)X^*(z)=X(z^*)

    因此,如果 x[n]x[n] 是实信号,且 X(z)X(z)z0z_0 有极点 (或零点),则 X(z)X(z) 一定在 z0z_0^* 也有极点 (或零点)。即实信号的 z 变换其零、极点必共轭成对出现

    # 卷积性质

    x1[n]zTX1(z),ROC:R1x2[n]zTX2(z),ROC:R2x1[n]x2[n]zTX1(z)X2(z)x_1[n]\stackrel{zT}{\longleftrightarrow}X_1(z),\ \ \text{ROC}:R_1\\ x_2[n]\stackrel{zT}{\longleftrightarrow}X_2(z),\ \ \text{ROC}:R_2\\ \dArr\\ x_1[n]*x_2[n]\stackrel{zT}{\longleftrightarrow}X_1(z)X_2(z)

    类似线性性质中,后者的 ROC 包括 R1R2R_1\cap R_2 ,否则若 R1R2=R_1\cap R_2=\varnothing ,则 x1[n]x2[n]x_1[n]*x_2[n] 的 LT 不存在。

    ROC 也有可能扩大:X1(z)X_1(z)X2(z)X_2(z) 相乘时,若发生了零极点相抵消的现象,且当被抵消的极点恰好在 ROC 的边界上时,就会使 ROC 扩大。

    # 时域差分

    y[n]=x[n]x[n1]=(δ[n]δ[n1])x[n]y[n]=x[n]-x[n-1]=\big(\delta[n]-\delta[n-1]\big)*x[n]

    y[n]y[n]x[n]x[n] 的的一次差分。可以认为是离散时间情况下的 “微分”。

    x[n]zTX(z),ROC:Rx[n]x[n1]zT(1z1)X(z)x[n]\stackrel{zT}{\longleftrightarrow}X(z),\ \ \text{ROC}:R\\ \dArr\\ x[n]-x[n-1]\stackrel{zT}{\longleftrightarrow}(1-z^{-1})X(z)

    后者的 ROC 至少是 RRz>0|z|>0 的相交。

    # z 域微分

    x[n]zTX(z),ROC:Rnx[n]zTzddzX(z),ROC:Rx[n]\stackrel{zT}{\longleftrightarrow}X(z),\ \ \text{ROC}:R\\ \dArr\\ nx[n]\stackrel{zT}{\longleftrightarrow}-z\frac{d}{dz}X(z),\ \ \text{ROC}:R\\

    # 时域累加

    w[n]=k=nx[k]=u[n]x[n]w[n]=\sum_{k=-\infin}^{n}x[k]=u[n]*x[n]

    w[n]w[n]x[n]x[n] 的的累加 / 求和。可以认为是离散时间情况下的 “积分”。

    x[n]zTX(z),ROC:Rk=nx[k]zT11z1X(z)x[n]\stackrel{zT}{\longleftrightarrow}X(z),\ \ \text{ROC}:R\\ \dArr\\ \sum_{k=-\infin}^{n}x[k]\stackrel{zT}{\longleftrightarrow}\frac{1}{1-z^{-1}}X(z)

    后者的 ROC 至少是 RRz>1|z|>1 的相交,也就是包括 RR 在单位圆外侧的部分。这是因为:

    k=nx[k]=x[n]u[n]\sum_{k=-\infin}^{n}x[k]=x[n]*u[n]

    # 初值与终值定理

    # 初值定理

    如果 x(t)x(t) 是因果信号,也就是 n<0n<0x[n]=0x[n]=0,则有初值定理

    x[0]=limzX(z)x[0]=\lim_{z\to\infin}X(z)

    或者非因果信号可以用单边 z 变换表示:

    x[0]=limzχ(z)x[0]=\lim_{z\to\infin}\chi(z)


    证明

    X(z)X(z) 按照定义式展开有:

    X(z)=x(0)+x(1)z1+x(2)z2++x(n)zn+\begin{aligned} X(z)=x(0)+x(1)z^{-1}+x(2)z^{-2}+\cdots+x(n)z^{-n}+\cdots \end{aligned}

    显然就有 x[0]=limzX(z)x[0]=\lim_{z\to\infin}X(z)

    # 终值定理

    如果 x[n]x[n] 是因果信号,X(z)X(z) 除了在 z=1z=1 可以有单阶极点外,其余极点均在单位圆内

    上述条件的目的在于保证终值存在

    则有终值定理

    limnx[n]=limz1(z1)X(z)\lim_{n\to\infin}x[n]=\lim_{z\to1}(z-1)X(z)


    证明

    limz1(z1)X(z)=limz1n=1(x[n+1]x[n])zn=limmn=1m(x[n+1]x[n])=limmn=1m(x[0]x[1]+x[1]x[0]++x[m+1]x[m])=limmx[m+1]=limnx[n]\begin{aligned} \lim_{z\to1}(z-1)X(z) &=\lim_{z\to1}\sum_{n=-1}^{\infin}\big(x[n+1]-x[n]\big)z^{-n}\\ &=\lim_{m\to\infin}\sum_{n=-1}^{m}\big(x[n+1]-x[n]\big)\\ &=\lim_{m\to\infin}\sum_{n=-1}^{m}\big(x[0]-x[-1]+x[1]-x[0]+\cdots+x[m+1]-x[m]\big)\\ &=\lim_{m\to\infin}x[m+1]\\ &=\lim_{n\to\infin}x[n] \end{aligned}

    # zT 分析 LTI 系统

    以卷积特性为基础,可建立 LTI 系统的 z 变换分析方法,即:

    Y(z)=X(z)H(z)Y(z)=X(z)\cdot H(z)

    设离散时间 LTI 系统的单位脉冲响应为 h[n]h[n],则 h[n]h[n] 的 z 变换 H(z)H(z) 称为系统函数或转移函数,它描述了一个离散时间 LTI 系统并体现其系统特性。其中 H(z)H(z)h[n]h[n] 的 z 变换,称为系统函数转移函数传递函数

    # 特征函数

    x[n]=ez0tx[n]=e^{z_0t} 时,响应为:

    y[n]=H(z0)ez0ty[n]=H(z_0)e^{z_0t}

    # 因果性

    • 系统是因果的当且仅当:如果 n<0n<0h[n]=0h[n]=0,则 H(z)H(z) 的 ROC 是最外部极点的外部,并且包括 z=z=\infin
    • 系统是反因果的当且仅当:如果 n>0n>0h[n]=0h[n]=0,则 H(z)H(z) 的 ROC 是最内部极点的内部,并且包括 z=0z=0

    对于 H(z)H(z) 表达式已知,判断因果性、反因果性与否可以直接看 limz0H(z)\lim_{z\to0}H(z)limzH(z)\lim_{z\to\infin}H(z) 是否收敛存在。

    # 稳定性

    LTI 系统若稳定,即 n=h[n]<\sum_{n=-\infin}^{\infin}|h[n]|\lt\infin,也即 h[n]h[n] 的 DTFT 存在,则 H(z)H(z) 的 ROC 必包括单位圆;反之亦然。所以 LTI 系统稳定的充要条件是 ROC 包含单位圆。

    对于因果稳定的 LTI 系统,其系统函数 H(z)H(z) 如果存在极点,则全部极点必须位于单位圆内。

    # 线性常微分方程

    * 线性常系数微分方程

    如果由线性常系数微分方程描述的系统满足初始松弛条件,则系统是因果 LTI 的,其 H(z)H(z) 的 ROC 必是最外侧极点的外部。

    对于由线性常系数微分方程描述的 LTI 系统:

    k=0Naky[nk]=k=0Mbkx[nk]\sum_{k=0}^{N}a_k\cdot y[n-k]=\sum_{k=0}^{M}b_k\cdot x[n-k]

    两边进行 z 变换,可得:

    k=0NakzkY(z)=k=0MbkzkX(z)Y(z)k=0Nakzk=X(z)k=0Mbkzk\sum_{k=0}^{N}a_k\cdot z^{-k}\cdot Y(z)=\sum_{k=0}^{M}b_k\cdot z^{-k}\cdot X(z)\\ Y(z)\sum_{k=0}^{N}a_k\cdot z^{-k}=X(z)\sum_{k=0}^{M}b_k\cdot z^{-k}

    所以:

    H(z)=Y(z)X(z)=k=0Mbkzkk=0NakzkH(z)={Y(z)\over X(z)}={\sum_{k=0}^{M}b_{k}z^{-k}\over\sum_{k=0}^{N}a_{k}z^{-k}}

    可得到 H(z)H(z) 是一个有理函数。系统的单位冲激响应 h[n]h[n] 可由 H(z)H(z) 反变换获得。

    # 因果 LTI 系统的方框图表示

    • 级联
    • 并联
    • 反馈联接

    # 直接型表示

    # 级联型表示

    H(z)H(z) 因式分解,在无重阶极点时可得:

    H(z)=k=0Mbkzkk=0Nakzk=b0a0k=1M(1+μkz1)k=1N(1+ηkzk)H(z)={\sum_{k=0}^{M}b_{k}z^{-k}\over\sum_{k=0}^{N}a_{k}z^{-k}} =\frac{b_0}{a_0}{\sum_{k=1}^{M}(1+\mu_k z^{-1})\over\sum_{k=1}^{N}(1+\eta_kz^{-k})}

    M=NM=N 为偶数时,可得:

    H(z)=b0a0k=1N21+β1kz1+β2kz21+α1kz1+α2kz2=b0a0k=1N2Hk(z)H(z)=\frac{b_0}{a_0}\sum_{k=1}^{\frac{N}{2}}{1+\beta_{1k} z^{-1}+\beta_{2k}z^{-2}\over1+\alpha_{1k}z^{-1}+\alpha_{2k}z^{-2}} =\frac{b_0}{a_0}\sum_{k=1}^{\frac{N}{2}}H_k(z)

    其中 Hk(z)H_k(z) 是二阶子系统。系统级联表示为 N2\frac{N}{2} 个二阶子系统的级联如下:

    级联表示

    # 并联型表示

    H(z)H(z) 展开为部分分式,在无重阶极点时可得:

    H(z)=b0a0+k=1NAk1+ηkz1H(z)=\frac{b_0}{a_0}+\sum_{k=1}^{N}\frac{A_k}{1+\eta_kz^{-1}}

    M=NM=N 为偶数时,可得:

    H(z)=b0a0+k=1N2r0k+r1kz11+α1kz1+α2kz2=b0a0+k=1N2Hk(z)H(z)=\frac{b_0}{a_0}+\sum_{k=1}^{\frac{N}{2}}{r_{0k}+r_{1k} z^{-1}\over1+\alpha_{1k}z^{-1}+\alpha_{2k}z^{-2}} =\frac{b_0}{a_0}+\sum_{k=1}^{\frac{N}{2}}H_k'(z)

    其中 Hk(z)H_k'(z) 是二阶子系统。系统级联表示为 N2\frac{N}{2} 个二阶子系统的并联如下:

    并联表示

    # 单边 z 变换

    Unilateral z Transform

    单边 z 变换是仅考虑大于 1-1 之后部分信号的双边 z 变换,也就是 x[n]u[n]x[n]\cdot u[n] 的双边 z 变换:

    χ(z)=n=0+x[n]zn\chi(z)=\sum_{n=0}^{+\infin}x[n]z^{-n}

    单边 z 变换的反变换与同 ROC 的双边 z 变换的反变换相同:

    x[n]=12πjX(z)zn1dz,n0x[n]=\frac{1}{2\pi j}\oint X(z)z^{n-1}dz,\ \ n\geq0

    X(z)X(z)χ(z)\chi(z) 不同,是因为 x[n]x[n]n<0n<0 的部分对 X(z)X(z) 有作用,而对 χ(z)\chi(z) 没有任何作用所致。

    由于单边拉普拉斯变换不存在类似双边拉普拉斯变换中与 ROC 相关的多义性,一般不再强调其 ROC ,任何单边 z 变换的 ROC 一定是最外侧极点的外部。

    因果信号的双边 z 变换和单边 z 变换完全相同。⭐️从而可以利用双边 LT 的基本变换对。

    # 性质

    由于单边拉普拉斯变换是特殊的双边拉普拉斯变换,因此其大部分性质与双边拉普拉斯变换相同,主要的不同是时移特性,分为:

    • 时移特性
      • 时延
      • 时间超前

    # 时延

    x[n]UzTχ(z)x[n1]UzTz1χ(z)+x[1]x[n2]UzTz2χ(z)+x[1]z1+x[2]......x[n]\stackrel{UzT}{\longleftrightarrow}\chi(z)\\ \dArr\\ \begin{aligned} x[n-1]&\stackrel{UzT}{\longleftrightarrow}z^{-1}\chi(z)+x[-1]\\ x[n-2]&\stackrel{UzT}{\longleftrightarrow}z^{-2}\chi(z)+x[-1]z^{-1}+x[-2]\\ &...... \end{aligned}

    # 时间超前

    x[n]UzTχ(z)x[n+1]UzTz(χ(z)x[0])x[n]\stackrel{UzT}{\longleftrightarrow}\chi(z)\\ \dArr\\ x[n+1]\stackrel{UzT}{\longleftrightarrow}z\big(\chi(z)-x[0]\big)

    # 差分方程

    利用单边 z 变换和时延性质来解具有非零初始条件的线性常系数差分方程,单边 z 变换在将线性常系数差分方程变换为 z 域代数方程时,可以自动将方程的初始条件引入,因而在解决增量线性系统问题时特别有用。

    往往假定系统为因果系统。该条件下得到的结果在进行单边 z 反变换的时候可以确定为右边信号

    以二阶为例(最高二阶差分),将初始条件带入后,整理得到如下形式:

    Aγ(z)=By[1]+Cy[2]+Dχ(z)γ(z)=BAy[1]+CAy[2]+DAχ(z)\begin{aligned} A\gamma(z)&=By[-1]+Cy[-2]+D\chi(z)\\ \gamma(z)&=\frac{B}{A}y[-1]+\frac{C}{A}y[-2]+\frac{D}{A}\chi(z) \end{aligned}

    其中 DAχ(z)\frac{D}{A}\chi(z) 对应的单边 z 反变换为零状态响应BAy[1]+CAy[2]\frac{B}{A}y[-1]+\frac{C}{A}y[-2] 对应的单边 z 反变换为零输入响应

    γ(z)=k111z1+\begin{aligned} \gamma(z)=k_1\frac{1}{1-z^{-1}}+\cdots \end{aligned}

    其中 k111z1k_1\frac{1}{1-z^{-1}}强迫响应k1k_1 为常数,也就是对应的单边 z 反变换为 k1u[n]k_1u[n] 形式的才是强迫响应;其他都是自然响应

    \ No newline at end of file diff --git a/index.html b/index.html index 2e562dbb..4ab3f6e3 100644 --- a/index.html +++ b/index.html @@ -1 +1 @@ -Balloon Party = 風船のパーティー = fuusen no party

    Categories

    Post List

    5.9k words 5 mins.

    如果能把图 GGG 画在平面上,使得除顶点外,边与边之间没有交叉,称 GGG 可嵌入平面,或称 GGG 是可平面图。 可平面图 GGG 的边不交叉的一种画法,称为 GGG 的一种平面嵌入, GGG 的平面嵌入表示的图称为平面图。 可平面图概念和平面图概念有时可以等同看待; 图的平面性问题主要涉及如下几个方面: 平面图的性质 平面图的判定 平面嵌入方法 (平面性算法) 涉及图的平面性问题的拓扑不变量 # 平面图的性质 一个平面图 GGG 把平面分成若干连通片,这些连通片称为 GGG 的区域,或 GGG 的一个面。GGG 的面组成的集合用 Φ\PhiΦ...
    3.7k words 3 mins.

    磁盘的 block 和文件系统的 block 不是一回事: 磁盘的 block :往往是固定的 512 B; 文件系统的 block 是基本单位,可以自定义,不一定和磁盘的 block 大小相等。 文件系统分配空间的基本单位往往是页面大小 4 KB , 文件系统结构:(自上而下分层) application programs,应用程序 logical file system,逻辑文件系统 file-organization module,文件组织模块 basic file system,基本文件系统 I/O control,I/O...
    1.6k words 1 mins.

    # 逻辑电平规范与抗噪 # 逻辑电平规范 ⭐️保证输出的电平作为下一级输入是有效的 # CMOS # TTL # 噪声容限 ⭐️保证输出的电平作为下一级输入是有效的基础上,输出与输入的界限相差即为噪声容限 电路抗噪度的一个衡量标准叫做噪声容限 (noisemargin),用伏特表示。对于一个给定的逻辑电路,有两个噪声容限值: 噪声容限高电平 (VNHV_{NH}VNH​) (Noisemargin High) 噪声容限低电平 (VNLV_{NL}VNL​) (Noisemargin...
    6.1k words 6 mins.

    # 边着色问题 图的边着色,本质上是对应实际问题中的 “划分” 问题或 “分类” 问题。 对 GGG 的边进行着色,若相邻边着不同颜色,则称对 GGG 进行正常边着色; 在对 GGG 正常边着色时,着相同颜色的边集称为该正常边着色的一个色组; 如果能用 kkk 种颜色对图 GGG 进行正常边着色,称 GGG 是 kkk 边可着色的; 对 GGG 进行正常边着色需要的最少颜色数称为 GGG...
    2.6k words 2 mins.

    I/O 的访问、管理分为两种类型:(或者说按照编址类型划分) I/O 指令(独立编址) 通过 I/O 端口号访问设备管理器 涉及模态转换、状态更迭时才采用 内存映射 I/O (统一编址) 设备的寄存器或存储被映射到内存物理地址空间中; 通过内存 load/store 指令完成 I/O 操作; MMU 设置映射,硬件跳线或程序在启动时设置地址; 实际上绝大多数采用的方式 I/O 控制: 轮询,或者程序直接控制 通过忙等待(轮询)检查 I/O 控制器状态 完全交给 CPU 中断 等待中断来得到 I/O 控制器状态信息 中断到来前 CPU 可以干其他事(CPU、I/O...
    1.1k words 1 mins.

    # 基础元件 # 555 Timer 可能会考哦 mono-stable one-shot (单稳态触发器) astable multi-vibration oscillator (非稳态多谐振荡器) f=1.44(R1+2R2)C1DC=(R1+R2R1+2R2)100%f=\frac{1.44}{(R_1+2R_2)C_1}\\ DC=\big(\frac{R_1+R_2}{R_1+2R_2}\big)100\% f=(R1​+2R2​)C1​1.44​DC=(R1​+2R2​R1​+R2​​)100% # S-R...
    2k words 2 mins.

    # 素理想 非交换环上也可以定义素理想、极大理想,但这里只考虑交换环上的素理想和极大理想! # 定义 设 RRR 是交换环,PPP 是 RRR 的真理想。若对任意 a,b∈Ra,b\in Ra,b∈R,ab∈Pab\in Pab∈P 蕴涵 a∈Pa\in Pa∈P 或 b∈Pb\in Pb∈P ,则称 PPP 为 RRR 的素理想 (prime ideal)。 素理想的概念 b 本质上来自数论中的基本结论:若 ppp 为素数,则 p|ab\Rightarrow p|a\or p|b。 # 整数环的素理想 设 nnn...
    885 words 1 mins.

    # 采样定理 略 # ADC # 运算放大器 运算放大器是一种线性放大器。 两个输入 反相 同相 和一个输出 特点: 很高的电压增益 很大的输入阻抗 很低的输出阻抗 运算放大器的 3 种使用方式: 用做反相放大器时,图中,虚地处电压为 0 V ,由于很大的输入阻抗没有电流流入运算放大器,输入电压处 VinV_{in}Vin​ 所有电流流向输出电压处 VoutV_{out}Vout​ ,可得: VoutVin=−RfRi\frac{V_{out}}{V_{in}}=-\frac{R_f}{R_i} Vin​Vout​​=−Ri​Rf​​ 其中 RfR_fRf​...
    18k words 16 mins.

    # 公式表 # 基本变换对 常用 (双边) z 变换对: # 二阶系统 # 性质表 双边 z 变换的性质: 单边 z 变换的性质: # 双边 z 变换 # 定义 X(z)=∑n=−∞+∞x[n]z−nX(z)=\sum_{n=-\infin}^{+\infin}x[n]z^{-n} X(z)=n=−∞∑+∞​x[n]z−n 称为 x[n]x[n]x[n] 的 z 变换,后面简记为 zT,其中 z=rejωz=re^{j\omega}z=rejω 为复数。 若 r=1r=1r=1 ,z=ejωz=e^{j\omega}z=ejω 位于单位圆上,则为...
    \ No newline at end of file +Balloon Party = 風船のパーティー = fuusen no party

    Categories

    Post List

    5.9k words 5 mins.

    如果能把图 GGG 画在平面上,使得除顶点外,边与边之间没有交叉,称 GGG 可嵌入平面,或称 GGG 是可平面图。 可平面图 GGG 的边不交叉的一种画法,称为 GGG 的一种平面嵌入, GGG 的平面嵌入表示的图称为平面图。 可平面图概念和平面图概念有时可以等同看待; 图的平面性问题主要涉及如下几个方面: 平面图的性质 平面图的判定 平面嵌入方法 (平面性算法) 涉及图的平面性问题的拓扑不变量 # 平面图的性质 一个平面图 GGG 把平面分成若干连通片,这些连通片称为 GGG 的区域,或 GGG 的一个面。GGG 的面组成的集合用 Φ\PhiΦ...
    3.7k words 3 mins.

    磁盘的 block 和文件系统的 block 不是一回事: 磁盘的 block :往往是固定的 512 B; 文件系统的 block 是基本单位,可以自定义,不一定和磁盘的 block 大小相等。 文件系统分配空间的基本单位往往是页面大小 4 KB , 文件系统结构:(自上而下分层) application programs,应用程序 logical file system,逻辑文件系统 file-organization module,文件组织模块 basic file system,基本文件系统 I/O control,I/O...
    1.6k words 1 mins.

    # 逻辑电平规范与抗噪 # 逻辑电平规范 ⭐️保证输出的电平作为下一级输入是有效的 # CMOS # TTL # 噪声容限 ⭐️保证输出的电平作为下一级输入是有效的基础上,输出与输入的界限相差即为噪声容限 电路抗噪度的一个衡量标准叫做噪声容限 (noisemargin),用伏特表示。对于一个给定的逻辑电路,有两个噪声容限值: 噪声容限高电平 (VNHV_{NH}VNH​) (Noisemargin High) 噪声容限低电平 (VNLV_{NL}VNL​) (Noisemargin...
    6.1k words 6 mins.

    # 边着色问题 图的边着色,本质上是对应实际问题中的 “划分” 问题或 “分类” 问题。 对 GGG 的边进行着色,若相邻边着不同颜色,则称对 GGG 进行正常边着色; 在对 GGG 正常边着色时,着相同颜色的边集称为该正常边着色的一个色组; 如果能用 kkk 种颜色对图 GGG 进行正常边着色,称 GGG 是 kkk 边可着色的; 对 GGG 进行正常边着色需要的最少颜色数称为 GGG...
    2.6k words 2 mins.

    I/O 的访问、管理分为两种类型:(或者说按照编址类型划分) I/O 指令(独立编址) 通过 I/O 端口号访问设备管理器 涉及模态转换、状态更迭时才采用 内存映射 I/O (统一编址) 设备的寄存器或存储被映射到内存物理地址空间中; 通过内存 load/store 指令完成 I/O 操作; MMU 设置映射,硬件跳线或程序在启动时设置地址; 实际上绝大多数采用的方式 I/O 控制: 轮询,或者程序直接控制 通过忙等待(轮询)检查 I/O 控制器状态 完全交给 CPU 中断 等待中断来得到 I/O 控制器状态信息 中断到来前 CPU 可以干其他事(CPU、I/O...
    1.1k words 1 mins.

    # 基础元件 # 555 Timer 可能会考哦 mono-stable one-shot (单稳态触发器) astable multi-vibration oscillator (非稳态多谐振荡器) f=1.44(R1+2R2)C1DC=(R1+R2R1+2R2)100%f=\frac{1.44}{(R_1+2R_2)C_1}\\ DC=\big(\frac{R_1+R_2}{R_1+2R_2}\big)100\% f=(R1​+2R2​)C1​1.44​DC=(R1​+2R2​R1​+R2​​)100% # S-R...
    2k words 2 mins.

    # 素理想 非交换环上也可以定义素理想、极大理想,但这里只考虑交换环上的素理想和极大理想! # 定义 设 RRR 是交换环,PPP 是 RRR 的真理想。若对任意 a,b∈Ra,b\in Ra,b∈R,ab∈Pab\in Pab∈P 蕴涵 a∈Pa\in Pa∈P 或 b∈Pb\in Pb∈P ,则称 PPP 为 RRR 的素理想 (prime ideal)。 素理想的概念 b 本质上来自数论中的基本结论:若 ppp 为素数,则 p|ab\Rightarrow p|a\or p|b。 # 整数环的素理想 设 nnn...
    885 words 1 mins.

    # 采样定理 略 # ADC # 运算放大器 运算放大器是一种线性放大器。 两个输入 反相 同相 和一个输出 特点: 很高的电压增益 很大的输入阻抗 很低的输出阻抗 运算放大器的 3 种使用方式: 用做反相放大器时,图中,虚地处电压为 0 V ,由于很大的输入阻抗没有电流流入运算放大器,输入电压处 VinV_{in}Vin​ 所有电流流向输出电压处 VoutV_{out}Vout​ ,可得: VoutVin=−RfRi\frac{V_{out}}{V_{in}}=-\frac{R_f}{R_i} Vin​Vout​​=−Ri​Rf​​ 其中 RfR_fRf​...
    18k words 16 mins.

    # 公式表 # 基本变换对 常用 (双边) z 变换对: # 二阶系统 # 性质表 双边 z 变换的性质: 单边 z 变换的性质: # 双边 z 变换 # 定义 X(z)=∑n=−∞+∞x[n]z−nX(z)=\sum_{n=-\infin}^{+\infin}x[n]z^{-n} X(z)=n=−∞∑+∞​x[n]z−n 称为 x[n]x[n]x[n] 的 z 变换,后面简记为 zT,其中 z=rejωz=re^{j\omega}z=rejω 为复数。 若 r=1r=1r=1 ,z=ejωz=e^{j\omega}z=ejω 位于单位圆上,则为...
    \ No newline at end of file diff --git a/js/app.js b/js/app.js index 9e00eb1f..603c6adf 100644 --- a/js/app.js +++ b/js/app.js @@ -1,4 +1,4 @@ -// build time:Fri Jun 28 2024 18:11:47 GMT+0800 (中国标准时间) +// build time:Mon Jul 01 2024 11:40:12 GMT+0800 (中国标准时间) var e={version:"0.2.5",hostname:"https://fuuzen.github.io",root:"/",statics:"/",favicon:{normal:"images/favicon.ico",hidden:"images/failure.ico"},darkmode:false,auto_scroll:true,js:{valine:"gh/amehime/MiniValine@4.2.2-beta10/dist/MiniValine.min.js",chart:"npm/frappe-charts@1.5.0/dist/frappe-charts.min.iife.min.js",copy_tex:"npm/katex@0.12.0/dist/contrib/copy-tex.min.js",fancybox:"combine/npm/jquery@3.5.1/dist/jquery.min.js,npm/@fancyapps/fancybox@3.5.7/dist/jquery.fancybox.min.js,npm/justifiedGallery@3.8.1/dist/js/jquery.justifiedGallery.min.js"},css:{valine:"css/comment.css",katex:"npm/katex@0.12.0/dist/katex.min.css",mermaid:"css/mermaid.css",fancybox:"combine/npm/@fancyapps/fancybox@3.5.7/dist/jquery.fancybox.min.css,npm/justifiedGallery@3.8.1/dist/css/justifiedGallery.min.css"},loader:{start:true,"switch":false},search:{appID:"Z7A3XW4R2I",apiKey:"12db1ad54372045549ef465881c17e743",indexName:"my-hexo-blog",hits:{per_page:10}},valine:{appId:"ZjtofcGYicZjb65Bkc4DBrMK-MdYXbMMI",appKey:"XdEJOK09W9oW6x0aehYIatY3",placeholder:"ヽ(○´∀`)ノ♪",avatar:"mp",pageSize:10,lang:"en",visitor:true,NoRecordIP:false,serverURLs:null,powerMode:true,tagMeta:{visitor:"新朋友",master:"主人",friend:"小伙伴",investor:"金主粑粑"},tagColor:{master:"var(--color-orange)",friend:"var(--color-aqua)",investor:"var(--color-pink)"},tagMember:{master:null,friend:null,investor:null}},quicklink:{timeout:3e3,priority:true},audio:[{title:"🌈二次元",list:["https://music.163.com/#/playlist?id=9222813860"]}],fireworks:["rgba(255,182,185,.9)","rgba(250,227,217,.9)","rgba(187,222,214,.9)","rgba(138,198,209,.9)"]};const t=function(e,t){return Math.floor(Math.random()*(t-e+1))+e};const a=function(){return u("main > .inner").offsetHeight};const n=function(e,t,a){if(a){t()}else{var n=document.createElement("script");n.onload=n.onreadystatechange=function(e,a){if(a||!n.readyState||/loaded|complete/.test(n.readyState)){n.onload=n.onreadystatechange=null;n=undefined;if(!a&&t)setTimeout(t,0)}};n.src=e;document.head.appendChild(n)}};const i=function(t,a){var n=e[t][a];if(n.indexOf("npm")>-1||n.indexOf("gh")>-1||n.indexOf("combine")>-1)return"//fastly.jsdelivr.net/"+n;if(n.indexOf("http")>-1)return n;return m+n};const r=function(e,t,a){if(LOCAL[e]){n(i("js",e),t||function(){window[e]=true},a||window[e])}};const s=function(e,t){if(window["css"+e])return;if(LOCAL[e]){document.head.createChild("link",{rel:"stylesheet",href:i("css",e)});window["css"+e]=true}};const o=function(e){var t=e.text||e.textContent||e.innerHTML||"";var a=e.parentNode;a.removeChild(e);var n=document.createElement("script");if(e.id){n.id=e.id}if(e.className){n.className=e.className}if(e.type){n.type=e.type}if(e.src){n.src=e.src;n.async=false}if(e.dataset.pjax!==undefined){n.dataset.pjax=""}if(t!==""){n.appendChild(document.createTextNode(t))}a.appendChild(n)};const c=function(e,t,a){var n={targets:typeof t=="number"?e.parentNode:document.scrollingElement,duration:500,easing:"easeInOutQuad",scrollTop:t||(typeof e=="number"?e:e?e.top()+document.documentElement.scrollTop-R:0),complete:function(){a&&a()}};anime(n)};const l=function(e,t,a){var n={};var i="none";switch(t){case 0:n={opacity:[1,0]};break;case 1:n={opacity:[0,1]};i="block";break;case"bounceUpIn":n={begin:function(t){e.display("block")},translateY:[{value:-60,duration:200},{value:10,duration:200},{value:-5,duration:200},{value:0,duration:200}],opacity:[0,1]};i="block";break;case"shrinkIn":n={begin:function(t){e.display("block")},scale:[{value:1.1,duration:300},{value:1,duration:200}],opacity:1};i="block";break;case"slideRightIn":n={begin:function(t){e.display("block")},translateX:[100,0],opacity:[0,1]};i="block";break;case"slideRightOut":n={translateX:[0,100],opacity:[1,0]};break;default:n=t;i=t.display;break}anime(Object.assign({targets:e,duration:200,easing:"linear"},n)).finished.then(function(){e.display(i);a&&a()})};const d={get:function(e){return localStorage.getItem(e)},set:function(e,t){localStorage.setItem(e,t);return t},del:function(e){localStorage.removeItem(e)}};const u=function(e,t){t=t||document;if(e.indexOf("#")===0){return t.getElementById(e.replace("#",""))}return t.querySelector(e)};u.all=function(e,t){t=t||document;return t.querySelectorAll(e)};u.each=function(e,t,a){return u.all(e,a).forEach(t)};Object.assign(HTMLElement.prototype,{createChild:function(e,t,a){var n=document.createElement(e);Object.assign(n,t);switch(a){case"after":this.insertAfter(n);break;case"replace":this.innerHTML="";default:this.appendChild(n)}return n},wrap:function(e){var t=document.createElement("div");Object.assign(t,e);this.parentNode.insertBefore(t,this);this.parentNode.removeChild(this);t.appendChild(this)},height:function(e){if(e){this.style.height=typeof e=="number"?e+"rem":e}return this.getBoundingClientRect().height},width:function(e){if(e){this.style.width=typeof e=="number"?e+"rem":e}return this.getBoundingClientRect().width},top:function(){return this.getBoundingClientRect().top},left:function(){return this.getBoundingClientRect().left},attr:function(e,t){if(t===null){return this.removeAttribute(e)}if(t){this.setAttribute(e,t);return this}else{return this.getAttribute(e)}},insertAfter:function(e){var t=this.parentNode;if(t.lastChild==this){t.appendChild(e)}else{t.insertBefore(e,this.nextSibling)}},display:function(e){if(e==null){return this.style.display}else{this.style.display=e;return this}},child:function(e){return u(e,this)},find:function(e){return u.all(e,this)},_class:function(e,t,a){var n=t.indexOf(" ")?t.split(" "):[t];var i=this;n.forEach(function(t){if(e=="toggle"){i.classList.toggle(t,a)}else{i.classList[e](t)}})},addClass:function(e){this._class("add",e);return this},removeClass:function(e){this._class("remove",e);return this},toggleClass:function(e,t){this._class("toggle",e,t);return this},hasClass:function(e){return this.classList.contains(e)}});var f=null;const h=/mobile/i.test(window.navigator.userAgent);const p=function(e,t){var a={type:"audio",mode:"random",btns:["play-pause","music"],controls:["mode","backward","play-pause","forward","volume"],events:{"play-pause":function(t){if(i.paused){e.player.play()}else{e.player.pause()}},music:function(e){if(r.el.hasClass("show")){r.hide()}else{r.el.addClass("show");s.scroll().title()}}}},n={random:function(e){return Math.floor(Math.random()*e)},parse:function(e){var t=[];[["music.163.com.*song.*id=(\\d+)","netease","song"],["music.163.com.*album.*id=(\\d+)","netease","album"],["music.163.com.*artist.*id=(\\d+)","netease","artist"],["music.163.com.*playlist.*id=(\\d+)","netease","playlist"],["music.163.com.*discover/toplist.*id=(\\d+)","netease","playlist"],["y.qq.com.*song/(\\w+).html","tencent","song"],["y.qq.com.*album/(\\w+).html","tencent","album"],["y.qq.com.*singer/(\\w+).html","tencent","artist"],["y.qq.com.*playsquare/(\\w+).html","tencent","playlist"],["y.qq.com.*playlist/(\\w+).html","tencent","playlist"],["xiami.com.*song/(\\w+)","xiami","song"],["xiami.com.*album/(\\w+)","xiami","album"],["xiami.com.*artist/(\\w+)","xiami","artist"],["xiami.com.*collect/(\\w+)","xiami","playlist"]].forEach(function(a){var n=new RegExp(a[0]);var i=n.exec(e);if(i!==null){t=[a[1],a[2],i[1]]}});return t},fetch:function(e){var t=[];return new Promise(function(a,i){e.forEach(function(e){var i=n.parse(e);if(i[0]){var r=JSON.stringify(i);var s=d.get(r);if(s){t.push.apply(t,JSON.parse(s));a(t)}else{fetch("https://api.i-meto.com/meting/api?server="+i[0]+"&type="+i[1]+"&id="+i[2]+"&r="+Math.random()).then(function(e){return e.json()}).then(function(e){d.set(r,JSON.stringify(e));t.push.apply(t,e);a(t)}).catch(function(e){})}}else{t.push(e);a(t)}})})},secondToTime:function(e){var t=function(e){return isNaN(e)?"00":e<10?"0"+e:""+e};var a=Math.floor(e/3600);var n=Math.floor((e-a*3600)/60);var i=Math.floor(e-a*3600-n*60);return(a>0?[a,n,i]:[n,i]).map(t).join(":")},nameMap:{dragStart:h?"touchstart":"mousedown",dragMove:h?"touchmove":"mousemove",dragEnd:h?"touchend":"mouseup"}},i=null;e.player={_id:n.random(999999),group:true,load:function(e){var t="";var a=this;if(e&&e.length>0){if(this.options.rawList!==e){this.options.rawList=e;s.clear()}}else{t="none";this.pause()}for(var n in g.el){g.el[n].display(t)}return this},fetch:function(){var e=this;return new Promise(function(t,a){if(s.data.length>0){t()}else{if(e.options.rawList){var i=[];e.options.rawList.forEach(function(t,a){i.push(new Promise(function(i,r){var o=a;var c;if(!t.list){o=0;e.group=false;c=[t]}else{e.group=true;c=t.list}n.fetch(c).then(function(e){s.add(o,e);i()})}))});Promise.all(i).then(function(){t(true)})}}}).then(function(t){if(t){s.create();m.create();e.mode()}})},mode:function(){var e=s.data.length;if(!e||s.errnum==e)return;var t=m.step=="next"?1:-1;var a=function(){var a=s.index+t;if(a>e||a<0){a=m.step=="next"?0:e-1}s.index=a};var i=function(){var t=n.random(e);if(s.index!==t){s.index=t}else{a()}};switch(this.options.mode){case"random":i();break;case"order":a();break;case"loop":if(m.step)a();if(s.index==-1)i();break}this.init()},"switch":function(e){if(typeof e=="number"&&e!=s.index&&s.current()&&!s.current().error){s.index=e;this.init()}},init:function(){var e=s.current();if(!e||e["error"]){this.mode();return}var t=false;if(!i.paused){t=true;this.stop()}i.attr("src",e.url);i.attr("title",e.name+" - "+e.artist);this.volume(d.get("_PlayerVolume")||"0.7");this.muted(d.get("_PlayerMuted"));p.create();if(this.options.type=="audio")l.create();if(t==true){this.play()}},play:function(){f&&f.player.pause();if(s.current().error){this.mode();return}var e=this;i.play().then(function(){s.scroll()}).catch(function(e){})},pause:function(){i.pause();document.title=y},stop:function(){i.pause();i.currentTime=0;document.title=y},seek:function(e){e=Math.max(e,0);e=Math.min(e,i.duration);i.currentTime=e;p.update(e/i.duration)},muted:function(e){if(e=="muted"){i.muted=e;d.set("_PlayerMuted",e);m.update(0)}else{d.del("_PlayerMuted");i.muted=false;m.update(i.volume)}},volume:function(e){if(!isNaN(e)){m.update(e);d.set("_PlayerVolume",e);i.volume=e}},mini:function(){r.hide()}};var r={el:null,create:function(){if(this.el)return;this.el=e.createChild("div",{className:"player-info",innerHTML:(e.player.options.type=="audio"?'
    ':"")+'
    '},"after");l.el=this.el.child(".preview");s.el=this.el.child(".playlist");m.el=this.el.child(".controller")},hide:function(){var e=this.el;e.addClass("hide");window.setTimeout(function(){e.removeClass("show hide")},300)}};var s={el:null,data:[],index:-1,errnum:0,add:function(e,t){var a=this;t.forEach(function(t,n){t.group=e;t.name=t.name||t.title||"Meida name";t.artist=t.artist||t.author||"Anonymous";t.cover=t.cover||t.pic;t.type=t.type||"normal";a.data.push(t)})},clear:function(){this.data=[];this.el.innerHTML="";if(this.index!==-1){this.index=-1;e.player.fetch()}},create:function(){var t=this.el;this.data.map(function(a,n){if(a.el)return;var r="list-"+e.player._id+"-"+a.group;var o=u("#"+r);if(!o){o=t.createChild("div",{id:r,className:e.player.group?"tab":"",innerHTML:"
      "});if(e.player.group){o.attr("data-title",e.player.options.rawList[a.group]["title"]).attr("data-id",e.player._id)}}a.el=o.child("ol").createChild("li",{title:a.name+" - "+a.artist,innerHTML:''+a.name+""+a.artist+"",onclick:function(t){var a=t.currentTarget;if(s.index===n&&p.el){if(i.paused){e.player.play()}else{e.player.seek(i.duration*p.percent(t,a))}return}e.player.switch(n);e.player.play()}});return a});he()},current:function(){return this.data[this.index]},scroll:function(){var e=this.current();var t=this.el.child("li.active");t&&t.removeClass("active");var a=this.el.child(".tab.active");a&&a.removeClass("active");t=this.el.find(".nav li")[e.group];t&&t.addClass("active");a=this.el.find(".tab")[e.group];a&&a.addClass("active");c(e.el,e.el.offsetTop);return this},title:function(){if(i.paused)return;var e=this.current();document.title="Now Playing..."+e["name"]+" - "+e["artist"]+" | "+y},error:function(){var e=this.current();e.el.removeClass("current").addClass("error");e.error=true;this.errnum++}};var o={el:null,data:null,index:0,create:function(e){var t=s.index;var a=this;var n=s.current().lrc;var i=function(n){if(t!==s.index)return;a.data=a.parse(n);var i="";a.data.forEach(function(e,t){i+=""+e[1]+"

      "});a.el=e.createChild("div",{className:"inner",innerHTML:i},"replace");a.index=0};if(n.startsWith("http"))this.fetch(n,i);else i(n)},update:function(e){if(!this.data)return;if(this.index>this.data.length-1||e=this.data[this.index+1][0])){for(var t=0;t=this.data[t][0]&&(!this.data[t+1]||e/g,"").replace(/^\s+|\s+$/g,"");if(r){const o=r.length;for(var c=0;c
      '+'

      '+t.name+"

      "+t.artist+""+'
      ';this.el.child(".cover").addEventListener("click",e.player.options.events["play-pause"]);o.create(this.el.child(".lrc"))}};var p={el:null,bar:null,create:function(){var e=s.current().el;if(e){if(this.el){this.el.parentNode.removeClass("current").removeEventListener(n.nameMap.dragStart,this.drag);this.el.remove()}this.el=e.createChild("div",{className:"progress"});this.el.attr("data-dtime",n.secondToTime(0));this.bar=this.el.createChild("div",{className:"bar"});e.addClass("current");e.addEventListener(n.nameMap.dragStart,this.drag);s.scroll()}},update:function(e){this.bar.width(Math.floor(e*100)+"%");this.el.attr("data-ptime",n.secondToTime(e*i.duration))},seeking:function(e){if(e)this.el.addClass("seeking");else this.el.removeClass("seeking")},percent:function(e,t){var a=((e.clientX||e.changedTouches[0].clientX)-t.left())/t.width();a=Math.max(a,0);return Math.min(a,1)},drag:function(t){t.preventDefault();var a=s.current().el;var r=function(e){e.preventDefault();var t=p.percent(e,a);p.update(t);o.update(t*i.duration)};var c=function(t){t.preventDefault();a.removeEventListener(n.nameMap.dragEnd,c);a.removeEventListener(n.nameMap.dragMove,r);var s=p.percent(t,a);p.update(s);e.player.seek(s*i.duration);i.disableTimeupdate=false;p.seeking(false)};i.disableTimeupdate=true;p.seeking(true);a.addEventListener(n.nameMap.dragMove,r);a.addEventListener(n.nameMap.dragEnd,c)}};var m={el:null,btns:{},step:"next",create:function(){if(!e.player.options.controls)return;var t=this;e.player.options.controls.forEach(function(a){if(t.btns[a])return;var r={onclick:function(n){t.events[a]?t.events[a](n):e.player.options.events[a](n)}};switch(a){case"volume":r.className=" "+(i.muted?"off":"on");r.innerHTML='
      ';r["on"+n.nameMap.dragStart]=t.events["volume"];r.onclick=null;break;case"mode":r.className=" "+e.player.options.mode;break;default:r.className="";break}r.className=a+r.className+" btn";t.btns[a]=t.el.createChild("div",r)});t.btns["volume"].bar=t.btns["volume"].child(".bar")},events:{mode:function(t){switch(e.player.options.mode){case"loop":e.player.options.mode="random";break;case"random":e.player.options.mode="order";break;default:e.player.options.mode="loop"}m.btns["mode"].className="mode "+e.player.options.mode+" btn";d.set("_PlayerMode",e.player.options.mode)},volume:function(t){t.preventDefault();var a=t.currentTarget;var r=false;var s=function(t){t.preventDefault();e.player.volume(m.percent(t,a));r=true};var o=function(t){t.preventDefault();a.removeEventListener(n.nameMap.dragEnd,o);a.removeEventListener(n.nameMap.dragMove,s);if(r){e.player.muted();e.player.volume(m.percent(t,a))}else{if(i.muted){e.player.muted();e.player.volume(i.volume)}else{e.player.muted("muted");m.update(0)}}};a.addEventListener(n.nameMap.dragMove,s);a.addEventListener(n.nameMap.dragEnd,o)},backward:function(t){m.step="prev";e.player.mode()},forward:function(t){m.step="next";e.player.mode()}},update:function(e){m.btns["volume"].className="volume "+(!i.muted&&e>0?"on":"off")+" btn";m.btns["volume"].bar.width(Math.floor(e*100)+"%")},percent:function(e,t){var a=((e.clientX||e.changedTouches[0].clientX)-t.left())/t.width();a=Math.max(a,0);return Math.min(a,1)}};var v={onerror:function(){s.error();e.player.mode()},ondurationchange:function(){if(i.duration!==1){p.el.attr("data-dtime",n.secondToTime(i.duration))}},onloadedmetadata:function(){e.player.seek(0);p.el.attr("data-dtime",n.secondToTime(i.duration))},onplay:function(){e.parentNode.addClass("playing");K(this.attr("title"));f=e},onpause:function(){e.parentNode.removeClass("playing");f=null},ontimeupdate:function(){if(!this.disableTimeupdate){p.update(this.currentTime/this.duration);o.update(this.currentTime)}},onended:function(t){e.player.mode();e.player.play()}};var g={el:{},create:function(){if(!e.player.options.btns)return;var t=this;e.player.options.btns.forEach(function(a){if(t.el[a])return;t.el[a]=e.createChild("div",{className:a+" btn",onclick:function(t){e.player.fetch().then(function(){e.player.options.events[a](t)})}})})}};var b=function(t){if(e.player.created)return;e.player.options=Object.assign(a,t);e.player.options.mode=d.get("_PlayerMode")||e.player.options.mode;g.create();i=e.createChild(e.player.options.type,v);r.create();e.parentNode.addClass(e.player.options.type);e.player.created=true};b(t);return e};var m=e.statics.indexOf("//")>0?e.statics:e.root;var v={x:"undefined",y:"undefined"};var g=0;var y,b;const w=document.getElementsByTagName("body")[0];const C=document.documentElement;const x=u("#container");const k=u("#loading");const L=u("#nav");const E=u("#header");const T=L.child(".toggle");const M=u("#quick");const N=u("#sidebar");const O=u("#brand");var j=u("#tool"),I,H,A,q;var S=u("#search");var R,_,P;var D=window.innerHeight;var W=window.innerWidth;var B=0,Y=window.location.href;var z;const X=lozad("img, [data-background-image]",{loaded:function(e){e.addClass("lozaded")}});const F={timer:null,lock:false,show:function(){clearTimeout(this.timer);document.body.removeClass("loaded");k.attr("style","display:block");F.lock=false},hide:function(t){if(!e.loader.start)t=-1;this.timer=setTimeout(this.vanish,t||3e3)},vanish:function(){if(F.lock)return;if(e.loader.start)l(k,0);document.body.addClass("loaded");F.lock=true}};const U=function(e){var t=u(".theme .ic");if(e=="dark"){C.attr("data-theme",e);t.removeClass("i-sun");t.addClass("i-moon")}else{C.attr("data-theme",null);t.removeClass("i-moon");t.addClass("i-sun")}};const G=function(e){if(C.attr("data-theme")=="dark")e="#222";u('meta[name="theme-color"]').attr("content",e)};const V=function(){window.matchMedia("(prefers-color-scheme: dark)").addListener(function(e){if(e.matches){U("dark")}else{U()}});var t=d.get("theme");if(t){U(t)}else{if(e.darkmode){U("dark")}}u(".theme").addEventListener("click",function(e){var t=e.currentTarget.child(".ic");var a=w.createChild("div",{id:"neko",innerHTML:'
      '});var n=function(){l(a,{delay:2500,opacity:0},function(){w.removeChild(a)})};if(t.hasClass("i-sun")){var i=function(){a.addClass("dark");U("dark");d.set("theme","dark");n()}}else{a.addClass("dark");var i=function(){a.removeClass("dark");U();d.set("theme","light");n()}}l(a,1,function(){setTimeout(i,210)})})};const J=function(){document.addEventListener("visibilitychange",function(){switch(document.visibilityState){case"hidden":u('[rel="icon"]').attr("href",m+e.favicon.hidden);document.title=LOCAL.favicon.hide;if(e.loader.switch)F.show();clearTimeout(b);break;case"visible":u('[rel="icon"]').attr("href",m+e.favicon.normal);document.title=LOCAL.favicon.show;if(e.loader.switch)F.hide(1e3);b=setTimeout(function(){document.title=y},2e3);break}})};const K=function(e){if(!e)return;var t=w.createChild("div",{innerHTML:e,className:"tip"});setTimeout(function(){t.addClass("hide");setTimeout(function(){w.removeChild(t)},300)},3e3)};const $=function(e){R=L.height();_=E.height();P=_+u("#waves").height();if(W!=window.innerWidth)ae(null,1);D=window.innerHeight;W=window.innerWidth;N.child(".panels").height(D+"px")};const Q=function(e){var t=window.innerHeight;var n=a();var i=n>t?n-t:document.body.scrollHeight-t;var r=window.pageYOffset>_;var s=window.pageYOffset>0;if(r){G("#FFF")}else{G("#222")}L.toggleClass("show",r);j.toggleClass("affix",s);O.toggleClass("affix",s);N.toggleClass("affix",window.pageYOffset>P&&document.body.offsetWidth>991);if(typeof v.y=="undefined"){v.y=window.pageYOffset}g=v.y-window.pageYOffset;if(g<0){L.removeClass("up");L.toggleClass("down",r)}else if(g>0){L.removeClass("down");L.toggleClass("up",r)}else{}v.y=window.pageYOffset;var o=Math.round(Math.min(100*window.pageYOffset/i,100))+"%";H.child("span").innerText=o;u(".percent").width(o)};const Z=function(){if(e.auto_scroll)d.set(Y,v.y)};const ee=function(t){var a=window.location.hash;var n=null;if(B){d.del(Y);return}if(a)n=u(decodeURI(a));else{n=e.auto_scroll?parseInt(d.get(Y)):0}if(n){c(n);B=1}if(t&&a&&!B){c(n);B=1}};const te=function(e,t){var a=w.createChild("textarea",{style:{top:window.scrollY+"px",position:"absolute",opacity:"0"},readOnly:true,value:e});const n=document.getSelection();const i=n.rangeCount>0?n.getRangeAt(0):false;a.select();a.setSelectionRange(0,e.length);a.readOnly=false;var r=document.execCommand("copy");t&&t(r);a.blur();if(i){n.removeAllRanges();n.addRange(i)}w.removeChild(a)};const ae=function(e,t){if(N.hasClass("on")){N.removeClass("on");T.removeClass("close");if(t){N.style=""}else{l(N,"slideRightOut")}}else{if(t){N.style=""}else{l(N,"slideRightIn",function(){N.addClass("on");T.addClass("close")})}}};const ne=function(){var e=N.child(".inner");var t=N.find(".panel");if(N.child(".tab")){e.removeChild(N.child(".tab"))}var a=document.createElement("ul"),n="active";a.className="tab";["contents","related","overview"].forEach(function(e){var t=N.child(".panel."+e);if(t.innerHTML.replace(/(^\s*)|(\s*$)/g,"").length<1){if(e=="contents"){q.display("none")}return}if(e=="contents"){q.display("")}var i=document.createElement("li");var r=document.createElement("span");var s=document.createTextNode(t.attr("data-title"));r.appendChild(s);i.appendChild(r);i.addClass(e+" item");if(n){t.addClass(n);i.addClass(n)}else{t.removeClass("active")}i.addEventListener("click",function(e){var t=event.currentTarget;if(t.hasClass("active"))return;N.find(".tab .item").forEach(function(e){e.removeClass("active")});N.find(".panel").forEach(function(e){e.removeClass("active")});N.child(".panel."+t.className.replace(" item","")).addClass("active");t.addClass("active")});a.appendChild(i);n=""});if(a.childNodes.length>1){e.insertBefore(a,e.childNodes[0]);N.child(".panels").style.paddingTop=""}else{N.child(".panels").style.paddingTop=".625rem"}};const ie=function(){var t=u.all(".contents li");if(t.length<1){return}var a=Array.prototype.slice.call(t)||[];var n=null;a=a.map(function(t,a){var i=t.child("a.toc-link");var s=u(decodeURI(i.attr("href")));if(!s)return;var o=s.child("a.anchor");var l=function(e){e.preventDefault();var t=u(decodeURI(e.currentTarget.attr("href")));n=a;c(t,null,function(){r(a);n=null})};i.addEventListener("click",l);o&&o.addEventListener("click",function(t){l(t);te(e.hostname+"/"+LOCAL.path+t.currentTarget.attr("href"))});return s});var i=N.child(".contents.panel");var r=function(e,n){var r=t[e];if(!r)return;if(r.hasClass("current")){return}u.each(".toc .active",function(e){e&&e.removeClass("active current")});a.forEach(function(e){e&&e.removeClass("active")});r.addClass("active current");a[e]&&a[e].addClass("active");var s=r.parentNode;while(!s.matches(".contents")){if(s.matches("li")){s.addClass("active");var o=u(s.child("a.toc-link").attr("href"));if(o){o.addClass("active")}}s=s.parentNode}if(getComputedStyle(N).display!="none"&&i.hasClass("active")){c(i,r.offsetTop-i.offsetHeight/4)}};var s=function(e){var t=0;var n=e[t];if(n.boundingClientRect.top>0){t=a.indexOf(n.target);return t===0?0:t-1}for(;t0){t.target.addClass("show");e.unobserve(t.target)}}})},{root:null,threshold:[.3]});u.each(".index.wrap article.item, .index.wrap section.item",function(t){e.observe(t)});u(".index.wrap .item:first-child").addClass("show")}u.each(".cards .item",function(e,t){["mouseenter","touchstart"].forEach(function(t){e.addEventListener(t,function(t){if(u(".cards .item.active")){u(".cards .item.active").removeClass("active")}e.addClass("active")})});["mouseleave"].forEach(function(t){e.addEventListener(t,function(t){e.removeClass("active")})})})};const de=function(){u.each("span.exturl",function(e){var t=document.createElement("a");t.href=decodeURIComponent(atob(e.dataset.url).split("").map(function(e){return"%"+("00"+e.charCodeAt(0).toString(16)).slice(-2)}).join(""));t.rel="noopener external nofollow noreferrer";t.target="_blank";t.className=e.className;t.title=e.title||e.innerText;t.innerHTML=e.innerHTML;if(e.dataset.backgroundImage){t.dataset.backgroundImage=e.dataset.backgroundImage}e.parentNode.replaceChild(t,e)})};const ue=function(e){if(u(e+" .md img")){s("fancybox");r("fancybox",function(){var t=jQuery.noConflict();u.each(e+" p.gallery",function(e){var t=document.createElement("div");t.className="gallery";t.attr("data-height",e.attr("data-height")||220);t.innerHTML=e.innerHTML.replace(/
      /g,"");e.parentNode.insertBefore(t,e);e.remove()});u.each(e+" .md img:not(.emoji):not(.vemoji)",function(e){var a=t(e);var n,i="image-info";if(!a.is("a img")){var r=a.attr("data-src")||a.attr("src");a.data("safe-src",r);var s=a.wrap('').parent("a");if(!a.is(".gallery img")){s.attr("data-fancybox","default").attr("rel","default")}else{i="jg-caption"}}if(n=e.attr("title")){s.attr("data-caption",n);var o=document.createElement("span");var c=document.createTextNode(n);o.appendChild(c);o.addClass(i);e.insertAfter(o)}});u.each(e+" div.gallery",function(e,a){t(e).justifiedGallery({rowHeight:t(e).data("height")||120,rel:"gallery-"+a}).on("jg.complete",function(){t(this).find("a").each(function(e,t){t.attr("data-fancybox","gallery-"+a)})})});t.fancybox.defaults.hash=false;t(e+" .fancybox").fancybox({loop:true,helpers:{overlay:{locked:false}}})},window.jQuery)}};const fe=function(){pe();if(!u(".md"))return;ue(".post.block");u(".post.block").oncopy=function(e){K(LOCAL.copyright);if(LOCAL.nocopy){e.preventDefault();return}var t=u("#copyright");if(window.getSelection().toString().length>30&&t){e.preventDefault();var a="# "+t.child(".author").innerText;var n="# "+t.child(".link").innerText;var i="# "+t.child(".license").innerText;var r=a+"
      "+n+"
      "+i+"

      "+window.getSelection().toString().replace(/\r\n/g,"
      ");var s=a+"\n"+n+"\n"+i+"\n\n"+window.getSelection().toString().replace(/\r\n/g,"\n");if(e.clipboardData){e.clipboardData.setData("text/html",r);e.clipboardData.setData("text/plain",s)}else if(window.clipboardData){return window.clipboardData.setData("text",s)}}};u.each("li ruby",function(e){var t=e.parentNode;if(e.parentNode.tagName!="LI"){t=e.parentNode.parentNode}t.addClass("ruby")});u.each("ol[start]",function(e){e.style.counterReset="counter "+parseInt(e.attr("start")-1)});u.each(".md table",function(e){e.wrap({className:"table-container"})});u.each(".highlight > .table-container",function(e){e.className="code-container"});u.each("figure.highlight",function(e){var t=e.child(".code-container");var a=e.child("figcaption");e.insertAdjacentHTML("beforeend",'
      ');var n=e.child(".copy-btn");if(LOCAL.nocopy){n.remove()}else{n.addEventListener("click",function(e){var a=e.currentTarget;var n="",i="";t.find("pre").forEach(function(e){i+=n+e.innerText;n="\n"});te(i,function(e){a.child(".ic").className=e?"ic i-check":"ic i-times";a.blur();K(LOCAL.copyright)})});n.addEventListener("mouseleave",function(e){setTimeout(function(){e.target.child(".ic").className="ic i-clipboard"},1e3)})}var i=e.child(".breakline-btn");i.addEventListener("click",function(t){var a=t.currentTarget;if(e.hasClass("breakline")){e.removeClass("breakline");a.child(".ic").className="ic i-align-left"}else{e.addClass("breakline");a.child(".ic").className="ic i-align-justify"}});var r=e.child(".fullscreen-btn");var s=function(){e.removeClass("fullscreen");e.scrollTop=0;w.removeClass("fullscreen");r.child(".ic").className="ic i-expand"};var o=function(t){var a=t.currentTarget;if(e.hasClass("fullscreen")){s();u&&u();c(e)}else{e.addClass("fullscreen");w.addClass("fullscreen");r.child(".ic").className="ic i-compress";d&&d()}};r.addEventListener("click",o);a&&a.addEventListener("click",o);if(t&&t.find("tr").length>15){t.style.maxHeight="300px";t.insertAdjacentHTML("beforeend",'
      ');var l=t.child(".show-btn");var d=function(){t.style.maxHeight="";l.addClass("open")};var u=function(){t.style.maxHeight="300px";l.removeClass("open")};l.addEventListener("click",function(e){if(l.hasClass("open")){s();u();c(t)}else{d()}})}});u.each("pre.mermaid > svg",function(e){e.style.maxWidth=""});u.each(".reward button",function(e){e.addEventListener("click",function(e){e.preventDefault();var t=u("#qr");if(t.display()==="inline-flex"){l(t,0)}else{l(t,1,function(){ t.display("inline-flex")})}})});u.each(".quiz > ul.options li",function(e){e.addEventListener("click",function(t){if(e.hasClass("correct")){e.toggleClass("right");e.parentNode.parentNode.addClass("show")}else{e.toggleClass("wrong")}})});u.each(".quiz > p",function(e){e.addEventListener("click",function(t){e.parentNode.toggleClass("show")})});u.each(".quiz > p:first-child",function(e){var t=e.parentNode;var a="choice";if(t.hasClass("true")||t.hasClass("false"))a="true_false";if(t.hasClass("multi"))a="multiple";if(t.hasClass("fill"))a="gap_fill";if(t.hasClass("essay"))a="essay";e.attr("data-type",LOCAL.quiz[a])});u.each(".quiz .mistake",function(e){e.attr("data-type",LOCAL.quiz.mistake)});u.each("div.tags a",function(e){e.className=["primary","success","info","warning","danger"][Math.floor(Math.random()*5)]});u.each(".md div.player",function(e){p(e,{type:e.attr("data-type"),mode:"order",btns:[]}).player.load(JSON.parse(e.attr("data-src"))).fetch()})};const he=function(){var e;u.each("div.tab",function(t,a){if(t.attr("data-ready"))return;var n=t.attr("data-id");var i=t.attr("data-title");var r=u("#"+n);if(!r){r=document.createElement("div");r.className="tabs";r.id=n;r.innerHTML='
      ';var s=r.child(".show-btn");s.addEventListener("click",function(e){c(r)});t.parentNode.insertBefore(r,t);e=true}else{e=false}var o=r.child(".nav ul");if(!o){o=r.createChild("div",{className:"nav",innerHTML:"
        "}).child("ul")}var l=o.createChild("li",{innerHTML:i});if(e){l.addClass("active");t.addClass("active")}l.addEventListener("click",function(e){var a=e.currentTarget;r.find(".active").forEach(function(e){e.removeClass("active")});t.addClass("active");a.addClass("active")});r.appendChild(t);t.attr("data-ready",true)})};const pe=function(){var e=u("#comments");if(!e){A.display("none");return}else{A.display("")}if(!window.IntersectionObserver){s("valine")}else{var t=new IntersectionObserver(function(e,t){var a=e[0];s("valine");if(a.isIntersecting||a.intersectionRatio>0){l(u("#comments"),"bounceUpIn");t.disconnect()}});t.observe(e)}};const me=function(t){if(e.search===null)return;if(!S){S=w.createChild("div",{id:"search",innerHTML:'
        '})}var a=instantsearch({indexName:e.search.indexName,searchClient:algoliasearch(e.search.appID,e.search.apiKey),searchFunction:function(e){var t=u(".search-input");if(t.value){e.search()}}});a.on("render",function(){t.refresh(u("#search-hits"))});a.addWidgets([instantsearch.widgets.configure({hitsPerPage:e.search.hits.per_page||10}),instantsearch.widgets.searchBox({container:".search-input-container",placeholder:LOCAL.search.placeholder,showReset:false,showSubmit:false,showLoadingIndicator:false,cssClasses:{input:"search-input"}}),instantsearch.widgets.stats({container:"#search-stats",templates:{text:function(e){var t=LOCAL.search.stats.replace(/\$\{hits}/,e.nbHits).replace(/\$\{time}/,e.processingTimeMS);return t+'
        '}}}),instantsearch.widgets.hits({container:"#search-hits",templates:{item:function(t){var a=t.categories?""+t.categories.join('')+"":"";return''+a+t._highlightResult.title.value+""},empty:function(e){return'
        '+LOCAL.search.empty.replace(/\$\{query}/,e.query)+"
        "}},cssClasses:{item:"item"}}),instantsearch.widgets.pagination({container:"#search-pagination",scrollTo:false,showFirst:false,showLast:false,templates:{first:'',last:'',previous:'',next:''},cssClasses:{root:"pagination",item:"pagination-item",link:"page-number",selectedItem:"current",disabledItem:"disabled-item"}})]);a.start();u.each(".search",function(e){e.addEventListener("click",function(){document.body.style.overflow="hidden";l(S,"shrinkIn",function(){u(".search-input").focus()})})});const n=function(){document.body.style.overflow="";l(S,0)};S.addEventListener("click",function(e){if(e.target===S){n()}});u(".close-btn").addEventListener("click",n);window.addEventListener("pjax:success",n);window.addEventListener("keyup",function(e){if(e.key==="Escape"){n()}})};const ve=function(){u.each(".overview .menu > .item",function(e){L.child(".menu").appendChild(e.cloneNode(true))});k.addEventListener("click",F.vanish);T.addEventListener("click",ae);u(".dimmer").addEventListener("click",ae);M.child(".down").addEventListener("click",se);M.child(".up").addEventListener("click",re);if(!j){j=E.createChild("div",{id:"tool",innerHTML:'
        0%
        '})}I=j.child(".player");H=j.child(".back-to-top");A=j.child(".chat");q=j.child(".contents");H.addEventListener("click",re);A.addEventListener("click",oe);q.addEventListener("click",ae);p(I);u("main").addEventListener("click",function(){I.player.mini()})};const ge=function(){Z();if(N.hasClass("on")){l(N,function(){N.removeClass("on");T.removeClass("close")})}u("#main").innerHTML="";u("#main").appendChild(k.lastChild.cloneNode(true));c(0)};const ye=function(t){B=0;Y=window.location.href;s("katex");r("copy_tex");s("mermaid");r("chart");r("valine",function(){var t=Object.assign({},e.valine);t=Object.assign(t,LOCAL.valine||{});t.el="#comments";t.pathname=LOCAL.path;t.pjax=z;t.lazyload=X;new MiniValine(t);setTimeout(function(){ee(1);ue(".v")},1e3)},window.MiniValine);if(!t){u.each("script[data-pjax]",o)}y=document.title;$();ce();ne();ie();de();fe();he();I.player.load(LOCAL.audio||e.audio||{});F.hide();setTimeout(function(){ee()},500);le();X.observe()};const be=function(){ve();z=new Pjax({selectors:["head title",".languages",".pjax","script[data-config]"],analytics:false,cacheBust:false});e.quicklink.ignores=LOCAL.ignores;quicklink.listen(e.quicklink);J();V();me(z);window.addEventListener("scroll",Q);window.addEventListener("resize",$);window.addEventListener("pjax:send",ge);window.addEventListener("pjax:success",ye);window.addEventListener("beforeunload",function(){Z()});ye(1)};window.addEventListener("DOMContentLoaded",be);console.log("%c Theme.Shoka v"+e.version+" %c https://shoka.lostyu.me/ ","color: white; background: #e9546b; padding:5px 0;","padding:4px;border:1px solid #e9546b;");var we=document.createElement("canvas");we.style.cssText="position:fixed;top:0;left:0;pointer-events:none;z-index:9999999";document.body.appendChild(we);var Ce=we.getContext("2d");var xe=30;var ke=0;var Le=0;var Ee="click";var Te=e.fireworks;function Me(){we.width=window.innerWidth*2;we.height=window.innerHeight*2;we.style.width=window.innerWidth+"px";we.style.height=window.innerHeight+"px";we.getContext("2d").scale(2,2)}function Ne(e){ke=e.clientX||e.touches&&e.touches[0].clientX;Le=e.clientY||e.touches&&e.touches[0].clientY}function Oe(e){var t=anime.random(0,360)*Math.PI/180;var a=anime.random(50,180);var n=[-1,1][anime.random(0,1)]*a;return{x:e.x+n*Math.cos(t),y:e.y+n*Math.sin(t)}}function je(e,t){var a={};a.x=e;a.y=t;a.color=Te[anime.random(0,Te.length-1)];a.radius=anime.random(16,32);a.endPos=Oe(a);a.draw=function(){Ce.beginPath();Ce.arc(a.x,a.y,a.radius,0,2*Math.PI,true);Ce.fillStyle=a.color;Ce.fill()};return a}function Ie(e,t){var a={};a.x=e;a.y=t;a.color="#FFF";a.radius=.1;a.alpha=.5;a.lineWidth=6;a.draw=function(){Ce.globalAlpha=a.alpha;Ce.beginPath();Ce.arc(a.x,a.y,a.radius,0,2*Math.PI,true);Ce.lineWidth=a.lineWidth;Ce.strokeStyle=a.color;Ce.stroke();Ce.globalAlpha=1};return a}function He(e){for(var t=0;t格攻击 - 密码学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        SageMath 🔧 一个非常好用的工具

        # 扩展维纳攻击

        私钥过小时,能够对 NN 进行分解。维纳证明了:

        d<13N14(q<p<2q)d\lt\frac{1}{3}N^{1\over 4}\\ (q\lt p\lt 2q)

        的时候,一定能够对 NN 进行分解。

        # 格规约

        # LLL

        Lenstra–Lenstra–Lovász

        基于格的约简算法,它将输入矩阵转换为其最简形式

        使用 LLL 或类似的格规约算法找到格中近似最短的非零向量

        # BKZ

        Block Korkine-Zolotarev

        是 LLL 算法的一种改进版本,能够更有效地处理高维格。BKZ 算法的基本思想是通过逐步降低格的基向量长度,将原始格转化为一个更紧密的格。这样做可以使格的基向量更加正交,并且可能使格的基向量更接近标准正交基。BKZ 算法通过多次应用基向量间的 Gram-Schmidt 正交化过程和基向量的坐标变换来实现这一点。

        一般用 BKZ 效果会比 LLL 好,其中的 block_size 设置越大效果越好,但算得越慢,最大取值是格的维度。

        # 格攻击

        格攻击其实是一个启发式算法

        # 基本域

        需要用到它的容积

        设矩阵 B=[b0,b1,,bn1]B=[b_0,b_1,\cdots,b_{n-1}] 为一组格基,则这组格基的基本域为:

        F(b0,,bn1):={t0b0++tn1bn10ti<1}\mathscr{F}(b_0,\cdots,b_{n-1}):=\{t_0b_0+\cdots+t_{n-1}b_{n-1}|0≤t_i<1\}

        基本域的容积等于格基的行列式(的绝对值):

        Vol(F)=det(L(B))\text{Vol}(\mathscr{F})=\det(\mathscr{L}(B))

        # 超球体

        高维空间的球体,也需要用到它的容积

        BR(a)\mathbb{B}_R(a) 为一个以点 aa 为圆心,半径为 RRnn 维球体,那么 BR(a)\mathbb{B}_R(a) 的容积(或者说体积)为:

        Vol(BR(a))=πn2RnΓ(1+n2)\text{Vol}(\mathbb{B}_R(a))={\pi^{\frac{n}{2}}R^n\over\Gamma(1+\frac{n}{2})}

        长度短于 σ(L)\sigma(L) 的格向量(约)只有一个,所以如果我们通过计算得到一个向量长度(约)小于 σ(L)\sigma(L) ,那么大概率可以确定它是一个最短向量,在维度不高的情况下,可以用 LLL 把他求出来:

        σ(L)=n2πe2(detL)1n\sigma(L)=\sqrt[2]{n\over2\pi e}(\det L)^{1\over n}

        SageMath 🔧 一个非常好用的工具

        # 扩展维纳攻击

        私钥过小时,能够对 NN 进行分解。维纳证明了:

        d<13N14(q<p<2q)d\lt\frac{1}{3}N^{1\over 4}\\ (q\lt p\lt 2q)

        的时候,一定能够对 NN 进行分解。

        # 格规约

        # LLL

        Lenstra–Lenstra–Lovász

        基于格的约简算法,它将输入矩阵转换为其最简形式

        使用 LLL 或类似的格规约算法找到格中近似最短的非零向量

        # BKZ

        Block Korkine-Zolotarev

        是 LLL 算法的一种改进版本,能够更有效地处理高维格。BKZ 算法的基本思想是通过逐步降低格的基向量长度,将原始格转化为一个更紧密的格。这样做可以使格的基向量更加正交,并且可能使格的基向量更接近标准正交基。BKZ 算法通过多次应用基向量间的 Gram-Schmidt 正交化过程和基向量的坐标变换来实现这一点。

        一般用 BKZ 效果会比 LLL 好,其中的 block_size 设置越大效果越好,但算得越慢,最大取值是格的维度。

        # 格攻击

        格攻击其实是一个启发式算法

        # 基本域

        需要用到它的容积

        设矩阵 B=[b0,b1,,bn1]B=[b_0,b_1,\cdots,b_{n-1}] 为一组格基,则这组格基的基本域为:

        F(b0,,bn1):={t0b0++tn1bn10ti<1}\mathscr{F}(b_0,\cdots,b_{n-1}):=\{t_0b_0+\cdots+t_{n-1}b_{n-1}|0≤t_i<1\}

        基本域的容积等于格基的行列式(的绝对值):

        Vol(F)=det(L(B))\text{Vol}(\mathscr{F})=\det(\mathscr{L}(B))

        # 超球体

        高维空间的球体,也需要用到它的容积

        BR(a)\mathbb{B}_R(a) 为一个以点 aa 为圆心,半径为 RRnn 维球体,那么 BR(a)\mathbb{B}_R(a) 的容积(或者说体积)为:

        Vol(BR(a))=πn2RnΓ(1+n2)\text{Vol}(\mathbb{B}_R(a))={\pi^{\frac{n}{2}}R^n\over\Gamma(1+\frac{n}{2})}

        长度短于 σ(L)\sigma(L) 的格向量(约)只有一个,所以如果我们通过计算得到一个向量长度(约)小于 σ(L)\sigma(L) ,那么大概率可以确定它是一个最短向量,在维度不高的情况下,可以用 LLL 把他求出来:

        σ(L)=n2πe2(detL)1n\sigma(L)=\sqrt[2]{n\over2\pi e}(\det L)^{1\over n}的参数上

        \ No newline at end of file +c-16-25.333-24-45-24-59z"/>的参数上

        \ No newline at end of file diff --git a/math/discrete-math/algebra/coset-of-subgroup/index.html b/math/discrete-math/algebra/coset-of-subgroup/index.html index 167dda6d..094cb7d1 100644 --- a/math/discrete-math/algebra/coset-of-subgroup/index.html +++ b/math/discrete-math/algebra/coset-of-subgroup/index.html @@ -1 +1 @@ -子群的陪集 - 代数结构 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 群子集的乘积运算

        # 定义

        AABB 是群 GG 的两个非空子集,称集合:

        AB={abaA,bB}AB = \{ab|a\in A, b\in B \}

        为群的子集 AABB乘积 (product)

        如果 gGg\in G 为群 GG 的一个元素,A={g}A=\{g\} ,则:

        • ABAB 简记为 gB={gbbB}gB = \{gb|b\in B\}
        • BABA 简记为 Bg={bgbB}Bg = \{bg|b\in B\}

        注意,即使有 AB=BAAB=BA,也不意味着 aA,bB,ab=ba\forall a\in A, b\in B,\ \ \ ab =ba,正确的结论应该是:aA,bB,aA,bB,ab=ba\forall a\in A, b\in B,\ \ \ \exist a'\in A, b'\in B,\ \ \ ab = b'a'

        # 交换群的交换律

        GG交换群时,对 GG 的任意子集 A,BA,B ,满足交换律,有 AB=BAAB=BA ;但对于非交换群,一般没有 AB=BAAB=BA ,也就是没有交换律成立。

        # 满足结合律

        群的子集运算满足结合律:A(BC)=(AB)CA(BC) = (AB)C

        # 对单元素满足消去律

        gA=gBgA=gBAg=BgAg=Bg,则 A=BA=B

        但对普遍的

        AC=BC⇏A=BAC=BC\not\rArr A=B


        证明(不妨考虑 gA=gBgA=gB

        aA\forall a\in A,有 gagAga\in gA

        因为 gA=gBgA=gB ,所以 gagBga\in gB ,所以:

        bB,ga=gb\exist b\in B,\ \ \ ga=gb

        由群满足消去律可得:b=ab=a

        所以 a=bBa=b\in B ,所以有 ABA\sube B ;同理可证 BAB\sube A,从而可得 A=BA=B

        # 幂运算

        HH 是群 GG 的子群,则 HH=HHH=H

        # 乘积的子群充要条件

        如果 A,BA,B 是群 GG 的两个子群,则 ABAB 也是群 GG 的子群当且仅当 AB=BAAB=BA


        必要性: ABGAB=BAAB\leq G\rArr AB=BA

        对于 a,b,abAB\forall a,b,\ \ ab\in AB,只需证 abBAab\in BA

        由于 ABGAB\leq G ,所以 abAB,(ab)1=ab\exist a'b'\in AB,\ \ \ (ab)^{-1}=a'b'

        ab=((ab)1)1=(ab)1=b1a1BAab=((ab)^{-1})^{-1}=(a'b')^{-1}=b'^{-1}a'^{-1}\in BA

        从而可得 ABBAAB\sube BA ;同理可证 BAABBA\sube AB,从而可得 AB=BAAB=BA


        充分性: AB=BAABGAB=BA\rArr AB\leq G

        对于 a1,b1,a2,b2,a1b1,a2b2AB\forall a_1,b_1,a_2,b_2,\ \ \ a_1b_1,a_2b_2\in AB,有:a1b1(a2b2)1ABa_1b_1(a_2b_2)^{-1}\in AB

        a1b1(a2b2)1=a1b1b21a21=a1(b1b21)a21ABA=A(BA)=A(AB)=(AA)B=ABa_1b_1(a_2b_2)^{-1}=a_1b_1b_2^{-1}a_2^{-1}=a_1(b_1b_2^{-1})a_2^{-1}\\ \in ABA=A(BA)=A(AB)=(AA)B=AB

        从而由子群判定定理二可知 ABABGG 的子群

        # 乘积的子集关系

        A,B,CA,B,C 是群 GG 的非空子集(不一定是群),若 BCB⊆C ,则有 ABACAB⊆ACBABCBA⊆BC


        证明

        对任意 xABx\in AB ,则存在 aA,bBa\in A,b\in B,使得 x=abx=ab 。由 bBb\in B ,且 BCB⊆C,因此 bCb\in C,因此 x=abACx=ab\in AC ,因此 ABBCAB⊆BC

        对任意 xBAx\in BA ,则存在 aA,bBa\in A,b\in B ,使得 x=bax=ba 。由 bBb\in B ,且 BCB⊆C ,因此 bCb\in C ,因此 x=abACx=ab\in AC ,因此 BACABA⊆CA

        # 子群的陪集及其性质

        # 陪集定义

        GG 是群,HHGG 的子群

        对任意 aGa\in G,群 GG子集 aH={ahhH}aH=\{ah|h\in H\}Ha={hahH}Ha=\{ha|h\in H\} ,分别称为 HHGG 中的左陪集 (left coset) 和右陪集 (right coset)

        子群 HH 的所有右陪集构成的集合族 G\HG\backslash HGG 的一个划分

        子群 HH 的所有左陪集构成的集合族 G/HG/H 也是 GG 的一个划分

        注意陪集不一定是群了

        # 陪集举例

        对于群 (U(5),5)(U(5),⊗_5),子群 H={1,4}H=\{1,4\}

        5⊗_51234
        11234
        22413
        33142
        44321

        1H={151,154}={1,4}2H={251,254}=2,33H={351,354}={3,2}4H={451,454}={4,1}1H=\{1⊗_5 1,1⊗_5 4\}=\{1,4\}\\ 2H=\{2⊗_5 1,2⊗_5 4\}={2,3}\\ 3H=\{3⊗_5 1,3⊗_5 4\}=\{3,2\}\\ 4H=\{4⊗_5 1,4⊗_5 4\}=\{4,1\}

        HH 有两个不同的陪集 {1,4}\{1, 4\}{2,3}\{2,3\}


        对于下面的置换群 (G,)(G,\circ) ,子群 H={f1,f2}H=\{f_1, f_2\}

        \circf1f_1f2f_2f3f_3f4f_4f5f_5f6f_6
        f1f_1f1f_1f2f_2f3f_3f4f_4f5f_5f6f_6
        f2f_2f2f_2f1f_1f6f_6f5f_5f4f_4f3f_3
        f3f_3f3f_3f5f_5f1f_1f6f_6f2f_2f4f_4
        f4f_4f4f_4f6f_6f5f_5f1f_1f3f_3f2f_2
        f5f_5f5f_5f3f_3f4f_4f2f_2f6f_6f1f_1
        f6f_6f6f_6f4f_4f2f_2f3f_3f1f_1f5f_5

        f1H={f1f1,f1f2}={f1,f2}f2H={f2f1,f2f2}={f2,f1}f3H={f3f1,f3f2}={f3,f5}f4H={f4f1,f4f2}={f4,f6}f5H={f5f1,f5f2}={f5,f3}f6H={f6f1,f6f2}={f6,f4}Hf3={f1f3,f2f3}={f3,f6}Hf4={f1f4,f2f4}={f4,f5}Hf5={f1f5,f2f5}={f5,f4}Hf6={f1f6,f2f6}={f6,f3}f_1 H=\{f_1\circ f_1, f_1\circ f_2\}=\{f_1, f_2\}\\ f_2 H=\{f_2\circ f_1, f_2\circ f_2\}=\{f_2, f_1\}\\ f_3 H=\{f_3\circ f_1, f_3\circ f_2\}=\{f_3, f_5\}\\ f_4 H=\{f_4\circ f_1, f_4\circ f_2\}=\{f_4, f_6\}\\ f_5 H=\{f_5\circ f_1, f_5\circ f_2\}=\{f_5, f_3\}\\ f_6 H=\{f_6\circ f_1, f_6\circ f_2\}=\{f_6, f_4\}\\ Hf_3=\{f_1\circ f_3, f_2\circ f_3\}=\{f_3, f_6\}\\ Hf_4=\{f_1\circ f_4, f_2\circ f_4\}=\{f_4, f_5\}\\ Hf_5=\{f_1\circ f_5, f_2\circ f_5\}=\{f_5, f_4\}\\ Hf_6=\{f_1\circ f_6, f_2\circ f_6\}=\{f_6, f_3\}

        HH 有三个不同的陪集,且左右陪集不相等,例如 f3HHf3f_3H≠Hf_3

        # 不同元素陪集相同充要条件

        a,bG\forall a,b\in G

        • Ha=HbHa=Hb 当且仅当 aHba\in Hb 当且仅当 ab1Hab^{-1}\in H
        • aH=bHaH=bH 当且仅当 abHa\in bH 当且仅当 b1aHb^{-1}a\in H

        # 陪集不变的充要条件

        HH 是群 GG 的子群,对任意的 aGa\in G

        • HaHHa=H 的充分必要条件是 aHa\in H

        引理

        eHe\in H ,所以 aeaHaa=ea\in Ha


        证明

        必要性:若 HaHHa=H ,则因为 aHaa\in Ha ,从而 aHa\in H

        充分性:反之若 aHa\in H ,则对任意 haHaha\in Ha ,由 HH 对群运算封闭,也有 haHha\in H ,从而有 HaHHa\subseteq H

        而这时对任意 hHh\in H ,由 aHa\in Ha1Ha^{-1}\in H ,从而 ha1Hha^{-1}\in H ,从而有 hha1aHah=ha^{-1}a\in Ha ,即有 HHaH\subseteq Ha

        于是又 HaH,HHaHa\subseteq H,H\subseteq Ha 得到 Ha=HHa=H

        # 陪集的子群充要条件

        HH 是群 GG 的子群,对任意的 aGa\in G

        • 右陪集 HaHaGG 的子群当且仅当 Ha=HHa=H 当且仅当 aHa\in H
        • 左陪集 aHaHGG 的子群当且仅当 aH=HaH=H 当且仅当 aHa\in H

        证明以右陪集为例:

        必要性:若 HaGHa\leq G ,所以 eHae\in Ha ,从而存在 hHh\in H 使得 ehae=ha ,从而 ah1e=h1Ha=h^{-1}e=h^{-1}\in H

        充分性:若 aHa\in H ,则 HaHHa=H 是子群。

        # 子群陪集运算性质

        HH 是群 GG 的子群,则 a,bG\forall a,b\in G 有:

        aHbab1HHa=Hba\in Hb\Longleftrightarrow ab^{-1}\in H\Longleftrightarrow Ha=Hb


        循环论证:


        (1)aHbab1Ha\in Hb\Longrightarrow ab^{-1}\in H

        aHba\in Hb ,即存在 hHh\in H 使得 a=hba=hb 。从而 ab1hHab^{-1}=h\in H


        (2)ab1HHa=Hbab^{-1}\in H\Longrightarrow Ha=Hb

        对于任意的 haHaha\in Ha ,这里 hHh\in H ,因此 hab1Hhab^{-1}\in H,从而:

        ha=hab1bHbha=hab^{-1}b\in Hb

        这表明 HaHbHa\subseteq Hb 。类似地,对任意的 hbHahb\in Ha ,这里 hHh\in H 。由于 ab1Hab^{-1}\in H ,则也有:

        ba1=(b1)1a1=(ab1)1Hba^{-1}=(b^{-1})^{-1}a^{-1}=(ab^{-1})^{-1}\in H

        因此 ha=hba1aHaha=hba^{-1}a\in Ha ,这表明 HbHaHb\subseteq Ha 。综合起来就有 Ha=BbHa=Bb


        (3)Ha=HbaHbHa=Hb\Longrightarrow a\in Hb

        Ha=HbHa=Hb ,则由 aHaa\in HaHH 是子群)可得 aHba\in Hb

        # 子群陪集构成等价关系

        HH 是群 GG 的子群,在 GG 上定义二元关系 RG×GR\in G\times G

        a,bG,aRbab1H\forall a,b\in G,\ \ \ a\text{R}b\Longleftrightarrow ab^{-1}\in H

        RRGG 上的等价关系,且 [a]R=Ha[a]_R=Ha


        证明

        上面已经证明了 ab1HHa=Hbab^{-1}\in H\Longleftrightarrow Ha=Hb,因此 aRbHa=Hba\text{R}b\Longleftrightarrow Ha= Hb

        从而 RR 确实是自反、对称和传递的,即 RR 是等价关系。

        对任意 bGb\in G ,我们有:

        bHaHb=HabRab[a]Rb\in Ha\Longleftrightarrow Hb=Ha\Longleftrightarrow b\text{R}a\Longleftrightarrow b\in[a]_R

        这就表明 Ha=[a]RHa=[a]_R

        # 子群陪集构成划分

        HGH≤G,则:

        • (i)a,bG\forall a,b\in G ,
          • Ha=HbHa=Hb
          • HaHb=Ha\cap Hb=\varnothing
        • (ii)U{HaaG}=G\mathbb U\{Ha|a\in G\}=G

        证明 —— 由等价类的性质立即可得。

        # 正规子群

        若对任意元素 aGa\in G 都有 aH=HaaH=Ha,即左右陪集相等,则称 HHGG 的正规子群,

        # 左右陪集关系

        一个子群的左陪集的所有元素的逆元素组成这个子群的一个右陪集

        符号化描述:

        HGH\leq GaGa\in G

        S={x1xaH}S=\{x^{-1}|x\in{aH}\}

        # 拉格朗日定理

        # 引理

        HHGG 的子群,对任意 aGa\in G ,有 H=aH=Ha|H|=|aH|=|Ha|,即这三个集合等势

        集合等势:若两个集合前存在双函数,即称它们等势。对于有限集,就是这两个集合有相同元素个数


        证明

        定义 φ:HHa\varphi:H\to Ha 为:

        hH,φ(h)=haHa\forall h\in H,\ \ \ \varphi(h)=ha\in Ha

        由群的消去律我们有:h1,h2H\forall h_1,h_2\in H

        φ(h1)=φ(h2)h1a=h2ah1=h2\varphi(h_1)=\varphi(h_2)\Longrightarrow h_1a=h_2a\Longrightarrow h_1=h_2

        从而 φ\varphi 是双函数,即 Ha=H|Ha|=|H|

        类似定义 φ:Ha\varphi':H\to a

        hH,φ(h)=ahaH\forall h\in H,\ \ \ \varphi'(h)=ah\in aH

        不难证明 aH=H|aH|=|H|

        # 指标

        HH 是群 GG 的子群。称子群 HH 在群 GG 中的左陪集或右陪集的个数(有限或无限)为 HHGG 中的指标 (index),记为 [G:H][G:H]

        也称 [G:H][G:H]GG 关于 HH 的陪集个数。

        # 拉格朗日定理

        HH 是有限群 GG 的子群,则:

        G=H[G:H]|G|=|H|\cdot[G:H]


        证明

        [G:H]=r[G:H]=r ,则子群 HH​ 有 rr 个不同的右陪集,设为 Ha1,Ha2,,HarH_{a_1},H_{a_2},\cdots,H_{a_r} 。由于 G=U{HaaG}G=U\{Ha|a\in G\} ,且对任意的 a,bGa,b\in GHa=HbHa=Hb 或者 HaHb=Ha\cap Hb=\varnothing ,因此:

        G=Ha1Ha2HarG=Ha1+Ha2++HarG=H_{a_1}\cup H_{a_2}\cup \cdots\cup H_{a_r}\\ |G|=|H_{a_1}|+|H_{a_2}|+\cdots+|H_{a_r}|

        而对任意的 aGa\in GHa=H|Ha|=|H| ,因此 G=rH=H[G:E]|G|=r|H|=|H|\cdot[G:E]

        # 推论 1 (元素阶与群阶的关系)

        GGnn 阶有限群,单位元是 ee ,则对 GG 的任意元素 aaaa 的阶 a|a|nn 的因子,且 an=ea^{n}=e


        证明

        对任意的 aGa\in G<a><a>GG 的子群,因此由拉格朗日定理 <a>|<a>|nn 的因子;

        但另一方面,若 a=r|a|=r ,则因 <a><a>aa 生成的子群,于是

        <a>={a0=e,a1,,ar1}<a>=\{a^{0}=e,a^{1},\cdots,a^{r-1}\}

        <a>=a=r|<a>|=|a|=r ,从而 a|a| 是群 G 的阶 n 的因子,进而由 == 定理 1.6?== 有 an=ea^{n}=e ,因为对任意的 mZm\in \mathbb{Z} , am=ea^{m}=e 当且仅当 rmr|m

        元素的阶是群的阶的因子;

        但并不是群的阶的每个因子,群都存在元素的阶恰好等于这个因子!

        # 推论 2 (费马小定理)

        pp 是素数,aa 是与 pp 互素的整数,则有:

        ap11(modp)a^{p-1}\equiv1(\bmod p)


        证明

        因为在群 U(p)U(p) ,即群 Zp\mathbb{Z}_p^* 中,由于 aapp 互素,aa(准确地说,aa 整除 pp 的余数)属于 Zp\mathbb{Z}_p^*

        Zp\mathbb{Z}_p^* 的阶是 p1p-1 ,因此 ap1a^{p-1} 等于单位元 1 ,对于 U(p)U(p) 的模 pp 乘运算而言,就是 ap11(modp)a^{p-1}\equiv 1(\bmod p)

        # 推论 3

        H,KH,K 都是 GG 的有限子群,证明:

        • (1) 记 S={hKhH}S=\{hK|h\in H\},则 SSHH划分,从而 HK=SK|HK|=|S|\cdot|K|
        • (2)HK=HKHK|HK|=\frac{|H|\cdot|K|}{|H\cap K|}

        (1)证明

        即证 h1Kh_1Kh2Kh_2K 要么相等,要么不相交

        h1K=h2Kh11h2Kh1Kh2K=h_1K=h_2K\lrArr h_1^{-1}h_2\in K\\ h_1K\cap h_2K=\varnothing


        (2)证明

        为方便起见,记 M=HKM=H\cap K ,由于 HHKK 都是 GG 的子群,因此 MM 也是 GG 的子群且也是 HH 的子群。

        定义函数 φ:H/MS\varphi:H/M\to S,对任意 hHh\in Hφ(hM)=hK\varphi(hM)=hK

        下面证明 φ\varphi 的定义是合适的,且是双函数,从而结合拉格朗日定理有 S=H/M=H/M|S|=|H/M|=|H|/|M| ,从而得到 HK=HKM|HK|=\frac{|H|\cdot|K|}{|M|}

        (a) 证明 φ\varphi 的定义是合适的:

        对任意 h1,h2Hh_1,h_2\in H ,若 h1M=h2Mh_1M=h_2M ,则 h11h2MKh_1^{-1}h_2\in M\subseteq K ,从而 φ(h1M)=φ(h2M)\varphi(h_1M)=\varphi(h_2M),这说明 φ(hM)\varphi(hM) 的值不会因为选取的代表 hh 不同面不同,即 φ\varphi 的定义是合适的。

        (b) 证明 φ\varphi 是单函数:

        对任意 h1,h2Hh_1,h_2\in H,若 φ(h1M)=φ(h2M)\varphi(h_1M)=\varphi(h_2M),即 h1K=h2Kh_1K=h_2K ,从而 h11h2Kh_1^{-1}h_2\in K

        h1,h2Hh_1,h_2\in H ,因此也有 h11h2Hh_1^{-1}h_2\in H

        从而 h11h2HK=Mh_1^{-1}h_2\in H\cap K=M

        从而 h1M=h2Mh_1M=h_2M ,这就表明 φ\varphi 是单函数。

        (c) 显然 φ\varphi 是满函数,对任意 hKShK\in S ,都有 φ(hM)=hk\varphi(hM)=hk

        因此 S=H/M|S|=|H/M| ,由拉格朗日定理有 S=H/M=H/M|S|=|H/M|=|H|/|M| ,最后得到:

        HK=HMK=HKHK|HK|=\frac{|H|}{|M|}|K|=\frac{|H|\cdot|K|}{|H\cap K|}

        # 举例

        不容易找到两个子群的交不等于 {e}\{e\} 的例子,利用计算机程序可找到如下的例子:U(33)={1,2,4,5,7,8,10,13,14,16,17,19,20,23,25,26,28,29,31,32}U(33)=\{1,2,4,5,7,8,10,13,14,16,17,19,20,23,25,26,28,29,31,32\}

        • 子群 {1,2,4,8,16,17,25,29,31,32}\{1,2,4,8,16,17,25,29,31,32\} 是循环子群,生成元为:2,8,17,292,8,17,29
        • 子群 {1,4,5,14,16,20,23,25,26,31}\{1,4,5,14,16,20,23,25,26,31\} 是循环子群,生成元为:5,14,20,265,14,20,26

        上述两个子群交集为 {1,4,16,25,31}\{1,4,16,25,31\}

        # 西罗定理

        可了解,不考

        GGnn 阶循环群,aa 是生成元。对于 nn 的每个正因子 kk , 有且仅有一个 kk 阶循环子群。并且由 anka^{\frac{n}{k}} 生成.

        \ No newline at end of file +子群的陪集 - 代数结构 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 群子集的乘积运算

        # 定义

        AABB 是群 GG 的两个非空子集,称集合:

        AB={abaA,bB}AB = \{ab|a\in A, b\in B \}

        为群的子集 AABB乘积 (product)

        如果 gGg\in G 为群 GG 的一个元素,A={g}A=\{g\} ,则:

        • ABAB 简记为 gB={gbbB}gB = \{gb|b\in B\}
        • BABA 简记为 Bg={bgbB}Bg = \{bg|b\in B\}

        注意,即使有 AB=BAAB=BA,也不意味着 aA,bB,ab=ba\forall a\in A, b\in B,\ \ \ ab =ba,正确的结论应该是:aA,bB,aA,bB,ab=ba\forall a\in A, b\in B,\ \ \ \exist a'\in A, b'\in B,\ \ \ ab = b'a'

        # 交换群的交换律

        GG交换群时,对 GG 的任意子集 A,BA,B ,满足交换律,有 AB=BAAB=BA ;但对于非交换群,一般没有 AB=BAAB=BA ,也就是没有交换律成立。

        # 满足结合律

        群的子集运算满足结合律:A(BC)=(AB)CA(BC) = (AB)C

        # 对单元素满足消去律

        gA=gBgA=gBAg=BgAg=Bg,则 A=BA=B

        但对普遍的

        AC=BC⇏A=BAC=BC\not\rArr A=B


        证明(不妨考虑 gA=gBgA=gB

        aA\forall a\in A,有 gagAga\in gA

        因为 gA=gBgA=gB ,所以 gagBga\in gB ,所以:

        bB,ga=gb\exist b\in B,\ \ \ ga=gb

        由群满足消去律可得:b=ab=a

        所以 a=bBa=b\in B ,所以有 ABA\sube B ;同理可证 BAB\sube A,从而可得 A=BA=B

        # 幂运算

        HH 是群 GG 的子群,则 HH=HHH=H

        # 乘积的子群充要条件

        如果 A,BA,B 是群 GG 的两个子群,则 ABAB 也是群 GG 的子群当且仅当 AB=BAAB=BA


        必要性: ABGAB=BAAB\leq G\rArr AB=BA

        对于 a,b,abAB\forall a,b,\ \ ab\in AB,只需证 abBAab\in BA

        由于 ABGAB\leq G ,所以 abAB,(ab)1=ab\exist a'b'\in AB,\ \ \ (ab)^{-1}=a'b'

        ab=((ab)1)1=(ab)1=b1a1BAab=((ab)^{-1})^{-1}=(a'b')^{-1}=b'^{-1}a'^{-1}\in BA

        从而可得 ABBAAB\sube BA ;同理可证 BAABBA\sube AB,从而可得 AB=BAAB=BA


        充分性: AB=BAABGAB=BA\rArr AB\leq G

        对于 a1,b1,a2,b2,a1b1,a2b2AB\forall a_1,b_1,a_2,b_2,\ \ \ a_1b_1,a_2b_2\in AB,有:a1b1(a2b2)1ABa_1b_1(a_2b_2)^{-1}\in AB

        a1b1(a2b2)1=a1b1b21a21=a1(b1b21)a21ABA=A(BA)=A(AB)=(AA)B=ABa_1b_1(a_2b_2)^{-1}=a_1b_1b_2^{-1}a_2^{-1}=a_1(b_1b_2^{-1})a_2^{-1}\\ \in ABA=A(BA)=A(AB)=(AA)B=AB

        从而由子群判定定理二可知 ABABGG 的子群

        # 乘积的子集关系

        A,B,CA,B,C 是群 GG 的非空子集(不一定是群),若 BCB⊆C ,则有 ABACAB⊆ACBABCBA⊆BC


        证明

        对任意 xABx\in AB ,则存在 aA,bBa\in A,b\in B,使得 x=abx=ab 。由 bBb\in B ,且 BCB⊆C,因此 bCb\in C,因此 x=abACx=ab\in AC ,因此 ABBCAB⊆BC

        对任意 xBAx\in BA ,则存在 aA,bBa\in A,b\in B ,使得 x=bax=ba 。由 bBb\in B ,且 BCB⊆C ,因此 bCb\in C ,因此 x=abACx=ab\in AC ,因此 BACABA⊆CA

        # 子群的陪集及其性质

        # 陪集定义

        GG 是群,HHGG 的子群

        对任意 aGa\in G,群 GG子集 aH={ahhH}aH=\{ah|h\in H\}Ha={hahH}Ha=\{ha|h\in H\} ,分别称为 HHGG 中的左陪集 (left coset) 和右陪集 (right coset)

        子群 HH 的所有右陪集构成的集合族 G\HG\backslash HGG 的一个划分

        子群 HH 的所有左陪集构成的集合族 G/HG/H 也是 GG 的一个划分

        注意陪集不一定是群了

        # 陪集举例

        对于群 (U(5),5)(U(5),⊗_5),子群 H={1,4}H=\{1,4\}

        5⊗_51234
        11234
        22413
        33142
        44321

        1H={151,154}={1,4}2H={251,254}=2,33H={351,354}={3,2}4H={451,454}={4,1}1H=\{1⊗_5 1,1⊗_5 4\}=\{1,4\}\\ 2H=\{2⊗_5 1,2⊗_5 4\}={2,3}\\ 3H=\{3⊗_5 1,3⊗_5 4\}=\{3,2\}\\ 4H=\{4⊗_5 1,4⊗_5 4\}=\{4,1\}

        HH 有两个不同的陪集 {1,4}\{1, 4\}{2,3}\{2,3\}


        对于下面的置换群 (G,)(G,\circ) ,子群 H={f1,f2}H=\{f_1, f_2\}

        \circf1f_1f2f_2f3f_3f4f_4f5f_5f6f_6
        f1f_1f1f_1f2f_2f3f_3f4f_4f5f_5f6f_6
        f2f_2f2f_2f1f_1f6f_6f5f_5f4f_4f3f_3
        f3f_3f3f_3f5f_5f1f_1f6f_6f2f_2f4f_4
        f4f_4f4f_4f6f_6f5f_5f1f_1f3f_3f2f_2
        f5f_5f5f_5f3f_3f4f_4f2f_2f6f_6f1f_1
        f6f_6f6f_6f4f_4f2f_2f3f_3f1f_1f5f_5

        f1H={f1f1,f1f2}={f1,f2}f2H={f2f1,f2f2}={f2,f1}f3H={f3f1,f3f2}={f3,f5}f4H={f4f1,f4f2}={f4,f6}f5H={f5f1,f5f2}={f5,f3}f6H={f6f1,f6f2}={f6,f4}Hf3={f1f3,f2f3}={f3,f6}Hf4={f1f4,f2f4}={f4,f5}Hf5={f1f5,f2f5}={f5,f4}Hf6={f1f6,f2f6}={f6,f3}f_1 H=\{f_1\circ f_1, f_1\circ f_2\}=\{f_1, f_2\}\\ f_2 H=\{f_2\circ f_1, f_2\circ f_2\}=\{f_2, f_1\}\\ f_3 H=\{f_3\circ f_1, f_3\circ f_2\}=\{f_3, f_5\}\\ f_4 H=\{f_4\circ f_1, f_4\circ f_2\}=\{f_4, f_6\}\\ f_5 H=\{f_5\circ f_1, f_5\circ f_2\}=\{f_5, f_3\}\\ f_6 H=\{f_6\circ f_1, f_6\circ f_2\}=\{f_6, f_4\}\\ Hf_3=\{f_1\circ f_3, f_2\circ f_3\}=\{f_3, f_6\}\\ Hf_4=\{f_1\circ f_4, f_2\circ f_4\}=\{f_4, f_5\}\\ Hf_5=\{f_1\circ f_5, f_2\circ f_5\}=\{f_5, f_4\}\\ Hf_6=\{f_1\circ f_6, f_2\circ f_6\}=\{f_6, f_3\}

        HH 有三个不同的陪集,且左右陪集不相等,例如 f3HHf3f_3H≠Hf_3

        # 不同元素陪集相同充要条件

        a,bG\forall a,b\in G

        • Ha=HbHa=Hb 当且仅当 aHba\in Hb 当且仅当 ab1Hab^{-1}\in H
        • aH=bHaH=bH 当且仅当 abHa\in bH 当且仅当 b1aHb^{-1}a\in H

        # 陪集不变的充要条件

        HH 是群 GG 的子群,对任意的 aGa\in G

        • HaHHa=H 的充分必要条件是 aHa\in H

        引理

        eHe\in H ,所以 aeaHaa=ea\in Ha


        证明

        必要性:若 HaHHa=H ,则因为 aHaa\in Ha ,从而 aHa\in H

        充分性:反之若 aHa\in H ,则对任意 haHaha\in Ha ,由 HH 对群运算封闭,也有 haHha\in H ,从而有 HaHHa\subseteq H

        而这时对任意 hHh\in H ,由 aHa\in Ha1Ha^{-1}\in H ,从而 ha1Hha^{-1}\in H ,从而有 hha1aHah=ha^{-1}a\in Ha ,即有 HHaH\subseteq Ha

        于是又 HaH,HHaHa\subseteq H,H\subseteq Ha 得到 Ha=HHa=H

        # 陪集的子群充要条件

        HH 是群 GG 的子群,对任意的 aGa\in G

        • 右陪集 HaHaGG 的子群当且仅当 Ha=HHa=H 当且仅当 aHa\in H
        • 左陪集 aHaHGG 的子群当且仅当 aH=HaH=H 当且仅当 aHa\in H

        证明以右陪集为例:

        必要性:若 HaGHa\leq G ,所以 eHae\in Ha ,从而存在 hHh\in H 使得 ehae=ha ,从而 ah1e=h1Ha=h^{-1}e=h^{-1}\in H

        充分性:若 aHa\in H ,则 HaHHa=H 是子群。

        # 子群陪集运算性质

        HH 是群 GG 的子群,则 a,bG\forall a,b\in G 有:

        aHbab1HHa=Hba\in Hb\Longleftrightarrow ab^{-1}\in H\Longleftrightarrow Ha=Hb


        循环论证:


        (1)aHbab1Ha\in Hb\Longrightarrow ab^{-1}\in H

        aHba\in Hb ,即存在 hHh\in H 使得 a=hba=hb 。从而 ab1hHab^{-1}=h\in H


        (2)ab1HHa=Hbab^{-1}\in H\Longrightarrow Ha=Hb

        对于任意的 haHaha\in Ha ,这里 hHh\in H ,因此 hab1Hhab^{-1}\in H,从而:

        ha=hab1bHbha=hab^{-1}b\in Hb

        这表明 HaHbHa\subseteq Hb 。类似地,对任意的 hbHahb\in Ha ,这里 hHh\in H 。由于 ab1Hab^{-1}\in H ,则也有:

        ba1=(b1)1a1=(ab1)1Hba^{-1}=(b^{-1})^{-1}a^{-1}=(ab^{-1})^{-1}\in H

        因此 ha=hba1aHaha=hba^{-1}a\in Ha ,这表明 HbHaHb\subseteq Ha 。综合起来就有 Ha=BbHa=Bb


        (3)Ha=HbaHbHa=Hb\Longrightarrow a\in Hb

        Ha=HbHa=Hb ,则由 aHaa\in HaHH 是子群)可得 aHba\in Hb

        # 子群陪集构成等价关系

        HH 是群 GG 的子群,在 GG 上定义二元关系 RG×GR\in G\times G

        a,bG,aRbab1H\forall a,b\in G,\ \ \ a\text{R}b\Longleftrightarrow ab^{-1}\in H

        RRGG 上的等价关系,且 [a]R=Ha[a]_R=Ha


        证明

        上面已经证明了 ab1HHa=Hbab^{-1}\in H\Longleftrightarrow Ha=Hb,因此 aRbHa=Hba\text{R}b\Longleftrightarrow Ha= Hb

        从而 RR 确实是自反、对称和传递的,即 RR 是等价关系。

        对任意 bGb\in G ,我们有:

        bHaHb=HabRab[a]Rb\in Ha\Longleftrightarrow Hb=Ha\Longleftrightarrow b\text{R}a\Longleftrightarrow b\in[a]_R

        这就表明 Ha=[a]RHa=[a]_R

        # 子群陪集构成划分

        HGH≤G,则:

        • (i)a,bG\forall a,b\in G ,
          • Ha=HbHa=Hb
          • HaHb=Ha\cap Hb=\varnothing
        • (ii)U{HaaG}=G\mathbb U\{Ha|a\in G\}=G

        证明 —— 由等价类的性质立即可得。

        # 正规子群

        若对任意元素 aGa\in G 都有 aH=HaaH=Ha,即左右陪集相等,则称 HHGG 的正规子群,

        # 左右陪集关系

        一个子群的左陪集的所有元素的逆元素组成这个子群的一个右陪集

        符号化描述:

        HGH\leq GaGa\in G

        S={x1xaH}S=\{x^{-1}|x\in{aH}\}

        # 拉格朗日定理

        # 引理

        HHGG 的子群,对任意 aGa\in G ,有 H=aH=Ha|H|=|aH|=|Ha|,即这三个集合等势

        集合等势:若两个集合前存在双函数,即称它们等势。对于有限集,就是这两个集合有相同元素个数


        证明

        定义 φ:HHa\varphi:H\to Ha 为:

        hH,φ(h)=haHa\forall h\in H,\ \ \ \varphi(h)=ha\in Ha

        由群的消去律我们有:h1,h2H\forall h_1,h_2\in H

        φ(h1)=φ(h2)h1a=h2ah1=h2\varphi(h_1)=\varphi(h_2)\Longrightarrow h_1a=h_2a\Longrightarrow h_1=h_2

        从而 φ\varphi 是双函数,即 Ha=H|Ha|=|H|

        类似定义 φ:Ha\varphi':H\to a

        hH,φ(h)=ahaH\forall h\in H,\ \ \ \varphi'(h)=ah\in aH

        不难证明 aH=H|aH|=|H|

        # 指标

        HH 是群 GG 的子群。称子群 HH 在群 GG 中的左陪集或右陪集的个数(有限或无限)为 HHGG 中的指标 (index),记为 [G:H][G:H]

        也称 [G:H][G:H]GG 关于 HH 的陪集个数。

        # 拉格朗日定理

        HH 是有限群 GG 的子群,则:

        G=H[G:H]|G|=|H|\cdot[G:H]


        证明

        [G:H]=r[G:H]=r ,则子群 HH​ 有 rr 个不同的右陪集,设为 Ha1,Ha2,,HarH_{a_1},H_{a_2},\cdots,H_{a_r} 。由于 G=U{HaaG}G=U\{Ha|a\in G\} ,且对任意的 a,bGa,b\in GHa=HbHa=Hb 或者 HaHb=Ha\cap Hb=\varnothing ,因此:

        G=Ha1Ha2HarG=Ha1+Ha2++HarG=H_{a_1}\cup H_{a_2}\cup \cdots\cup H_{a_r}\\ |G|=|H_{a_1}|+|H_{a_2}|+\cdots+|H_{a_r}|

        而对任意的 aGa\in GHa=H|Ha|=|H| ,因此 G=rH=H[G:E]|G|=r|H|=|H|\cdot[G:E]

        # 推论 1 (元素阶与群阶的关系)

        GGnn 阶有限群,单位元是 ee ,则对 GG 的任意元素 aaaa 的阶 a|a|nn 的因子,且 an=ea^{n}=e


        证明

        对任意的 aGa\in G<a><a>GG 的子群,因此由拉格朗日定理 <a>|<a>|nn 的因子;

        但另一方面,若 a=r|a|=r ,则因 <a><a>aa 生成的子群,于是

        <a>={a0=e,a1,,ar1}<a>=\{a^{0}=e,a^{1},\cdots,a^{r-1}\}

        <a>=a=r|<a>|=|a|=r ,从而 a|a| 是群 G 的阶 n 的因子,进而由 == 定理 1.6?== 有 an=ea^{n}=e ,因为对任意的 mZm\in \mathbb{Z} , am=ea^{m}=e 当且仅当 rmr|m

        元素的阶是群的阶的因子;

        但并不是群的阶的每个因子,群都存在元素的阶恰好等于这个因子!

        # 推论 2 (费马小定理)

        pp 是素数,aa 是与 pp 互素的整数,则有:

        ap11(modp)a^{p-1}\equiv1(\bmod p)


        证明

        因为在群 U(p)U(p) ,即群 Zp\mathbb{Z}_p^* 中,由于 aapp 互素,aa(准确地说,aa 整除 pp 的余数)属于 Zp\mathbb{Z}_p^*

        Zp\mathbb{Z}_p^* 的阶是 p1p-1 ,因此 ap1a^{p-1} 等于单位元 1 ,对于 U(p)U(p) 的模 pp 乘运算而言,就是 ap11(modp)a^{p-1}\equiv 1(\bmod p)

        # 推论 3

        H,KH,K 都是 GG 的有限子群,证明:

        • (1) 记 S={hKhH}S=\{hK|h\in H\},则 SSHH划分,从而 HK=SK|HK|=|S|\cdot|K|
        • (2)HK=HKHK|HK|=\frac{|H|\cdot|K|}{|H\cap K|}

        (1)证明

        即证 h1Kh_1Kh2Kh_2K 要么相等,要么不相交

        h1K=h2Kh11h2Kh1Kh2K=h_1K=h_2K\lrArr h_1^{-1}h_2\in K\\ h_1K\cap h_2K=\varnothing


        (2)证明

        为方便起见,记 M=HKM=H\cap K ,由于 HHKK 都是 GG 的子群,因此 MM 也是 GG 的子群且也是 HH 的子群。

        定义函数 φ:H/MS\varphi:H/M\to S,对任意 hHh\in Hφ(hM)=hK\varphi(hM)=hK

        下面证明 φ\varphi 的定义是合适的,且是双函数,从而结合拉格朗日定理有 S=H/M=H/M|S|=|H/M|=|H|/|M| ,从而得到 HK=HKM|HK|=\frac{|H|\cdot|K|}{|M|}

        (a) 证明 φ\varphi 的定义是合适的:

        对任意 h1,h2Hh_1,h_2\in H ,若 h1M=h2Mh_1M=h_2M ,则 h11h2MKh_1^{-1}h_2\in M\subseteq K ,从而 φ(h1M)=φ(h2M)\varphi(h_1M)=\varphi(h_2M),这说明 φ(hM)\varphi(hM) 的值不会因为选取的代表 hh 不同面不同,即 φ\varphi 的定义是合适的。

        (b) 证明 φ\varphi 是单函数:

        对任意 h1,h2Hh_1,h_2\in H,若 φ(h1M)=φ(h2M)\varphi(h_1M)=\varphi(h_2M),即 h1K=h2Kh_1K=h_2K ,从而 h11h2Kh_1^{-1}h_2\in K

        h1,h2Hh_1,h_2\in H ,因此也有 h11h2Hh_1^{-1}h_2\in H

        从而 h11h2HK=Mh_1^{-1}h_2\in H\cap K=M

        从而 h1M=h2Mh_1M=h_2M ,这就表明 φ\varphi 是单函数。

        (c) 显然 φ\varphi 是满函数,对任意 hKShK\in S ,都有 φ(hM)=hk\varphi(hM)=hk

        因此 S=H/M|S|=|H/M| ,由拉格朗日定理有 S=H/M=H/M|S|=|H/M|=|H|/|M| ,最后得到:

        HK=HMK=HKHK|HK|=\frac{|H|}{|M|}|K|=\frac{|H|\cdot|K|}{|H\cap K|}

        # 举例

        不容易找到两个子群的交不等于 {e}\{e\} 的例子,利用计算机程序可找到如下的例子:U(33)={1,2,4,5,7,8,10,13,14,16,17,19,20,23,25,26,28,29,31,32}U(33)=\{1,2,4,5,7,8,10,13,14,16,17,19,20,23,25,26,28,29,31,32\}

        • 子群 {1,2,4,8,16,17,25,29,31,32}\{1,2,4,8,16,17,25,29,31,32\} 是循环子群,生成元为:2,8,17,292,8,17,29
        • 子群 {1,4,5,14,16,20,23,25,26,31}\{1,4,5,14,16,20,23,25,26,31\} 是循环子群,生成元为:5,14,20,265,14,20,26

        上述两个子群交集为 {1,4,16,25,31}\{1,4,16,25,31\}

        # 西罗定理

        可了解,不考

        GGnn 阶循环群,aa 是生成元。对于 nn 的每个正因子 kk , 有且仅有一个 kk 阶循环子群。并且由 anka^{\frac{n}{k}} 生成.

        \ No newline at end of file diff --git a/math/discrete-math/algebra/group-basics/index.html b/math/discrete-math/algebra/group-basics/index.html index 7a465120..d9f5cde1 100644 --- a/math/discrete-math/algebra/group-basics/index.html +++ b/math/discrete-math/algebra/group-basics/index.html @@ -1 +1 @@ -群论基础 - 代数结构 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 群 (Group) 的定义

        如果集合 G 及其二元运算 \cdot 满足下面的性质,则称 (G,)(G,\cdot ) 是群:

        • 二元运算 \cdot 满足结合律:即 a,b,cG∀a,b,c\in G(ab)c=a(bc)(a\cdot b)\cdot c=a\cdot(b\cdot c)
        • 单位元 ee,即存在 eGe\in G,使得 aG∀a\in Gea=ae=ae\cdot a=a\cdot e=a
        • aG∀a\in G 关于 \cdot逆元 a1a^{-1},即 aG∀a\in G,存在 bGb\in G,使得 ab=ba=ea\cdot b=b\cdot a=e,记 bba1a^{-1}

        从代数系统角度说,群有三个运算:

        • 二元运算 \cdot
        • 一元运算 ()1(-)^{-1}
        • 零元运算 ee

        其中二元运算占主导地位,在谈到群时经常只明确给出它的二元运算。进一步,在上下文能明确其运算时,人们通常直接称集合 GG 是群

        如果集合 SS 有满足结合律的二元运算 \cdot ,则称 (S,)(S,\cdot)半群 (semi-group)

        如果 \cdot 还有单位元 ee,则称 (S,,e)(S,\cdot,e)独异点 (monoid)

        # 常用群举例

        # 整数加群

        整数集 Z 及整数加法运算构成群,称为整数加群

        • 单位元是 00
        • 每个整数 zz 的逆元是 z-z

        # 模 m 剩余类加群 Zm\mathbb{Z}_m

        对整数 m2m≥2,记 Zm={0,1,,m1}\mathbb{Z}_m=\{0,1,\cdots,m-1\}Zm\mathbb{Z}_m 及 模 mmm\oplus_m 构成群,称为mm 剩余类加群

        00 是单位元,对每个整数 zZmz\in \mathbb Z_m,它的逆元是 ((z)+m)modm((-z)+m)\bmod m

        模m剩余类加法群

        # Q\mathbb{Q^*} R\mathbb{R^*} C\mathbb{C^*} 乘法群

        整数集 Z\mathbb Z 关于整数乘法不构成群,但非零有理数集Q\mathbb Q^*、非零实数集 R\mathbb R^*、非零复数集 C\mathbb C^* 关于数的乘法运算构成群

        # 实矩阵乘法群

        实数域上的 n 阶方阵构成的集合为 Mn(R)M_n (\mathbb R),则它关于矩阵加法构成群;记实数域上的 n 阶可逆方阵构成的集合为 GLn(R)GL_n (\mathbb R),则它关于矩阵乘法构成群

        # 双函数复合群

        集合 S 上的所有双函数 f:SSf:S→S 及复合构成群

        • 恒等函数是单位元
        • 逆函数给出逆元

        # 模 m 单位群 U (m)

        Zm\mathbb Z_m 及 模 m 乘 m⊗_m 不构成群,但令U(m)={aaZm且a与m互质}U(m)=\{a∣a\in Z_m \text{且a与m互质}\},则 U (m) 与m⊗_m 构成群,称为模 m 单位群

        • 1 是单位元
        • 对每个整数 zU(m)z\in U(m),它的逆元是同余方程 xz1(modm)xz≡1(\text{mod} m) 的解
          • 利用贝祖定理(即 a,bZ+s,tZ,gcd(a,b)=as+bt∀a,b\in \mathbb Z^+,\exist s, t\in \mathbb Z,gcd⁡(a,b)=as+bt)可证明 U (m) 对 m⊗_m 封闭,以及上述同余方程解的存在性,也即逆元的存在性

        当 p 是素数,则 U (p) 群也记为 Zp\mathbb Z_p^*

        模 m 单位群的阶:U(m)=ϕ(m)|U(m)|=ϕ(m),这里 ϕ(m)ϕ(m) 是欧拉函数,即:

        U(m)=ϕ(m)=mi=1s(11pi)|U(m)|=ϕ(m)=m\prod_{i=1}^{s}(1-\frac{1}{p_i} )

        这里 p1,,psp_1,⋯, p_s 是 m 的 s 个不同的素因子

        # 具体群举例

        1

        2

        3克莱因4元群

        4

        # 群的一些术语

        # 交换群

        群的运算不一定满足交换律,满足交换律的群称为交换群,也称为阿贝尔群

        # 群的阶

        群 G 的元素个数(准确地说 G 的基数)|G | 称为群 G 的阶 (order),如果 G 是有穷集,则称为有穷群(有限群),否则称为无穷群(无限群)

        # 加群

        当群称为加群时通常将它:

        • 单位元记为 0
        • 元素 a 的逆元记为 - a
        • 运算称为加法
        • 运算的结果称为和
        • 运算用加号 + 表示

        通常只有当群是交换群的时候,才使用加号 '+' 表示这个群的运算

        # 乘群

        将不是加群的群称为乘群,并将:

        • 乘群的单位元通常用 1 或 e 表示
        • 元素 a 的逆元用a1a^{-1} 表示
        • 运算称为乘法
        • 运算结果称为积
        • 运算用 \circ\cdot 或 * 表示,但通常省略不写!

        # 群的一些基本性质

        群有单位元,因此群不可能是空集

        群要求每个元素都有逆元,而运算的零元不可能有逆,因此群没有零元(除平凡群 ({e},)(\{e\},\circ) 外)!

        # 逆运算基本性质

        GG 是群,有:

        aG(a1)1=a∀a\in G,(a^{-1})^{-1}= a

        a,bG,(ab)1=b1a1∀a, b\in G, (ab)^{-1}= b^{-1} a^{-1}


        证明

        根据逆元的定义有 aaa1a^{-1} 互为逆元。

        (b^{-1} a^{-1})(ab)=b^{-1}(a^{-1}a)b=b^{-1}eb=b^{-1} b=e$

        类似地:

        ab(b1a1)=eab(b^{-1}a^{-1})=e

        因此 ababb1a1b^{-1}a^{-1} 互为逆元。

        # 群满足消去律

        群的二元运算满足消去律

        GG 是群,对任意 a,b,cGa,b,c\in G, ab=acab=ac 蕴涵 b=cb=c ,同样 ba=caba=ca 蕴涵 b=cb=c


        证明

        ab=acab=ac,则有 a1(ab)=a1(ac)a^{-1}(ab)=a^{-1} (ac)

        而群运算满足结合律,因此 (a1a)b=(a1a)c(a^{-1} a)b=(a^{-1} a)c

        a1a=ea^{-1} a=eee 是单位元,从而有 b=cb=c

        同理可证 ba=caba=ca 蕴涵 b=cb=c

        # 群的充要条件 1

        给定非空集 GG 及其上的二元运算(假定是乘法运算而省略其运算符号)。GG 关于该二元运算构成群的充分必要条件是:

        • GG 的二元运算满足结合律
        • GG 的二元运算有左单位元:即存在 eGe\in G 使得对任意的 aGa\in Gea=aea=a
        • GG 的每个元素相对左单位元有左逆元:即 aG\forall a\in G,存在 aGa'\in G 使得 aa=ea'a=e

        “左” 换成 “右” 也一样

        证明:

        aG,aG,aa=eaaa=aa(aa)=aae=a\forall a\in G,\ \ \exist a'\in G,\ \ a'a=e\\ aa'a=a\\ a(a'a)=a\\ ae=a

        所以 e 也是右单位元;

        aG,aG,aa=eaG,aa=eaa=eaa=aaaa=a(aa)a=aea=aa=e\forall a\in G,\ \ \exist a'\in G,\ \ a'a=e\\ \exist a''\in G,\ \ a''a'=e \\ aa'=eaa'=a''a'aa'=a''(a'a)a'=a''ea'=a''a'=e

        所以 每个元素的左逆元也是右逆元。

        # 群的充要条件 2

        给定非空集 GG 及其二元运算,且该二元运算满足结合律GG 构成群的充分必要条件是:

        对任意的 a,bGa,b\in G,方程 ax=bax=bya=bya=bGG 中都有解。

        # 问题 5

        给定非空集 GG 及其上的二元运算(假定是乘法运算而省略其运算符号)满足:

        • 该运算有左单位元 ee ,即 aG∀a\in Gea=aea=a
        • 对关于该左单位元,每个元素有右逆,即任意 aGa\in G ,存在 bb 使得 ab=eab=e

        请问 GG 关于该运算是否一定构成群?

        # 群的充分条件

        给定非空有限集 GG 及其二元运算,该二元运算满足结合律满足消去律,即对任意 a,b,cGa, b, c\in Gab=acab = ac 蕴含 b=cb = c 以及 ba=caba = ca 蕴含 b=cb = c

        证明 GG 关于该运算构成群。

        # 群的幂运算

        # 定义

        (G,)(G,\circ) 是群,定义群的幂运算:aG,nZ∀a\in G, ∀n\in Z ,定义 aann 次幂,记为 ana^n

        群元素的阶

        # 加群的幂运算

        对于加群,元素的幂运算实际上是倍数运算

        • 例如对于整数加群 (Z,+)(Z, +) ,整数 zznn 次幂实际上是 nznz 。这时有:
          • 0z=00z=0
          • (nz)=n(z)(-nz)= n(-z)
          • nz+mz=(n+m)znz + mz = (n+m)z
          • m(nz)=(mn)zm(nz)= (mn)z

        # 基本性质

        (G,)(G, \circ) 是群,对任意 aG,m,nZa\in G,\ \ m, n\in Z ,有:

        anam=an+m(an)m=anma^n\circ a^m= a^{n+m}\\(a^n )^m=a^{nm}


        证明

        • 先对于 m0m≥0,对 mm 实施数学归纳法;
        • 再对于 m<0m<0 ,利用 am=(a1)ma^m=(a^{-1})^{-m}

        # 常用性质

        a,ba,b 是群 GG 的元素,容易得到:

        nZ,(aba1)n=abna1\forall n\in \mathbb Z\ ,\ (aba^{-1} )^n=ab^n a^{-1}

        # 群元素的阶

        # 定义

        (G,)(G,∘) 元素 aa阶 (order),记为 a|a| ,是指最小的正整数 kk 使得 ak=ea^k=e ,若这样的正整数不存在,则称元素 aa 的阶无穷

        # 任意幂的阶

        GG 是群,ee 是其单位元,aaGG 的任意元素,且 a=n|a|=n

        • 对任意整数 mmam=ea^m=e 当且仅当 nmn|m
        • 特别的,m=1m=-1 时,a1=n|a^{-1}|=n
        • 对任意整数 mm : am=ngcd(n,m)|a^m| = \frac{n}{gcd(n, m)}

        定理9证明

        # 阶内运算可轮换

        a,b,ca,b,c 是群 GG 的元素,证明:

        • a=a1=cac1|a|=|a^{-1}|=|cac^{-1}|
        • ab=ba|ab|=|ba|
        • abc=bca=cab|abc|=|bca|=|cab|

        (2) 因为 ab=a(ba)a1ab = a(ba)a^{-1} ,所以由 (1) 可知:

        ab=a(ba)a1=ba|ab|=|a(ba)a^{-1}|=|ba|


        (3) 因为:

        • abc=a(bc)abc=a(bc)
        • bca=(bc)abca=(bc)a
        • cab=(ca)bcab=(ca)b

        因此由 (1) 可知:

        abc=a(bc)=(bc)a=bca=b(ca)=(ca)b=cab|abc|=|a(bc)|=|(bc)a|=|bca|=|b(ca)|=|(ca)b|=|cab|

        虽然 abc=bca|abc|=|bca| ,但是 abcbac|abc|\neq|bac| !这与置换群有关

        # 子群

        # 定义

        GG 是群,HHGG 的子集,如果 HH 关于 GG 的运算也构成群,则称 HHGG 的子群,记为:

        HGH\leq G

        HHGG 的子群,且 HGH≠G ,则称 HHGG真子群,记为:

        H<GH\lt G

        显然 H={e}H=\{e\}GG 的子群,这里 eeGG 的单位元,将 {e}\{e\}GG 自己称为群 GG平凡子群,其他子群称为非平凡子群

        从代数系统角度说,子群就是子代数,下面定理说明子群对群的三个运算都封闭!

        # 子群判定定理 1

        给定 GG 是群,HGH⊆G 关于 GG 的运算构成群,也就是 HHGG 的子群当且仅当:

        • HHGG 的 (二元) 运算封闭
        • HH 对零元运算封闭,即 GG 的单位元应属于 HH
        • HH 对一元运算封闭,即 aH,a1H\forall a\in H,\ \ a^{-1}\in H

        # 子群判定定理 2

        群 G 的非空子集 H 是 G 的子群当且仅当:

        • eHe\in H
        • a,bH,abH∀a, b\in H,\ \ \ ab\in H

        子群判定定理一证明1子群判定定理一证明2

        实际上是构造 aa=ajia^{|a|} = a^{j-i} ,其中 a>0|a|>0aa 的阶

        # 子群判定定理 3

        群 G 的非空子集 H 是 G 的子群当且仅当:

        • a,bH,ab1H∀a, b\in H,\ \ \ ab^{-1}\in H
        • a,bH,b1aH∀a, b\in H,\ \ \ b^{-1}a\in H

        # 群的中心

        定义群 G 的中心

        C={aGxG,xa=ax}C=\{a\in G∣∀x\in G, xa=ax\}

        即 C 是 G 中那些与 G 的任何元素都可交换的元素构成的集合.

        证明:C 是 G 的子群 ( CGC\leq G )

        • 首先 C 是否是非空集?哪个元素一定属于 C?
        • 其次,如何运用子群的判定定理证明 C 是 G 的子群?
          • 例如利用子群判定定理二,如何证明对任意 a, b\in C,都有 ab-1\in C ?
          • 或者利用子群判定定理一,证明对任意 a\in C 有 a-1\in C,以及对任意 a,b\in C 有 ab\in C ?

        首先,由群的性质:xG,xe=ex=e\forall x\in G,\ \ \ xe=ex=e ,所以 eCe\in C ,因此 C 非空

        a,bC\forall a,b\in C ,为证明 ab1Cab^{-1}\in C 只需证明它与 G 中所有元素都可交换即可。

        xG\forall x\in G ,有:

        (ab1)x=ab1(x1)1=a(x1b)1=bCa(bx1)1=a(xb1)=结合律(ax)b1=aC(xa)b1=结合律x(ab1)\begin{aligned} (ab^{-1})x &=ab^{-1}(x^{-1})^{-1}\\ &=a(x^{-1}b)^{-1}\\ &\overset{b\in C}{=}a(bx^{-1})^{-1}\\ &=a(xb^{-1})\\ &\overset{\text{结合律}}{=}(ax)b^{-1}\\ &\overset{a\in C}{=}(xa)b^{-1}\\ &\overset{\text{结合律}}{=}x(ab^{-1}) \end{aligned}

        # 交并与子群的关系

        GG 是群,且 HGH≤G , KGK≤G ,证明:

        • $ H\cap K≤G$(交保持子群,可推广)
        • $ H∪K≤G$ 当且仅当 HKH⊆KKHK⊆H并保持子群的充要条件

        问题15证明1问题15证明2

        # 生成子群

        简单地说,群 GG 的一个子集的生成子群就是包含这个子集的最小的子群。群 GG 的子集 SS 的生成子群记为:

        S⟨S⟩

        两个子群的交总是子群这个命题可推广到任意多个子群的交也总是子群,从而对于群 GG 的某个子集 SS ,可给出包含 SS 的最小子群,这个子群就是 SS生成子群 S⟨S⟩ ,可由所有包含 SS 的子群的交给出,即:

        <S>={HSHHG}\text{<}S\text{>}=\cap\{H|S\subseteq H \land H\leq G\}

        • 如果 SS 为有限集 S={a1,a2,,an}S=\{a_1, a_2,\cdots,a_n\} ,将 S⟨S⟩ 直接记为 a1,a2,,an⟨a_1, a_2,\cdots,a_n ⟩
        • 对于群 GG ,若存在 aGa\in G ,使得 G=aG=⟨a⟩ ,即 GG 是由元素 aa 生成的群,这时称 G 为循环群

        空集生成的子群是什么?

        # 生成子群元素的表示

        GG 是群,SSGG 的非空子集,则:

        S={a1l1a2l2...aklkaiS,li=±1,kN}⟨S⟩=\cap\{a_{1}^{l_{1}}a_{2}^{l_{2}}...a_{k}^{l_{k}}| a_{i}\in S,\ \ \ l_{i}=\pm 1,\ \ \ k\in\mathbb{N} \}

        S⟨S⟩ 的任意元素可表示成有限个 S 的元素或其逆的乘积

        问题16证明

        S={a}S=\{a\} 时,即有 ⟨a⟩=\

        # 循环群

        GG 是群,如果存在 aGa\in G 使得

        G=a={akkZ}G=⟨a⟩=\{a^k | k\in Z\}

        则称 GG循环群 (cyclic group),并称 aaGG 的一个生成元

        • GG 是无限集时称为无限循环群
        • GGnn 个元素时,称为 n 阶循环群

        # 性质 (不考证明)

        GG 是群,ee 是它的单位元

        1. a1=a⟨a^{-1}⟩= ⟨a⟩
        2. GG 是有限群,则 G=aG=aG=⟨a⟩⟺|G|= |a|
        3. GG 是无限循环群,则
          • G={e,a,a1,a2,a2,,an,an,}G = \{e, a, a^{-1}, a^2, a^{-2}, ⋯, a^n, a^{-n}, ⋯\}
          • k,lZ,ak=alk=l\forall k, l\in Z,\ \ \ a^k= a^l⟺k= l
        4. GGnn 阶循环群,则
          • G={e,a,a2,a3,,an1}G = \{e, a, a^2, a^{3}, ⋯, a^{n-1}\}
          • k,lZ,ak=alnkl\forall k, l\in Z,\ \ \ a^k = a^l ⟺n | k-l

        # 循环群举例

        整数加群 Z\mathbb Z 是无限循环群

        显然 Z=1=1\mathbb Z= ⟨1⟩= ⟨-1⟩ , 即 1 和 -1 是它的生成元

        对任意 dZd\in \mathbb Z ,若 d±1d≠±1 ,显然有 1∉d1\not\in⟨d⟩ ,因此 1 和 -1 是 Z\mathbb Z 的仅有的两个生成元。


        mm 加群 Zm\mathbb Z_mmm 阶循环群,显然 Zm=1\mathbb Z_m=⟨1⟩


        U(5)U(5) 是 4 阶循环群,U(5)=2=3U(5)=⟨2⟩=⟨3⟩

        mm 是素数时,U(m)U(m)m1m-1 阶循环群(证明这个需要不少数论知识)

        mm 不是素数时:

        • U(m)U(m) 可能是循环群(例如 U(9)={1,2,4,5,7,8}=2U(9)=\{1,2,4,5,7,8\}=⟨2⟩
        • 也可能不是循环群(例如U(15)={1,2,4,7,8,11,13,14}U(15)=\{1,2,4,7,8,11,13,14\}

        # 生成元的数量与结构

        定理17


        定理17证明

        # 子群保持循环群

        循环群的任意子群都是循环群


        定理18证明

        # 问题

        如果一个群的所有非平凡子群都是循环群,那么这个群一定是循环群吗?

        不一定

        U(12)={1,5,7,11}U(12)=\{1,5,7,11\} 不是循环群,但它的每个非平凡子群都是循环群!

        1=1,5=2,7=2,11=2|1|=1 , |5| = 2 , |7| = 2 , |11| = 2 ,可见 U(12)U(12) 的所有非平凡子群都是二阶群,不难证明所有二阶群都是循环群

        # 生成元任意幂的生成子群

        推论19

        # 循环群的子群⭐️

        推论20

        推论20证明1推论20证明2

        例题:

        Z14\mathbb{Z}_{14} 加群的所有子群是:

        • 1=Z14⟨1⟩=\mathbb{Z}_{14}
        • 2={0,2,4,6,8,10,12,14}⟨2⟩=\{0,2,4,6,8,10,12,14\}
        • 7={0,7}⟨7⟩=\{0,7\}
        • 0=0⟨0⟩={0}

        U(14)U(14) 的所有子群是:

        • 3⟨3⟩
        • 32=9⟨3^2=9⟩
        • 33=13⟨3^3=13⟩
        • 36=1⟨3^6=1⟩
        \ No newline at end of file +群论基础 - 代数结构 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 群 (Group) 的定义

        如果集合 G 及其二元运算 \cdot 满足下面的性质,则称 (G,)(G,\cdot ) 是群:

        • 二元运算 \cdot 满足结合律:即 a,b,cG∀a,b,c\in G(ab)c=a(bc)(a\cdot b)\cdot c=a\cdot(b\cdot c)
        • 单位元 ee,即存在 eGe\in G,使得 aG∀a\in Gea=ae=ae\cdot a=a\cdot e=a
        • aG∀a\in G 关于 \cdot逆元 a1a^{-1},即 aG∀a\in G,存在 bGb\in G,使得 ab=ba=ea\cdot b=b\cdot a=e,记 bba1a^{-1}

        从代数系统角度说,群有三个运算:

        • 二元运算 \cdot
        • 一元运算 ()1(-)^{-1}
        • 零元运算 ee

        其中二元运算占主导地位,在谈到群时经常只明确给出它的二元运算。进一步,在上下文能明确其运算时,人们通常直接称集合 GG 是群

        如果集合 SS 有满足结合律的二元运算 \cdot ,则称 (S,)(S,\cdot)半群 (semi-group)

        如果 \cdot 还有单位元 ee,则称 (S,,e)(S,\cdot,e)独异点 (monoid)

        # 常用群举例

        # 整数加群

        整数集 Z 及整数加法运算构成群,称为整数加群

        • 单位元是 00
        • 每个整数 zz 的逆元是 z-z

        # 模 m 剩余类加群 Zm\mathbb{Z}_m

        对整数 m2m≥2,记 Zm={0,1,,m1}\mathbb{Z}_m=\{0,1,\cdots,m-1\}Zm\mathbb{Z}_m 及 模 mmm\oplus_m 构成群,称为mm 剩余类加群

        00 是单位元,对每个整数 zZmz\in \mathbb Z_m,它的逆元是 ((z)+m)modm((-z)+m)\bmod m

        模m剩余类加法群

        # Q\mathbb{Q^*} R\mathbb{R^*} C\mathbb{C^*} 乘法群

        整数集 Z\mathbb Z 关于整数乘法不构成群,但非零有理数集Q\mathbb Q^*、非零实数集 R\mathbb R^*、非零复数集 C\mathbb C^* 关于数的乘法运算构成群

        # 实矩阵乘法群

        实数域上的 n 阶方阵构成的集合为 Mn(R)M_n (\mathbb R),则它关于矩阵加法构成群;记实数域上的 n 阶可逆方阵构成的集合为 GLn(R)GL_n (\mathbb R),则它关于矩阵乘法构成群

        # 双函数复合群

        集合 S 上的所有双函数 f:SSf:S→S 及复合构成群

        • 恒等函数是单位元
        • 逆函数给出逆元

        # 模 m 单位群 U (m)

        Zm\mathbb Z_m 及 模 m 乘 m⊗_m 不构成群,但令U(m)={aaZm且a与m互质}U(m)=\{a∣a\in Z_m \text{且a与m互质}\},则 U (m) 与m⊗_m 构成群,称为模 m 单位群

        • 1 是单位元
        • 对每个整数 zU(m)z\in U(m),它的逆元是同余方程 xz1(modm)xz≡1(\text{mod} m) 的解
          • 利用贝祖定理(即 a,bZ+s,tZ,gcd(a,b)=as+bt∀a,b\in \mathbb Z^+,\exist s, t\in \mathbb Z,gcd⁡(a,b)=as+bt)可证明 U (m) 对 m⊗_m 封闭,以及上述同余方程解的存在性,也即逆元的存在性

        当 p 是素数,则 U (p) 群也记为 Zp\mathbb Z_p^*

        模 m 单位群的阶:U(m)=ϕ(m)|U(m)|=ϕ(m),这里 ϕ(m)ϕ(m) 是欧拉函数,即:

        U(m)=ϕ(m)=mi=1s(11pi)|U(m)|=ϕ(m)=m\prod_{i=1}^{s}(1-\frac{1}{p_i} )

        这里 p1,,psp_1,⋯, p_s 是 m 的 s 个不同的素因子

        # 具体群举例

        1

        2

        3克莱因4元群

        4

        # 群的一些术语

        # 交换群

        群的运算不一定满足交换律,满足交换律的群称为交换群,也称为阿贝尔群

        # 群的阶

        群 G 的元素个数(准确地说 G 的基数)|G | 称为群 G 的阶 (order),如果 G 是有穷集,则称为有穷群(有限群),否则称为无穷群(无限群)

        # 加群

        当群称为加群时通常将它:

        • 单位元记为 0
        • 元素 a 的逆元记为 - a
        • 运算称为加法
        • 运算的结果称为和
        • 运算用加号 + 表示

        通常只有当群是交换群的时候,才使用加号 '+' 表示这个群的运算

        # 乘群

        将不是加群的群称为乘群,并将:

        • 乘群的单位元通常用 1 或 e 表示
        • 元素 a 的逆元用a1a^{-1} 表示
        • 运算称为乘法
        • 运算结果称为积
        • 运算用 \circ\cdot 或 * 表示,但通常省略不写!

        # 群的一些基本性质

        群有单位元,因此群不可能是空集

        群要求每个元素都有逆元,而运算的零元不可能有逆,因此群没有零元(除平凡群 ({e},)(\{e\},\circ) 外)!

        # 逆运算基本性质

        GG 是群,有:

        aG(a1)1=a∀a\in G,(a^{-1})^{-1}= a

        a,bG,(ab)1=b1a1∀a, b\in G, (ab)^{-1}= b^{-1} a^{-1}


        证明

        根据逆元的定义有 aaa1a^{-1} 互为逆元。

        (b^{-1} a^{-1})(ab)=b^{-1}(a^{-1}a)b=b^{-1}eb=b^{-1} b=e$

        类似地:

        ab(b1a1)=eab(b^{-1}a^{-1})=e

        因此 ababb1a1b^{-1}a^{-1} 互为逆元。

        # 群满足消去律

        群的二元运算满足消去律

        GG 是群,对任意 a,b,cGa,b,c\in G, ab=acab=ac 蕴涵 b=cb=c ,同样 ba=caba=ca 蕴涵 b=cb=c


        证明

        ab=acab=ac,则有 a1(ab)=a1(ac)a^{-1}(ab)=a^{-1} (ac)

        而群运算满足结合律,因此 (a1a)b=(a1a)c(a^{-1} a)b=(a^{-1} a)c

        a1a=ea^{-1} a=eee 是单位元,从而有 b=cb=c

        同理可证 ba=caba=ca 蕴涵 b=cb=c

        # 群的充要条件 1

        给定非空集 GG 及其上的二元运算(假定是乘法运算而省略其运算符号)。GG 关于该二元运算构成群的充分必要条件是:

        • GG 的二元运算满足结合律
        • GG 的二元运算有左单位元:即存在 eGe\in G 使得对任意的 aGa\in Gea=aea=a
        • GG 的每个元素相对左单位元有左逆元:即 aG\forall a\in G,存在 aGa'\in G 使得 aa=ea'a=e

        “左” 换成 “右” 也一样

        证明:

        aG,aG,aa=eaaa=aa(aa)=aae=a\forall a\in G,\ \ \exist a'\in G,\ \ a'a=e\\ aa'a=a\\ a(a'a)=a\\ ae=a

        所以 e 也是右单位元;

        aG,aG,aa=eaG,aa=eaa=eaa=aaaa=a(aa)a=aea=aa=e\forall a\in G,\ \ \exist a'\in G,\ \ a'a=e\\ \exist a''\in G,\ \ a''a'=e \\ aa'=eaa'=a''a'aa'=a''(a'a)a'=a''ea'=a''a'=e

        所以 每个元素的左逆元也是右逆元。

        # 群的充要条件 2

        给定非空集 GG 及其二元运算,且该二元运算满足结合律GG 构成群的充分必要条件是:

        对任意的 a,bGa,b\in G,方程 ax=bax=bya=bya=bGG 中都有解。

        # 问题 5

        给定非空集 GG 及其上的二元运算(假定是乘法运算而省略其运算符号)满足:

        • 该运算有左单位元 ee ,即 aG∀a\in Gea=aea=a
        • 对关于该左单位元,每个元素有右逆,即任意 aGa\in G ,存在 bb 使得 ab=eab=e

        请问 GG 关于该运算是否一定构成群?

        # 群的充分条件

        给定非空有限集 GG 及其二元运算,该二元运算满足结合律满足消去律,即对任意 a,b,cGa, b, c\in Gab=acab = ac 蕴含 b=cb = c 以及 ba=caba = ca 蕴含 b=cb = c

        证明 GG 关于该运算构成群。

        # 群的幂运算

        # 定义

        (G,)(G,\circ) 是群,定义群的幂运算:aG,nZ∀a\in G, ∀n\in Z ,定义 aann 次幂,记为 ana^n

        群元素的阶

        # 加群的幂运算

        对于加群,元素的幂运算实际上是倍数运算

        • 例如对于整数加群 (Z,+)(Z, +) ,整数 zznn 次幂实际上是 nznz 。这时有:
          • 0z=00z=0
          • (nz)=n(z)(-nz)= n(-z)
          • nz+mz=(n+m)znz + mz = (n+m)z
          • m(nz)=(mn)zm(nz)= (mn)z

        # 基本性质

        (G,)(G, \circ) 是群,对任意 aG,m,nZa\in G,\ \ m, n\in Z ,有:

        anam=an+m(an)m=anma^n\circ a^m= a^{n+m}\\(a^n )^m=a^{nm}


        证明

        • 先对于 m0m≥0,对 mm 实施数学归纳法;
        • 再对于 m<0m<0 ,利用 am=(a1)ma^m=(a^{-1})^{-m}

        # 常用性质

        a,ba,b 是群 GG 的元素,容易得到:

        nZ,(aba1)n=abna1\forall n\in \mathbb Z\ ,\ (aba^{-1} )^n=ab^n a^{-1}

        # 群元素的阶

        # 定义

        (G,)(G,∘) 元素 aa阶 (order),记为 a|a| ,是指最小的正整数 kk 使得 ak=ea^k=e ,若这样的正整数不存在,则称元素 aa 的阶无穷

        # 任意幂的阶

        GG 是群,ee 是其单位元,aaGG 的任意元素,且 a=n|a|=n

        • 对任意整数 mmam=ea^m=e 当且仅当 nmn|m
        • 特别的,m=1m=-1 时,a1=n|a^{-1}|=n
        • 对任意整数 mm : am=ngcd(n,m)|a^m| = \frac{n}{gcd(n, m)}

        定理9证明

        # 阶内运算可轮换

        a,b,ca,b,c 是群 GG 的元素,证明:

        • a=a1=cac1|a|=|a^{-1}|=|cac^{-1}|
        • ab=ba|ab|=|ba|
        • abc=bca=cab|abc|=|bca|=|cab|

        (2) 因为 ab=a(ba)a1ab = a(ba)a^{-1} ,所以由 (1) 可知:

        ab=a(ba)a1=ba|ab|=|a(ba)a^{-1}|=|ba|


        (3) 因为:

        • abc=a(bc)abc=a(bc)
        • bca=(bc)abca=(bc)a
        • cab=(ca)bcab=(ca)b

        因此由 (1) 可知:

        abc=a(bc)=(bc)a=bca=b(ca)=(ca)b=cab|abc|=|a(bc)|=|(bc)a|=|bca|=|b(ca)|=|(ca)b|=|cab|

        虽然 abc=bca|abc|=|bca| ,但是 abcbac|abc|\neq|bac| !这与置换群有关

        # 子群

        # 定义

        GG 是群,HHGG 的子集,如果 HH 关于 GG 的运算也构成群,则称 HHGG 的子群,记为:

        HGH\leq G

        HHGG 的子群,且 HGH≠G ,则称 HHGG真子群,记为:

        H<GH\lt G

        显然 H={e}H=\{e\}GG 的子群,这里 eeGG 的单位元,将 {e}\{e\}GG 自己称为群 GG平凡子群,其他子群称为非平凡子群

        从代数系统角度说,子群就是子代数,下面定理说明子群对群的三个运算都封闭!

        # 子群判定定理 1

        给定 GG 是群,HGH⊆G 关于 GG 的运算构成群,也就是 HHGG 的子群当且仅当:

        • HHGG 的 (二元) 运算封闭
        • HH 对零元运算封闭,即 GG 的单位元应属于 HH
        • HH 对一元运算封闭,即 aH,a1H\forall a\in H,\ \ a^{-1}\in H

        # 子群判定定理 2

        群 G 的非空子集 H 是 G 的子群当且仅当:

        • eHe\in H
        • a,bH,abH∀a, b\in H,\ \ \ ab\in H

        子群判定定理一证明1子群判定定理一证明2

        实际上是构造 aa=ajia^{|a|} = a^{j-i} ,其中 a>0|a|>0aa 的阶

        # 子群判定定理 3

        群 G 的非空子集 H 是 G 的子群当且仅当:

        • a,bH,ab1H∀a, b\in H,\ \ \ ab^{-1}\in H
        • a,bH,b1aH∀a, b\in H,\ \ \ b^{-1}a\in H

        # 群的中心

        定义群 G 的中心

        C={aGxG,xa=ax}C=\{a\in G∣∀x\in G, xa=ax\}

        即 C 是 G 中那些与 G 的任何元素都可交换的元素构成的集合.

        证明:C 是 G 的子群 ( CGC\leq G )

        • 首先 C 是否是非空集?哪个元素一定属于 C?
        • 其次,如何运用子群的判定定理证明 C 是 G 的子群?
          • 例如利用子群判定定理二,如何证明对任意 a, b\in C,都有 ab-1\in C ?
          • 或者利用子群判定定理一,证明对任意 a\in C 有 a-1\in C,以及对任意 a,b\in C 有 ab\in C ?

        首先,由群的性质:xG,xe=ex=e\forall x\in G,\ \ \ xe=ex=e ,所以 eCe\in C ,因此 C 非空

        a,bC\forall a,b\in C ,为证明 ab1Cab^{-1}\in C 只需证明它与 G 中所有元素都可交换即可。

        xG\forall x\in G ,有:

        (ab1)x=ab1(x1)1=a(x1b)1=bCa(bx1)1=a(xb1)=结合律(ax)b1=aC(xa)b1=结合律x(ab1)\begin{aligned} (ab^{-1})x &=ab^{-1}(x^{-1})^{-1}\\ &=a(x^{-1}b)^{-1}\\ &\overset{b\in C}{=}a(bx^{-1})^{-1}\\ &=a(xb^{-1})\\ &\overset{\text{结合律}}{=}(ax)b^{-1}\\ &\overset{a\in C}{=}(xa)b^{-1}\\ &\overset{\text{结合律}}{=}x(ab^{-1}) \end{aligned}

        # 交并与子群的关系

        GG 是群,且 HGH≤G , KGK≤G ,证明:

        • $ H\cap K≤G$(交保持子群,可推广)
        • $ H∪K≤G$ 当且仅当 HKH⊆KKHK⊆H并保持子群的充要条件

        问题15证明1问题15证明2

        # 生成子群

        简单地说,群 GG 的一个子集的生成子群就是包含这个子集的最小的子群。群 GG 的子集 SS 的生成子群记为:

        S⟨S⟩

        两个子群的交总是子群这个命题可推广到任意多个子群的交也总是子群,从而对于群 GG 的某个子集 SS ,可给出包含 SS 的最小子群,这个子群就是 SS生成子群 S⟨S⟩ ,可由所有包含 SS 的子群的交给出,即:

        <S>={HSHHG}\text{<}S\text{>}=\cap\{H|S\subseteq H \land H\leq G\}

        • 如果 SS 为有限集 S={a1,a2,,an}S=\{a_1, a_2,\cdots,a_n\} ,将 S⟨S⟩ 直接记为 a1,a2,,an⟨a_1, a_2,\cdots,a_n ⟩
        • 对于群 GG ,若存在 aGa\in G ,使得 G=aG=⟨a⟩ ,即 GG 是由元素 aa 生成的群,这时称 G 为循环群

        空集生成的子群是什么?

        # 生成子群元素的表示

        GG 是群,SSGG 的非空子集,则:

        S={a1l1a2l2...aklkaiS,li=±1,kN}⟨S⟩=\cap\{a_{1}^{l_{1}}a_{2}^{l_{2}}...a_{k}^{l_{k}}| a_{i}\in S,\ \ \ l_{i}=\pm 1,\ \ \ k\in\mathbb{N} \}

        S⟨S⟩ 的任意元素可表示成有限个 S 的元素或其逆的乘积

        问题16证明

        S={a}S=\{a\} 时,即有 ⟨a⟩=\

        # 循环群

        GG 是群,如果存在 aGa\in G 使得

        G=a={akkZ}G=⟨a⟩=\{a^k | k\in Z\}

        则称 GG循环群 (cyclic group),并称 aaGG 的一个生成元

        • GG 是无限集时称为无限循环群
        • GGnn 个元素时,称为 n 阶循环群

        # 性质 (不考证明)

        GG 是群,ee 是它的单位元

        1. a1=a⟨a^{-1}⟩= ⟨a⟩
        2. GG 是有限群,则 G=aG=aG=⟨a⟩⟺|G|= |a|
        3. GG 是无限循环群,则
          • G={e,a,a1,a2,a2,,an,an,}G = \{e, a, a^{-1}, a^2, a^{-2}, ⋯, a^n, a^{-n}, ⋯\}
          • k,lZ,ak=alk=l\forall k, l\in Z,\ \ \ a^k= a^l⟺k= l
        4. GGnn 阶循环群,则
          • G={e,a,a2,a3,,an1}G = \{e, a, a^2, a^{3}, ⋯, a^{n-1}\}
          • k,lZ,ak=alnkl\forall k, l\in Z,\ \ \ a^k = a^l ⟺n | k-l

        # 循环群举例

        整数加群 Z\mathbb Z 是无限循环群

        显然 Z=1=1\mathbb Z= ⟨1⟩= ⟨-1⟩ , 即 1 和 -1 是它的生成元

        对任意 dZd\in \mathbb Z ,若 d±1d≠±1 ,显然有 1∉d1\not\in⟨d⟩ ,因此 1 和 -1 是 Z\mathbb Z 的仅有的两个生成元。


        mm 加群 Zm\mathbb Z_mmm 阶循环群,显然 Zm=1\mathbb Z_m=⟨1⟩


        U(5)U(5) 是 4 阶循环群,U(5)=2=3U(5)=⟨2⟩=⟨3⟩

        mm 是素数时,U(m)U(m)m1m-1 阶循环群(证明这个需要不少数论知识)

        mm 不是素数时:

        • U(m)U(m) 可能是循环群(例如 U(9)={1,2,4,5,7,8}=2U(9)=\{1,2,4,5,7,8\}=⟨2⟩
        • 也可能不是循环群(例如U(15)={1,2,4,7,8,11,13,14}U(15)=\{1,2,4,7,8,11,13,14\}

        # 生成元的数量与结构

        定理17


        定理17证明

        # 子群保持循环群

        循环群的任意子群都是循环群


        定理18证明

        # 问题

        如果一个群的所有非平凡子群都是循环群,那么这个群一定是循环群吗?

        不一定

        U(12)={1,5,7,11}U(12)=\{1,5,7,11\} 不是循环群,但它的每个非平凡子群都是循环群!

        1=1,5=2,7=2,11=2|1|=1 , |5| = 2 , |7| = 2 , |11| = 2 ,可见 U(12)U(12) 的所有非平凡子群都是二阶群,不难证明所有二阶群都是循环群

        # 生成元任意幂的生成子群

        推论19

        # 循环群的子群⭐️

        推论20

        推论20证明1推论20证明2

        例题:

        Z14\mathbb{Z}_{14} 加群的所有子群是:

        • 1=Z14⟨1⟩=\mathbb{Z}_{14}
        • 2={0,2,4,6,8,10,12,14}⟨2⟩=\{0,2,4,6,8,10,12,14\}
        • 7={0,7}⟨7⟩=\{0,7\}
        • 0=0⟨0⟩={0}

        U(14)U(14) 的所有子群是:

        • 3⟨3⟩
        • 32=9⟨3^2=9⟩
        • 33=13⟨3^3=13⟩
        • 36=1⟨3^6=1⟩
        \ No newline at end of file diff --git a/math/discrete-math/algebra/homomorphism/index.html b/math/discrete-math/algebra/homomorphism/index.html index aac058c0..ce8d259b 100644 --- a/math/discrete-math/algebra/homomorphism/index.html +++ b/math/discrete-math/algebra/homomorphism/index.html @@ -1 +1 @@ -群的同态与同构 - 代数结构 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 群的同态

        # 定义

        GGGG' 是两个群,函数 φ:GGφ:G→G' 如果满足:

        a,bG,φ(ab)=φ(a)φ(b)∀a,b\in G,\ \ φ(ab)= φ(a)φ(b)

        φφ 是群 GGGG'同态 (homomorphism)(一个函数)

        φ(a)φ(b)φ(a)φ(b)GG' 中的运算,而 ababGG 中的运算

        考察同态 φ\varphi 作为函数的性质,可以将同态分为:

        • 如果 G=GG=G' ,则称 φφ 为自同态 (automorphism);
        • 如果同态 φ:GGφ:G→G' 是满函数,则称为满同态 (epimorphism),并称 GGGG' 同态,记为φ:GGφ:G∼G'
        • 如果同态 φ:GGφ:G→G' 是单函数,则称为单同态 (monomorphism);
        • 如果同态 φ:GGφ:G→G' 是双函数,则称为同构 (isomorphism),并称群 GGGG' 同构,记为 φ:GGφ:G≅G'

        # 群同态的单位元

        eeGG 的单位元,则 φ(e)\varphi(e)GG' 的单位元;

        GG' 的单位元 ee' ,则:

        φ(e)φ(e)=φ(ee)=φ(e)=φ(e)e\varphi(e)\varphi(e)=\varphi(ee)=\varphi(e)=\varphi(e)e'

        从而由群的运算满足消去律得 φ(e)=e\varphi(e)=e'

        # 群同态的逆元

        群同态与求逆元操作可交换:

        aG,(φ(a))1=φ(a1)\forall a\in G,\ \ (\varphi(a))^{-1}=\varphi(a^{-1})


        证明, 对任意的 aGa\in G :

        φ(a1)φ(a)=φ(a1a)=φ(e)=eφ(a)φ(a1)=φ(aa1)=φ(e)=e\varphi(a^{-1})\varphi(a)=\varphi(a^{-1}a)=\varphi(e)=e'\\ \varphi(a)\varphi(a^{-1})=\varphi(aa^{-1})=\varphi(e)=e'

        这就表明在群 GG' 中,φ(a)\varphi(a) 的逆元是 φ(a1)\varphi(a^{-1}) ,即 φ(a))1=φ(a1)\varphi(a))^{-1}=\varphi(a^{-1})

        # 群同态例子

        G,GG,G' 是群,ee'GG' 的单位元,函数 φ:GGφ:G→G'aG,φ(a)=e∀a\in G,\ \ φ(a)=e'φ\varphi 是同态,称为 GGGG'零同态


        对于整数加群 (Z,+)(Z,+) 和模 n 加群 (Zn,n)(Z_n,⊕_n) ,函数 φ:ZZnφ:Z→Z_nzZ,φ(z)=zmodn∀z\in Z, φ(z)= z \bmod nφ\varphi满同态

        z1,z2Z,φ(z1+z2)=(z1+z2)modn=(z1modn)n(z2modn)=φ(z1)nφ(z2)\begin{aligned} &∀z_1, z_2\in Z,\\ φ(z_1+z_2 )&=(z_1+z_2 )\bmod n\\ &= (z_1 \bmod n) ⊕_n (z_2 \bmod n)\\ &= φ(z_1 ) ⊕_n φ(z_2 ) \end{aligned}


        对于实数加群 (R,+)(\mathbb R,+) 和非零实数集关于乘法构成的群 (R,)(\mathbb R^*, \cdot ) ,固定某实数 aR(a0,1)a\in \mathbb R(a≠0,1) ,定义 φ:RR,rR,φ(r)=arφ:\mathbb R→\mathbb R^*,\ \ ∀r\in \mathbb R, φ(r)= a^rφ\varphi单同态

        r,rR,φ(r+r)=ar+r=arar=φ(r)φ(r)\begin{aligned} &∀r,r'\in \mathbb R,\\ φ(r+r')&=a^{r+r'}\\ &=a^r\cdot a^{r'}\\ &=φ(r)\cdot φ(r') \end{aligned}

        # 群的同构

        # 内自同构

        内自同构

        # 平移与正则表示

        左平移、左正则表示的定义如下:

        置换群 GlG_l 称为群 GG左正则表示 (left regular representation),左乘置换 φaφ_a 称为由元素 aa 确定的左平移 (left translation)。

        左平移和左正则表示

        当然,类似的可以定义右平移、右正则表示

        # 凯莱定理

        每个群都同构与一个置换群(变换群)

        凯莱定理证明

        # 循环群的结构定理

        1. G=<a>G=<a> 是无限循环群,则 G(Z,+)G\cong(\mathbb Z,+)
        2. G=<a>G=<a>nn 阶循环群,则 G(Zn,n)G\cong(\mathbb Z_n,\oplus_n)

        循环群结构定理证明

        无限循环群中 ak=ala^k=a^l 蕴涵 k=lk=l

        nn 阶循环群中 ak=ala^k=a^l 蕴涵 nkln|k-l

        # 群同态的基本性质

        # 群同态与元素的阶

        群同态与元素的阶

        群同态与子群-证明

        # 群同态与子群

        群同态与子群

        ⭐️特别地, H=GH=G 的时候这里就有:

        φ(G)G\varphi(G)\leq G'


        群同态与子群-证明

        # 群同态与正规子群

        群同态与正规子群

        φ(N)\varphi(N) 不一定是 GG' 的正规子群!φ1(M)\varphi^{-1}(M) 不一定是 GG 的正规子群!

        群同态与正规子群-证明

        # 群的同态基本定理

        # 群同态的核

        # 定义

        群同态的核

        :: primary

        所有同态映射结果为单位元的原像的集合

        :::

        # 核是正规子群

        群同态的核基本性质1


        由上述引理可证:

        φ:GG\varphi:G\to G' 是群 GG 到群 GG' 的同态,则 Ker(φ)\text{Ker}(\varphi)GG正规子群,也就是 Ker(φ)G\text{Ker}(\varphi)\leq G

        # 群的同态基本定理

        群的同态基本定理

        群的同态基本定理-证明

        ::: primaty

        结合群同态与子群的关系,就有:商群 G/Ker(φ)φ(G)=G的某个子群GG/\text{Ker}(\varphi)\cong\varphi(G)=G'\text{的某个子群}\leq G'

        :::

        # 正规子群与商群

        由正规子群导出的商群的自然映射满同态,且它的核是该正规子群,由群的同态基本定理有该正规子群与其导出的商群同构。

        商群与自然映射

        # 应用举例

        正规子群与商群应用举例

        # 第二同构定理

        第二同构定理


        证明

        (1) 利用子群判定定理证明:h1,h2H,k1,k2K,(h1k1)(h2k2)1HK∀h_1,h_2\in H,k_1,k_2\in K,\ \ (h_1k_1)(h_2k_2)^{-1}\in HK

        (2) 利用正规子群判断定理证明:对任意 hH,nHK,hnh1HKh\in H, n\in H\cap K, hnh^{-1}\in H\cap K;以及对 h1H,k1K,nK\forall h_1\in H,\forall k_1\in K,\forall n\in K ,有:(h1k1)n(h1k1)1HK(h_1k_1)n(h_1k_1)^{-1}\in HK

        (3) 构造同态 φ:HHK/Kφ:H→ HK/K ,使得 Ker(φ)=HK\text{Ker}(φ)=H\cap K

        # 第三同构定理

        第三同构定理

        # Zm\mathbb Z_mZn\mathbb Z_n 的同态数

        结论 —— 所有 Zm\mathbb Z_mZn\mathbb Z_n 的同态映射如下:

        {φa:xaxa=0,lcm(m,n)m,2lcm(m,n)m,3lcm(m,n)m,,[gcd(m,n)1]lcm(m,n)m=nlcm(m,n)m}\{\varphi_a:\overline{x}\to a\overline{x}|a=0,\frac{\text{lcm}(m,n)}{m},2\frac{\text{lcm}(m,n)}{m},3\frac{\text{lcm}(m,n)}{m},\cdots,[\gcd(m,n)-1]\frac{\text{lcm}(m,n)}{m}=n-\frac{\text{lcm}(m,n)}{m}\}

        总共有 gcd(m,n)\gcd(m,n) 个同态映射。

        上面的两个 x\overline{x} (xax\overline{x}\to a\overline{x}) ,我们虽然使用了同一个记号,但它们代表的意义是不同的:

        • 前一个表示的是 Zm\mathbb Z_m 中的剩余类 {x+mzzZ}\{x+mz|z\in\mathbb Z\}
        • 后一个表示的是 Zn\mathbb Z_n 中的剩余类 {x+nzzZ}\{x+nz|z\in\mathbb Z\}

        今后,在遇到此类情况时,我们都采用这样的记号,不再一一说明。读者应根据上下文,了解两个 x\overline{x} 的不同含义,以免混淆。

        摘抄自《近世代数》(第二版,韩士安,林磊) 的参考答案。

        \ No newline at end of file +群的同态与同构 - 代数结构 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 群的同态

        # 定义

        GGGG' 是两个群,函数 φ:GGφ:G→G' 如果满足:

        a,bG,φ(ab)=φ(a)φ(b)∀a,b\in G,\ \ φ(ab)= φ(a)φ(b)

        φφ 是群 GGGG'同态 (homomorphism)(一个函数)

        φ(a)φ(b)φ(a)φ(b)GG' 中的运算,而 ababGG 中的运算

        考察同态 φ\varphi 作为函数的性质,可以将同态分为:

        • 如果 G=GG=G' ,则称 φφ 为自同态 (automorphism);
        • 如果同态 φ:GGφ:G→G' 是满函数,则称为满同态 (epimorphism),并称 GGGG' 同态,记为φ:GGφ:G∼G'
        • 如果同态 φ:GGφ:G→G' 是单函数,则称为单同态 (monomorphism);
        • 如果同态 φ:GGφ:G→G' 是双函数,则称为同构 (isomorphism),并称群 GGGG' 同构,记为 φ:GGφ:G≅G'

        # 群同态的单位元

        eeGG 的单位元,则 φ(e)\varphi(e)GG' 的单位元;

        GG' 的单位元 ee' ,则:

        φ(e)φ(e)=φ(ee)=φ(e)=φ(e)e\varphi(e)\varphi(e)=\varphi(ee)=\varphi(e)=\varphi(e)e'

        从而由群的运算满足消去律得 φ(e)=e\varphi(e)=e'

        # 群同态的逆元

        群同态与求逆元操作可交换:

        aG,(φ(a))1=φ(a1)\forall a\in G,\ \ (\varphi(a))^{-1}=\varphi(a^{-1})


        证明, 对任意的 aGa\in G :

        φ(a1)φ(a)=φ(a1a)=φ(e)=eφ(a)φ(a1)=φ(aa1)=φ(e)=e\varphi(a^{-1})\varphi(a)=\varphi(a^{-1}a)=\varphi(e)=e'\\ \varphi(a)\varphi(a^{-1})=\varphi(aa^{-1})=\varphi(e)=e'

        这就表明在群 GG' 中,φ(a)\varphi(a) 的逆元是 φ(a1)\varphi(a^{-1}) ,即 φ(a))1=φ(a1)\varphi(a))^{-1}=\varphi(a^{-1})

        # 群同态例子

        G,GG,G' 是群,ee'GG' 的单位元,函数 φ:GGφ:G→G'aG,φ(a)=e∀a\in G,\ \ φ(a)=e'φ\varphi 是同态,称为 GGGG'零同态


        对于整数加群 (Z,+)(Z,+) 和模 n 加群 (Zn,n)(Z_n,⊕_n) ,函数 φ:ZZnφ:Z→Z_nzZ,φ(z)=zmodn∀z\in Z, φ(z)= z \bmod nφ\varphi满同态

        z1,z2Z,φ(z1+z2)=(z1+z2)modn=(z1modn)n(z2modn)=φ(z1)nφ(z2)\begin{aligned} &∀z_1, z_2\in Z,\\ φ(z_1+z_2 )&=(z_1+z_2 )\bmod n\\ &= (z_1 \bmod n) ⊕_n (z_2 \bmod n)\\ &= φ(z_1 ) ⊕_n φ(z_2 ) \end{aligned}


        对于实数加群 (R,+)(\mathbb R,+) 和非零实数集关于乘法构成的群 (R,)(\mathbb R^*, \cdot ) ,固定某实数 aR(a0,1)a\in \mathbb R(a≠0,1) ,定义 φ:RR,rR,φ(r)=arφ:\mathbb R→\mathbb R^*,\ \ ∀r\in \mathbb R, φ(r)= a^rφ\varphi单同态

        r,rR,φ(r+r)=ar+r=arar=φ(r)φ(r)\begin{aligned} &∀r,r'\in \mathbb R,\\ φ(r+r')&=a^{r+r'}\\ &=a^r\cdot a^{r'}\\ &=φ(r)\cdot φ(r') \end{aligned}

        # 群的同构

        # 内自同构

        内自同构

        # 平移与正则表示

        左平移、左正则表示的定义如下:

        置换群 GlG_l 称为群 GG左正则表示 (left regular representation),左乘置换 φaφ_a 称为由元素 aa 确定的左平移 (left translation)。

        左平移和左正则表示

        当然,类似的可以定义右平移、右正则表示

        # 凯莱定理

        每个群都同构与一个置换群(变换群)

        凯莱定理证明

        # 循环群的结构定理

        1. G=<a>G=<a> 是无限循环群,则 G(Z,+)G\cong(\mathbb Z,+)
        2. G=<a>G=<a>nn 阶循环群,则 G(Zn,n)G\cong(\mathbb Z_n,\oplus_n)

        循环群结构定理证明

        无限循环群中 ak=ala^k=a^l 蕴涵 k=lk=l

        nn 阶循环群中 ak=ala^k=a^l 蕴涵 nkln|k-l

        # 群同态的基本性质

        # 群同态与元素的阶

        群同态与元素的阶

        群同态与子群-证明

        # 群同态与子群

        群同态与子群

        ⭐️特别地, H=GH=G 的时候这里就有:

        φ(G)G\varphi(G)\leq G'


        群同态与子群-证明

        # 群同态与正规子群

        群同态与正规子群

        φ(N)\varphi(N) 不一定是 GG' 的正规子群!φ1(M)\varphi^{-1}(M) 不一定是 GG 的正规子群!

        群同态与正规子群-证明

        # 群的同态基本定理

        # 群同态的核

        # 定义

        群同态的核

        :: primary

        所有同态映射结果为单位元的原像的集合

        :::

        # 核是正规子群

        群同态的核基本性质1


        由上述引理可证:

        φ:GG\varphi:G\to G' 是群 GG 到群 GG' 的同态,则 Ker(φ)\text{Ker}(\varphi)GG正规子群,也就是 Ker(φ)G\text{Ker}(\varphi)\leq G

        # 群的同态基本定理

        群的同态基本定理

        群的同态基本定理-证明

        ::: primaty

        结合群同态与子群的关系,就有:商群 G/Ker(φ)φ(G)=G的某个子群GG/\text{Ker}(\varphi)\cong\varphi(G)=G'\text{的某个子群}\leq G'

        :::

        # 正规子群与商群

        由正规子群导出的商群的自然映射满同态,且它的核是该正规子群,由群的同态基本定理有该正规子群与其导出的商群同构。

        商群与自然映射

        # 应用举例

        正规子群与商群应用举例

        # 第二同构定理

        第二同构定理


        证明

        (1) 利用子群判定定理证明:h1,h2H,k1,k2K,(h1k1)(h2k2)1HK∀h_1,h_2\in H,k_1,k_2\in K,\ \ (h_1k_1)(h_2k_2)^{-1}\in HK

        (2) 利用正规子群判断定理证明:对任意 hH,nHK,hnh1HKh\in H, n\in H\cap K, hnh^{-1}\in H\cap K;以及对 h1H,k1K,nK\forall h_1\in H,\forall k_1\in K,\forall n\in K ,有:(h1k1)n(h1k1)1HK(h_1k_1)n(h_1k_1)^{-1}\in HK

        (3) 构造同态 φ:HHK/Kφ:H→ HK/K ,使得 Ker(φ)=HK\text{Ker}(φ)=H\cap K

        # 第三同构定理

        第三同构定理

        # Zm\mathbb Z_mZn\mathbb Z_n 的同态数

        结论 —— 所有 Zm\mathbb Z_mZn\mathbb Z_n 的同态映射如下:

        {φa:xaxa=0,lcm(m,n)m,2lcm(m,n)m,3lcm(m,n)m,,[gcd(m,n)1]lcm(m,n)m=nlcm(m,n)m}\{\varphi_a:\overline{x}\to a\overline{x}|a=0,\frac{\text{lcm}(m,n)}{m},2\frac{\text{lcm}(m,n)}{m},3\frac{\text{lcm}(m,n)}{m},\cdots,[\gcd(m,n)-1]\frac{\text{lcm}(m,n)}{m}=n-\frac{\text{lcm}(m,n)}{m}\}

        总共有 gcd(m,n)\gcd(m,n) 个同态映射。

        上面的两个 x\overline{x} (xax\overline{x}\to a\overline{x}) ,我们虽然使用了同一个记号,但它们代表的意义是不同的:

        • 前一个表示的是 Zm\mathbb Z_m 中的剩余类 {x+mzzZ}\{x+mz|z\in\mathbb Z\}
        • 后一个表示的是 Zn\mathbb Z_n 中的剩余类 {x+nzzZ}\{x+nz|z\in\mathbb Z\}

        今后,在遇到此类情况时,我们都采用这样的记号,不再一一说明。读者应根据上下文,了解两个 x\overline{x} 的不同含义,以免混淆。

        摘抄自《近世代数》(第二版,韩士安,林磊) 的参考答案。

        \ No newline at end of file diff --git a/math/discrete-math/algebra/normal-subgroup/index.html b/math/discrete-math/algebra/normal-subgroup/index.html index fe84473a..c6d58533 100644 --- a/math/discrete-math/algebra/normal-subgroup/index.html +++ b/math/discrete-math/algebra/normal-subgroup/index.html @@ -1 +1 @@ -正规子群与商群 - 代数结构 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 正规子群

        # 定义

        HH 是群 GG子群 (⚠️得先是子群),如果 aG∀a\in G 都有 Ha=aHHa=aH ,则称 HHGG正规子群 (normal subgroup) 或不变子群 (invariant subgroup),记作:

        HGH\unlhd G

        条件 Ha=aHHa=aH 仅表示两个集合 HaHaaHaH 相等。不能错误地认为,由 Ha=aHHa=aH 可推出对 HH 的任意元素 hhha=ahha=ah

        正确的理解是:对任意 hHh\in H ,存在 hHh^′\in H 使得 ha=ahha = ah'

        GG 的单位元子群 {e}\{e\} 和群 GG 本身都是 GG 的正规子群。这两个正规子群称为 GG平凡正规子群。如果 GG 只有平凡正规子群,且 G{e}G≠\{e\} ,则称 GG单群 (simple group)

        # 交换群的子群

        显然交换群 GG 的任意子群都是 GG 的正规子群

        # 等价定义

        HH 是群 GG 的子群,如果 HH任意一个左陪集也是它的一个右陪集,则 HHGG 的正规子群


        证明:

        对任意 aGa\in GHH 的左陪集 aHaH 也是 HH 的右陪集,即存在 bGb\in G ,使得 aH=HbaH=Hb ;

        同时 aaHa\in aH ,因此 aHba\in Hb

        而又有 aHaa\in Ha ,所以 aHaHba\in Ha\cap Hb

        根据陪集的性质有 Ha=HbHa=Hb ,因此 Ha=Hb=aHHa=Hb=aH ,因此 HHGG 的正规子群。

        # 判定定理

        GG 是群,NNGG 的子群,则下列四个条件等价:

        1. NNGG 的正规子群,即 aG,aN=Na\forall a\in G,\ \ \ aN=Na
        2. NGN\leq GaG,aNa1=N\forall a\in G,\ \ \ aNa^{-1}=N
        3. NGN\leq GaG,aNa1N\forall a\in G,\ \ \ aNa^{-1}\subseteq N
        4. NGN\leq GaG,nN,ana1N\forall a\in G,\forall n\in N,\ \ \ ana^{-1}\in N;⭐️

        这些命题都用集合语言表达,要证明的无非是集合相等或子集关系


        121\rArr 2

        因为 NGN\unlhd G ,所以对任意的 aGa\in G ,有 Na=aNNa=aN ,从而 aNa1=(Na)a1=N(aa1)=NaNa^{-1}=(Na)a^{-1}=N(aa^{-1})=N


        232\rArr 3

        显然 aNa1=NaNa^{-1}=N 蕴含 aNa1NaNa^{-1}\subseteq N


        343\rArr 4

        显然 aNa1=NaNa^{-1}=N 蕴含 aG,nN,ana1N\forall a\in G,\forall n\in N,\ \ \ ana^{-1}\in N


        414\rArr 1

        若对任意 aG,nNa\in G,n\in Nana1Nana^{-1}\in N 从而可得:

        an=(ana1)aNaan=(ana^{-1})a\in Na

        也就是得到 aNNaaN\subseteq Na

        类似的,仍然由对任意 aG,nNa\in G,n\in Nana1Nana^{-1}\in N 可得:

        a1na=(a1)n(a1)1Na^{-1}na=(a^{-1})n(a^{-1})^{-1}\in N

        从而有:

        na=aa1na=a(a1na)aNna=aa^{-1}na=a(a^{-1}na)\in aN

        也就是得到 NaaNNa\subseteq aN

        综合起来就有 Na=aNNa=aN ,即 NNGG 的正规子群。

        # 子集与正规子群

        对于:

        HKGH\subseteq K\subseteq G

        • HHGG 的正规子群,则 HH 也是 KK 的正规子群
        • HHKK 的正规子群,且 KKGG 的正规子群,则推不出 HHGG 的正规子群!也就是正规子群关系 /\unlhd/\unrhd 对子集关系不具有传递性⚠️

        # 子群交保持正规子群

        GG 是群,H,KH,KGG 的正规子群,则 HKH\cap K 也是 GG 的正规子群


        证明

        对于子群的交,首先由子群判定定理,不难得到当 HHKK 是子群时,HKH\cap K 也是子群。

        对任意 aG,hHKa\in G,h\in H\cap K ,由 HHKK 是正规子群有 aha1Haha^{-1}\in Haha1Kaha^{-1}\in K ,从而 aha1HKaha^{-1}\in H\cap K ,因此 HKH\cap K 也是正规子群。

        # 子群乘保持正规子群

        GG 是群,H,KH,KGG 的正规子群,则 HKHK 也是 GG 的正规子群


        对于 HKHK ,首先对任意 h1k1,h2k2HKh_1k_1,h_2k_2\in HK ,有 h1k1(h2k2)1=h1k1k21h21h_1k_1(h_2k_2)^{-1}=h_1k_1k_2^{-1}h_2^{-1}

        由于 KK 是子群,有 k1k21Kk_1k_2^{-1}\in K ,即存在 k1k21=kKk_1k_2^{-1}=k\in K ;

        KK 是正规子群,满足 h1K=Kh1h_1K=Kh_1 ,所以存在 kKk'\in K 使得 h1k=kh1h_1k=k'h_1

        因此:

        h1k1(h2k2)1=h1k1k21h21=h1kh21=kh1h21h_1k_1(h_2k_2)^{-1}=h_1k_1k_2^{-1}h_2^{-1}=h_1kh_2^{-1}=k'h_1h_2^{-1}

        由于 HH 是子群,有 h1h21Hh_1h_2^{-1}\in H ,即存在 h1h21=hHh_1h_2^{-1}=h\in H ;

        KK 是正规子群,满足 hK=KhhK=Kh ,所以存在 kKk''\in K 使得 kh=hkk'h=hk''

        因此:

        h1k1(h2k2)1=kh1h21=kh=hkHKh_1k_1(h_2k_2)^{-1}=k'h_1h_2^{-1}=k'h=hk''\in HK

        所以 HKHKGG 的子群。

        进一步,对任意 aG,hkHKa\in G,\ \ \ hk \in HK ,由 HH 是正规子群,存在 hHh'\in H ,使得 ah=haah= h'a ,从而有 ahka1=haka1ahka^{-1}=h'aka^{-1}

        KK 是正规子群,所以 aka1Kaka^{-1}\in K ,所以 ahka1=haka1HKahka^{-1} =h'aka^{-1}\in HK ,从而 HKHK 也是正规子群。

        # 双陪集正规子群

        名字乱起的,只为了好记

        一个子群只有 2 个陪集,则它一定是正规子群


        证明

        显然这 2 个陪集就是 HHGHG-H

        aG\forall a\in G ,若 aHa\in H ,则显然因为 eHe\in H ,所以 ae=ea=aHae=ea=a\in H ,且 aeaH,eaHaae\in aH,ea\in Ha ,所以此时 aH=H=HaaH=H=Ha

        aHa\notin H ,则因为 eHe\in H ,所以 ae=ea=aHae=ea=a\notin H ,且 aeaH,eaHaae\in aH,ea\in Ha ,所以此时 aHH,HaHaH\neq H,Ha\neq H ,所以 aH=GH=HaaH=G-H=Ha

        # 唯一阶正规子群

        名字乱起的,只为了好记

        HHGG 的子群,则 aG\forall a\in G ,有 aHa1aHa^{-1} 也是 GG 的子群,且 H=aHa1|H|=|aHa^{-1}|

        如果群 GG 的子群 HH 不与 GG 的任意其他子群等势,则 HHGG 的正规子群.


        证明

        h1,h2H,ah1a1,ah2a1aHa1ah1a1(ah2a1)1=a(h1h21)a1HGh1h21Hah1a1(ah2a1)1=a(h1h21)a1aHa1\forall h_1,h_2\in H,\ \ \ ah_1a^{-1},ah_2a^{-1}\in aHa^{-1}\\ ah_1a^{-1}(ah_2a^{-1})^{-1}=a(h_1h_2^{-1})a^{-1}\\ H\leq G\rArr h_1h_2^{-1}\in H\\ \rArr ah_1a^{-1}(ah_2a^{-1})^{-1}=a(h_1h_2^{-1})a^{-1}\in aHa^{-1}

        可知 aHa1aHa^{-1} 也是 GG 的子群。定义映射:

        φ:HaHa1,φ(h)=aha1\varphi:H\to aHa^{-1},\varphi(h)=aha^{-1}

        显然容易证明 φ\varphi 是双函数,所以 H=aHa1|H|=|aHa^{-1}|

        # 正规子群的指标

        GG 是群,HHGG 的正规子群,且 [G:H]=m[G:H]=m ,对任意 gGg\in G ,有 gmHg^m\in H


        证明:

        gG,gmHgG,hH,gm=h\forall g\in G,\ \ g^{m}\in H\lrArr \forall g\in G,\exist h\in H,\ \ g^{m}=h

        gm=hgGH=hgG=hHe=e\begin{aligned} g^{m}&=h\\ g^{\frac{|G|}{|H|}}&=h\\ g^{|G|}&=h^{|H|}\\ e&=e \end{aligned}

        # 商群

        # 定义

        对群 GG 的正规子群 NN ,不必区分左陪集 aNaN 和右陪集 NaNa ,直接称 aNaNNaNa 为陪集。

        G/NG/N 表示它的所有陪集组成的集合,即 G/N={NaaG}G/N=\{Na|a\in G\} ,可在 G/NG/N 上定义运算 使得 G/NG/N 也构成群:

        Na,Nb,NaNb=Nab∀Na,Nb,\ \ Na∘Nb=Nab


        由于运算 是根据陪集的代表(例如陪集 NaNa 的代表是 aa )定义的,因此也要验证运算∘的定义是合适的,即:

        Na=Na,Nb=NbNa=Na', Nb=Nb' 时,需要有 Nab=NabNab = Na'b'

        Nab=(Na)b=(Na)b=(aN)b=a(Nb)=a(Nb)=(aN)b=NabNa'b'=(Na')b'=(Na)b'=(aN)b'=a(Nb')=a(Nb)=(aN)b=Nab


        证明运算 使得 G/NG/N 也构成群:

        满足结合律;对任意的 a,b,cGa,b,c\in G ,有:

        (HaHb)Hc=(Hab)Hc=H(ab)c=Ha(bc)=HaHbc=Ha(HbHc)\begin{aligned} (Ha\circ Hb)\circ Hc &=(Hab)\circ Hc\\ &=H(ab)c\\ &=Ha(bc)\\ &=Ha\circ Hbc\\ &=Ha\circ(Hb\circ Hc) \end{aligned}

        单位元是 He=HHe=H ;对 aG\forall a\in G ,有:

        HHa=HeHa=HHa=HaHe=H(ae)=HaH\circ Ha=He\circ Ha=H\circ Ha=Ha\circ He=H(ae)=Ha

        G/HG/H 的每个元素 HaHa 都有逆元 Ha1GHa^{-1}\in G ;对任意的 HaG/HHa\in G/H ,有 Ha1G/HHa^{-1}\in G/H ,且:

        Ha1Ha=H(a1a)=He=H(aa1)=HaHa1Ha^{-1}\circ Ha=H(a^{-1}a)=He=H(aa^{-1})=Ha\circ Ha^{-1}


        GG 为群,HHGG 的正规子群。HH 的所有陪集 G/HG/H 关于上述运算 构成的群称为群 GG 关于(正规 / 不变)子群 HH商群 (quotient group),仍记作 G/HG/H

        商群就是群依据某个正规子群得到的划分

        # 商群元素的阶

        GG交换群HHGG 的所有阶数有限的元素构成的集合,则 HHGG 的正规子群,且商群 G/HG/H 只有单位元的阶数是有限的


        证明

        由于 GG 是交换群,因此只要证明:

        所有阶数有限的元素构成的集合 H={akZ+,ak=e}H=\{a|\exist k\in \mathbb{Z}^+,\ a^k = e\} ,这里 eeGG 的单位元,只要证明 HHGG 的子群即可。

        对任意 a,bHa,b\in H ,存在 i,jZ+i,j\in \mathbb{Z}^+ ,使得 ai=e,bj=ea^i=e,\ b^j=e ,从而

        (ab1)ij=aij(b1)ij=(ai)j(bj)i=ejei=e(ab^{-1})^{ij}=a^{ij}(b^{-1})^{ij}=(a^{i})^{j}(b^{j})^{-i}=e^{j}e^{-i}=e

        因此 ab1ab^{-1} 也是有限阶元,即 ab1Hab^{-1}\in H ,根据子群判定定理有 HHGG 的子群。

        而对 hH,gG\forall h\in H,\forall g\in G ,易知 ghg1=h|ghg^{-1}|=|h| ,所以 ghg1Hghg^{-1}\in H ,所以 HHGG 的正规子群。

        要证明 G/HG/H 只有单位元的阶数有限,只要证明它的任意元素,如果阶数有限,则这个元素就是单位元。

        对于任意陪集 xHG/HxH\in G/H ,如果它是有限阶的,即存在正整数 kk 使得 (xH)k=xkH=H(xH)^{k}=x^{k}H=H ,从而 xkHx^{k}\in H ,即 xkx^{k} 是有限阶的,从而存在正整数 jj 使得 (xk)j=xkj=e(x^{k})^{j}=x^{kj}=e ,这意味着 xx 也是有限阶的,即 xHx\in H ,从而 xH=HxH=H 是单位元,因此在商群 G/HG/H 中,只有单位元 HH 是有限阶的。

        # 商群保持循环群

        HH 是循环群 GG 的正规子群,则其商群 G/HG/H 也是循环群。


        显然若 aaHH 的生成元,则 aHaH 也是 G/HG/H 的生成元

        # 有限交换群性质

        GGnn有限交换群,则对 nn 的任意因子 ppGG 必有 pp 阶元。


        证明:对 nn 使用数学归纳法。

        (归纳基)显然当 n=2n=2 时,任意 2 阶群都有 2 阶元。

        显然 2 阶群 {e,a}\{e,a\} 一定满足 a2=aa1=ea^2=aa^{-1}=e,因为唯一的非单位元的逆一定是它自己。

        (归纳步)对于 n=k+1n=k+1 ,假设对于任意 2,3,,k2,3,\cdots,k 阶有限交换群,对于群的阶的任意素因子 pp 都有 pp 阶元,考虑 k+1k+1 阶有限交换群 GG ,考虑 ppk+1k+1 的任意素因子。

        为应用归纳假设,需要考虑阶数比 k+1k+1 阶小的有限交换群 GG',而且 GG'GG 有许多相似的性质,这样就需要利用商群的构造:即找到正规子群 HH

        由于 GG 是交换群,所以子群 HH 就是正规子群,使得 G=G/HG'=G/HG/HG/H 的阶是 GG 的阶除以 HH 的阶,只要 HH 不是只有单位元,那么 G/HG/H 的阶比 GG 的阶小。

        对于 GG 的阶 k+1k+1 的素因子 pp

        (1)如果 ppa|a| 的因子,则:

        sZ+,a=spas=agcd(a,s)=as=p\exist s\in\mathbb{Z}^+,\ \ |a|=sp\\ |a^s|=\frac{|a|}{\gcd(|a|,s)}=\frac{|a|}{s}=p

        所以 asa^s 就是 pp 阶元。


        (2)如果 pp 不是 a|a| 的因子,则:

        H=<a>H=<a>aa 的输出子群,H=a|H|=|a|,则因为 GG 是交换群,所以 HH 是正规子群。

        商群 G/HG/H 仍是交换群,且:G/H=k+1a<k+1|G/H|=\frac{k+1}{|a|}\lt k+1

        ppk+1=k+1aak+1=\frac{k+1}{|a|}|a| 的素因子,且 pp 不是 a|a| 的因子,由数论知识得到 pp 就是 k+1a\frac{k+1}{|a|} 的因子;

        所以由归纳假设, bHG/H,bH=p\exist bH\in G/H,\ \ |bH|=p ;所以 (bH)p=bpH=H(bH)^p=b^pH=H ,所以 bpHb^p\in H ;且 H=a|H|=|a| ,所以 (bp)a=(ba)p=e(b^{p})^{|a|}=(b^{|a|})^p=e

        另一方面,由于 pp 不是 a|a| 的因子,所以由 bH=p|bH|=p 可得 baH=(bH)aHb^{|a|}H=(bH)^{|a|}\neq H;所以 baHb^{|a|}\notin H ,所以 baeb^{|a|}\neq e

        所以 bab^{|a|} 就是 pp 阶元

        \ No newline at end of file +正规子群与商群 - 代数结构 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 正规子群

        # 定义

        HH 是群 GG子群 (⚠️得先是子群),如果 aG∀a\in G 都有 Ha=aHHa=aH ,则称 HHGG正规子群 (normal subgroup) 或不变子群 (invariant subgroup),记作:

        HGH\unlhd G

        条件 Ha=aHHa=aH 仅表示两个集合 HaHaaHaH 相等。不能错误地认为,由 Ha=aHHa=aH 可推出对 HH 的任意元素 hhha=ahha=ah

        正确的理解是:对任意 hHh\in H ,存在 hHh^′\in H 使得 ha=ahha = ah'

        GG 的单位元子群 {e}\{e\} 和群 GG 本身都是 GG 的正规子群。这两个正规子群称为 GG平凡正规子群。如果 GG 只有平凡正规子群,且 G{e}G≠\{e\} ,则称 GG单群 (simple group)

        # 交换群的子群

        显然交换群 GG 的任意子群都是 GG 的正规子群

        # 等价定义

        HH 是群 GG 的子群,如果 HH任意一个左陪集也是它的一个右陪集,则 HHGG 的正规子群


        证明:

        对任意 aGa\in GHH 的左陪集 aHaH 也是 HH 的右陪集,即存在 bGb\in G ,使得 aH=HbaH=Hb ;

        同时 aaHa\in aH ,因此 aHba\in Hb

        而又有 aHaa\in Ha ,所以 aHaHba\in Ha\cap Hb

        根据陪集的性质有 Ha=HbHa=Hb ,因此 Ha=Hb=aHHa=Hb=aH ,因此 HHGG 的正规子群。

        # 判定定理

        GG 是群,NNGG 的子群,则下列四个条件等价:

        1. NNGG 的正规子群,即 aG,aN=Na\forall a\in G,\ \ \ aN=Na
        2. NGN\leq GaG,aNa1=N\forall a\in G,\ \ \ aNa^{-1}=N
        3. NGN\leq GaG,aNa1N\forall a\in G,\ \ \ aNa^{-1}\subseteq N
        4. NGN\leq GaG,nN,ana1N\forall a\in G,\forall n\in N,\ \ \ ana^{-1}\in N;⭐️

        这些命题都用集合语言表达,要证明的无非是集合相等或子集关系


        121\rArr 2

        因为 NGN\unlhd G ,所以对任意的 aGa\in G ,有 Na=aNNa=aN ,从而 aNa1=(Na)a1=N(aa1)=NaNa^{-1}=(Na)a^{-1}=N(aa^{-1})=N


        232\rArr 3

        显然 aNa1=NaNa^{-1}=N 蕴含 aNa1NaNa^{-1}\subseteq N


        343\rArr 4

        显然 aNa1=NaNa^{-1}=N 蕴含 aG,nN,ana1N\forall a\in G,\forall n\in N,\ \ \ ana^{-1}\in N


        414\rArr 1

        若对任意 aG,nNa\in G,n\in Nana1Nana^{-1}\in N 从而可得:

        an=(ana1)aNaan=(ana^{-1})a\in Na

        也就是得到 aNNaaN\subseteq Na

        类似的,仍然由对任意 aG,nNa\in G,n\in Nana1Nana^{-1}\in N 可得:

        a1na=(a1)n(a1)1Na^{-1}na=(a^{-1})n(a^{-1})^{-1}\in N

        从而有:

        na=aa1na=a(a1na)aNna=aa^{-1}na=a(a^{-1}na)\in aN

        也就是得到 NaaNNa\subseteq aN

        综合起来就有 Na=aNNa=aN ,即 NNGG 的正规子群。

        # 子集与正规子群

        对于:

        HKGH\subseteq K\subseteq G

        • HHGG 的正规子群,则 HH 也是 KK 的正规子群
        • HHKK 的正规子群,且 KKGG 的正规子群,则推不出 HHGG 的正规子群!也就是正规子群关系 /\unlhd/\unrhd 对子集关系不具有传递性⚠️

        # 子群交保持正规子群

        GG 是群,H,KH,KGG 的正规子群,则 HKH\cap K 也是 GG 的正规子群


        证明

        对于子群的交,首先由子群判定定理,不难得到当 HHKK 是子群时,HKH\cap K 也是子群。

        对任意 aG,hHKa\in G,h\in H\cap K ,由 HHKK 是正规子群有 aha1Haha^{-1}\in Haha1Kaha^{-1}\in K ,从而 aha1HKaha^{-1}\in H\cap K ,因此 HKH\cap K 也是正规子群。

        # 子群乘保持正规子群

        GG 是群,H,KH,KGG 的正规子群,则 HKHK 也是 GG 的正规子群


        对于 HKHK ,首先对任意 h1k1,h2k2HKh_1k_1,h_2k_2\in HK ,有 h1k1(h2k2)1=h1k1k21h21h_1k_1(h_2k_2)^{-1}=h_1k_1k_2^{-1}h_2^{-1}

        由于 KK 是子群,有 k1k21Kk_1k_2^{-1}\in K ,即存在 k1k21=kKk_1k_2^{-1}=k\in K ;

        KK 是正规子群,满足 h1K=Kh1h_1K=Kh_1 ,所以存在 kKk'\in K 使得 h1k=kh1h_1k=k'h_1

        因此:

        h1k1(h2k2)1=h1k1k21h21=h1kh21=kh1h21h_1k_1(h_2k_2)^{-1}=h_1k_1k_2^{-1}h_2^{-1}=h_1kh_2^{-1}=k'h_1h_2^{-1}

        由于 HH 是子群,有 h1h21Hh_1h_2^{-1}\in H ,即存在 h1h21=hHh_1h_2^{-1}=h\in H ;

        KK 是正规子群,满足 hK=KhhK=Kh ,所以存在 kKk''\in K 使得 kh=hkk'h=hk''

        因此:

        h1k1(h2k2)1=kh1h21=kh=hkHKh_1k_1(h_2k_2)^{-1}=k'h_1h_2^{-1}=k'h=hk''\in HK

        所以 HKHKGG 的子群。

        进一步,对任意 aG,hkHKa\in G,\ \ \ hk \in HK ,由 HH 是正规子群,存在 hHh'\in H ,使得 ah=haah= h'a ,从而有 ahka1=haka1ahka^{-1}=h'aka^{-1}

        KK 是正规子群,所以 aka1Kaka^{-1}\in K ,所以 ahka1=haka1HKahka^{-1} =h'aka^{-1}\in HK ,从而 HKHK 也是正规子群。

        # 双陪集正规子群

        名字乱起的,只为了好记

        一个子群只有 2 个陪集,则它一定是正规子群


        证明

        显然这 2 个陪集就是 HHGHG-H

        aG\forall a\in G ,若 aHa\in H ,则显然因为 eHe\in H ,所以 ae=ea=aHae=ea=a\in H ,且 aeaH,eaHaae\in aH,ea\in Ha ,所以此时 aH=H=HaaH=H=Ha

        aHa\notin H ,则因为 eHe\in H ,所以 ae=ea=aHae=ea=a\notin H ,且 aeaH,eaHaae\in aH,ea\in Ha ,所以此时 aHH,HaHaH\neq H,Ha\neq H ,所以 aH=GH=HaaH=G-H=Ha

        # 唯一阶正规子群

        名字乱起的,只为了好记

        HHGG 的子群,则 aG\forall a\in G ,有 aHa1aHa^{-1} 也是 GG 的子群,且 H=aHa1|H|=|aHa^{-1}|

        如果群 GG 的子群 HH 不与 GG 的任意其他子群等势,则 HHGG 的正规子群.


        证明

        h1,h2H,ah1a1,ah2a1aHa1ah1a1(ah2a1)1=a(h1h21)a1HGh1h21Hah1a1(ah2a1)1=a(h1h21)a1aHa1\forall h_1,h_2\in H,\ \ \ ah_1a^{-1},ah_2a^{-1}\in aHa^{-1}\\ ah_1a^{-1}(ah_2a^{-1})^{-1}=a(h_1h_2^{-1})a^{-1}\\ H\leq G\rArr h_1h_2^{-1}\in H\\ \rArr ah_1a^{-1}(ah_2a^{-1})^{-1}=a(h_1h_2^{-1})a^{-1}\in aHa^{-1}

        可知 aHa1aHa^{-1} 也是 GG 的子群。定义映射:

        φ:HaHa1,φ(h)=aha1\varphi:H\to aHa^{-1},\varphi(h)=aha^{-1}

        显然容易证明 φ\varphi 是双函数,所以 H=aHa1|H|=|aHa^{-1}|

        # 正规子群的指标

        GG 是群,HHGG 的正规子群,且 [G:H]=m[G:H]=m ,对任意 gGg\in G ,有 gmHg^m\in H


        证明:

        gG,gmHgG,hH,gm=h\forall g\in G,\ \ g^{m}\in H\lrArr \forall g\in G,\exist h\in H,\ \ g^{m}=h

        gm=hgGH=hgG=hHe=e\begin{aligned} g^{m}&=h\\ g^{\frac{|G|}{|H|}}&=h\\ g^{|G|}&=h^{|H|}\\ e&=e \end{aligned}

        # 商群

        # 定义

        对群 GG 的正规子群 NN ,不必区分左陪集 aNaN 和右陪集 NaNa ,直接称 aNaNNaNa 为陪集。

        G/NG/N 表示它的所有陪集组成的集合,即 G/N={NaaG}G/N=\{Na|a\in G\} ,可在 G/NG/N 上定义运算 使得 G/NG/N 也构成群:

        Na,Nb,NaNb=Nab∀Na,Nb,\ \ Na∘Nb=Nab


        由于运算 是根据陪集的代表(例如陪集 NaNa 的代表是 aa )定义的,因此也要验证运算∘的定义是合适的,即:

        Na=Na,Nb=NbNa=Na', Nb=Nb' 时,需要有 Nab=NabNab = Na'b'

        Nab=(Na)b=(Na)b=(aN)b=a(Nb)=a(Nb)=(aN)b=NabNa'b'=(Na')b'=(Na)b'=(aN)b'=a(Nb')=a(Nb)=(aN)b=Nab


        证明运算 使得 G/NG/N 也构成群:

        满足结合律;对任意的 a,b,cGa,b,c\in G ,有:

        (HaHb)Hc=(Hab)Hc=H(ab)c=Ha(bc)=HaHbc=Ha(HbHc)\begin{aligned} (Ha\circ Hb)\circ Hc &=(Hab)\circ Hc\\ &=H(ab)c\\ &=Ha(bc)\\ &=Ha\circ Hbc\\ &=Ha\circ(Hb\circ Hc) \end{aligned}

        单位元是 He=HHe=H ;对 aG\forall a\in G ,有:

        HHa=HeHa=HHa=HaHe=H(ae)=HaH\circ Ha=He\circ Ha=H\circ Ha=Ha\circ He=H(ae)=Ha

        G/HG/H 的每个元素 HaHa 都有逆元 Ha1GHa^{-1}\in G ;对任意的 HaG/HHa\in G/H ,有 Ha1G/HHa^{-1}\in G/H ,且:

        Ha1Ha=H(a1a)=He=H(aa1)=HaHa1Ha^{-1}\circ Ha=H(a^{-1}a)=He=H(aa^{-1})=Ha\circ Ha^{-1}


        GG 为群,HHGG 的正规子群。HH 的所有陪集 G/HG/H 关于上述运算 构成的群称为群 GG 关于(正规 / 不变)子群 HH商群 (quotient group),仍记作 G/HG/H

        商群就是群依据某个正规子群得到的划分

        # 商群元素的阶

        GG交换群HHGG 的所有阶数有限的元素构成的集合,则 HHGG 的正规子群,且商群 G/HG/H 只有单位元的阶数是有限的


        证明

        由于 GG 是交换群,因此只要证明:

        所有阶数有限的元素构成的集合 H={akZ+,ak=e}H=\{a|\exist k\in \mathbb{Z}^+,\ a^k = e\} ,这里 eeGG 的单位元,只要证明 HHGG 的子群即可。

        对任意 a,bHa,b\in H ,存在 i,jZ+i,j\in \mathbb{Z}^+ ,使得 ai=e,bj=ea^i=e,\ b^j=e ,从而

        (ab1)ij=aij(b1)ij=(ai)j(bj)i=ejei=e(ab^{-1})^{ij}=a^{ij}(b^{-1})^{ij}=(a^{i})^{j}(b^{j})^{-i}=e^{j}e^{-i}=e

        因此 ab1ab^{-1} 也是有限阶元,即 ab1Hab^{-1}\in H ,根据子群判定定理有 HHGG 的子群。

        而对 hH,gG\forall h\in H,\forall g\in G ,易知 ghg1=h|ghg^{-1}|=|h| ,所以 ghg1Hghg^{-1}\in H ,所以 HHGG 的正规子群。

        要证明 G/HG/H 只有单位元的阶数有限,只要证明它的任意元素,如果阶数有限,则这个元素就是单位元。

        对于任意陪集 xHG/HxH\in G/H ,如果它是有限阶的,即存在正整数 kk 使得 (xH)k=xkH=H(xH)^{k}=x^{k}H=H ,从而 xkHx^{k}\in H ,即 xkx^{k} 是有限阶的,从而存在正整数 jj 使得 (xk)j=xkj=e(x^{k})^{j}=x^{kj}=e ,这意味着 xx 也是有限阶的,即 xHx\in H ,从而 xH=HxH=H 是单位元,因此在商群 G/HG/H 中,只有单位元 HH 是有限阶的。

        # 商群保持循环群

        HH 是循环群 GG 的正规子群,则其商群 G/HG/H 也是循环群。


        显然若 aaHH 的生成元,则 aHaH 也是 G/HG/H 的生成元

        # 有限交换群性质

        GGnn有限交换群,则对 nn 的任意因子 ppGG 必有 pp 阶元。


        证明:对 nn 使用数学归纳法。

        (归纳基)显然当 n=2n=2 时,任意 2 阶群都有 2 阶元。

        显然 2 阶群 {e,a}\{e,a\} 一定满足 a2=aa1=ea^2=aa^{-1}=e,因为唯一的非单位元的逆一定是它自己。

        (归纳步)对于 n=k+1n=k+1 ,假设对于任意 2,3,,k2,3,\cdots,k 阶有限交换群,对于群的阶的任意素因子 pp 都有 pp 阶元,考虑 k+1k+1 阶有限交换群 GG ,考虑 ppk+1k+1 的任意素因子。

        为应用归纳假设,需要考虑阶数比 k+1k+1 阶小的有限交换群 GG',而且 GG'GG 有许多相似的性质,这样就需要利用商群的构造:即找到正规子群 HH

        由于 GG 是交换群,所以子群 HH 就是正规子群,使得 G=G/HG'=G/HG/HG/H 的阶是 GG 的阶除以 HH 的阶,只要 HH 不是只有单位元,那么 G/HG/H 的阶比 GG 的阶小。

        对于 GG 的阶 k+1k+1 的素因子 pp

        (1)如果 ppa|a| 的因子,则:

        sZ+,a=spas=agcd(a,s)=as=p\exist s\in\mathbb{Z}^+,\ \ |a|=sp\\ |a^s|=\frac{|a|}{\gcd(|a|,s)}=\frac{|a|}{s}=p

        所以 asa^s 就是 pp 阶元。


        (2)如果 pp 不是 a|a| 的因子,则:

        H=<a>H=<a>aa 的输出子群,H=a|H|=|a|,则因为 GG 是交换群,所以 HH 是正规子群。

        商群 G/HG/H 仍是交换群,且:G/H=k+1a<k+1|G/H|=\frac{k+1}{|a|}\lt k+1

        ppk+1=k+1aak+1=\frac{k+1}{|a|}|a| 的素因子,且 pp 不是 a|a| 的因子,由数论知识得到 pp 就是 k+1a\frac{k+1}{|a|} 的因子;

        所以由归纳假设, bHG/H,bH=p\exist bH\in G/H,\ \ |bH|=p ;所以 (bH)p=bpH=H(bH)^p=b^pH=H ,所以 bpHb^p\in H ;且 H=a|H|=|a| ,所以 (bp)a=(ba)p=e(b^{p})^{|a|}=(b^{|a|})^p=e

        另一方面,由于 pp 不是 a|a| 的因子,所以由 bH=p|bH|=p 可得 baH=(bH)aHb^{|a|}H=(bH)^{|a|}\neq H;所以 baHb^{|a|}\notin H ,所以 baeb^{|a|}\neq e

        所以 bab^{|a|} 就是 pp 阶元

        \ No newline at end of file diff --git a/math/discrete-math/algebra/permutation-group/index.html b/math/discrete-math/algebra/permutation-group/index.html index 52303156..02edfc19 100644 --- a/math/discrete-math/algebra/permutation-group/index.html +++ b/math/discrete-math/algebra/permutation-group/index.html @@ -1 +1 @@ -置换群 - 代数结构 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 置换群的定义

        S={1,2,,n}S=\{1,2,\cdots,n\} ; S 上的任意双函数 σ:SS\sigma:S\to S 称为 S 上的一个 n 元置换 (permutation),记为:

        σ=(12nσ(1)σ(2)σ(n))\sigma= \begin{pmatrix} 1 & 2 & \cdots & n \\ \sigma(1) & \sigma(2) & \cdots & \sigma(n) \end{pmatrix} \quad

        # 置换群的构成

        # 置换乘法

        σ,τ\sigma,\tau 都是 S 上的 n 元置换,则 σ,τ\sigma,\tau 作为函数的复合 στ\sigma\circ\tauτσ\tau\circ\sigma 也是 n 元置换,称为 σ,τ\sigma,\tau乘积,记为 στ\sigma\tauτσ\tau\sigma

        • στ(i)=σ(τ(i))στ(i)=σ(τ(i))
        • τσ(i)=τ(σ(i))τσ(i)=τ(σ(i))

        # 恒等置换

        记为 e ,为 S 上的恒等函数。显然它是置换乘法的单位元

        # 逆置换

        对于某个置换 σ\sigma

        σ=(12nσ(1)σ(2)σ(n))\sigma= \begin{pmatrix} 1 & 2 & \cdots & n \\ \sigma(1) & \sigma(2) & \cdots & \sigma(n) \end{pmatrix} \quad

        σ\sigma​ 关于 (置换) 乘法的逆置换是:

        σ=(σ(1)σ(2)σ(n)12n)\sigma= \begin{pmatrix} \sigma(1) & \sigma(2) & \cdots & \sigma(n)\\ 1 & 2 & \cdots & n \end{pmatrix} \quad

        # 对称群

        SS 上的所有 n 元置换关于置换乘法构成群,这个群称 n 元对称群(symmetric group),并记为 SnS_n , SnS_n 的任意子群都称作 n 元置换群(permutation group)。

        # 轮换

        若 S 上的 n 元置换 σ\sigma 满足:

        σ(i1)=i2,σ(i2)=i3,,σ(ik1)=ik,σ(ik)=i1\sigma(i_1)=i_2,\ \ \ \sigma(i_2)=i_3,\ \cdots\ ,\ \sigma(i_{k-1})=i_k,\ \ \ \sigma(i_k)=i_1

        其中 ijS,j=1,,ki_j\in S,\ \ \ j=1,\cdots,k ;且满足其他元素不变,则称 σ\sigma 为 S 上的 k 阶轮换(cycle),记为:

        σ=(i1i2ik)\sigma=(i_1\ i_2\ \cdots\ i_k)

        特别地有:

        • 2 阶轮换也叫做对换
        • 1 阶轮换 (1)(1) 就是恒等置换

        σ=(i1i2ir)\sigma=(i_1\ i_2\ \cdots\ i_r)τ=(j1j2js)\tau=(j_1\ j_2\ \cdots\ j_s) 是两个轮换,如果对任意的 k=1,2,,rk=1,2,\cdots,r 及任意的 l=1,2,,sl=1,2,\cdots,s 都有 ikjli_k\neq j_l ,称 σ\sigmaτ\tau两个不相交的轮换σ\sigmaτ\tau 不相交

        # 轮换性质 1

        σSnσ\in S_n 是 n 阶置换,证明:σ(i1i2ik)σ1=(σ(i1)σ(i2)σ(ik))σ(i_1\ i_2\ \cdots\ i_k)σ^{-1}= (σ(i_1)\ σ(i_2 )\ \cdots\ σ(i_k ))

        轮换性质1证明

        # 轮换性质 2

        不相交轮换的可交换性

        任意两个不相交轮换的乘积可交换,即若 σ 和 τ 是两个不相交的轮换,则 στ=τσστ=τσ

        轮换性质2eg

        # 轮换性质 3

        置换的轮换表示

        每个置换都可表示为一些不相交轮换的乘积

        轮换性质3eg

        通常省略轮换表示中的一阶轮换,例如这里可以直接记为:σ=(15236)(78)σ=(1\ 5\ 2\ 3\ 6)(7\ 8)

        # 置换的对换表示

        任意置换都可表示成对换的乘积,因为任意轮换都可表示成对换的乘积

        σ=(i1i2ik)\sigma=(i_1\ i_2\ \cdots\ i_k)S={1,2,,n}S=\{1,2,\cdots,n\} 上的 k 阶轮换,则:

        σ=(i1i2)(i2i3)(ik1ik)\sigma=(i_1\ i_2)(i_2\ i_3)\cdots(i_{k-1}\ i_k)

        轮换性质4证明

        • 置换的轮换表示中的轮换是不交的,而对换表示式中的对换是允许有交
        • 轮换表示式在某种意义下(不考虑因子次序和 1 轮换个数)是惟一的,但对换表示式不惟一

        轮换性质4eg1

        轮换性质4eg2

        ⭐️一般地,k 阶轮换 (i1i2i3ik)(i_1\ i_2\ i_3\ \cdots i_k) 也等于 (i1ik)(i1ik1)(i1i2)(i_1\ i_k)(i_1\ i_{k-1} )\cdots(i_1\ i_2)

        # 奇偶性

        一个置换表为对换的乘积,所用的对换个数的奇偶性是惟一的。

        • 可表示成偶数个对换的乘积的置换称为偶置换 (even permutation)

        • 可表示成奇数个对换的乘积的置换称为奇置换 (odd permutation)

        • 任何两个偶置换的积是偶置换;

        • 两个奇置换的积是偶置换;

        • 一个偶置换与一个奇置换的积是奇置换;

        • 一个偶置换的逆置换仍然是偶置换;

        • 一个奇置换的逆仍然是奇置换;

        n>1n>1 时,在全体 n 元置换中,奇置换与偶置换各有 n!2\frac{n!}{2}

        # 交错群

        在 n 元对称群 S_n 中,全体偶置换构成 S_n 的子群称这个子群称为 n 元交错群 (alternative group)。

        交错群eg

        123
        f1f_1123
        f2f_2213
        f3f_3321
        f4f_4132
        f5f_5231
        f6f_6312

        S3S_3 中的偶置换包括 (1)(1)(表示成 0 个对换的乘积)、(123)=(12)(23)(1\ 2\ 3)=(1\ 2)(2\ 3)(132)=(13)(32)(1\ 3\ 2)=(1\ 3)(3\ 2) ,这些偶置换构成了 3 元交错群 A3={(1),(123),(132)}={f1,f5,f6}A_3 = \{(1), (1\ 2\ 3), (1\ 3\ 2)\} = \{f_1, f_5, f_6 \}

        \ No newline at end of file +置换群 - 代数结构 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 置换群的定义

        S={1,2,,n}S=\{1,2,\cdots,n\} ; S 上的任意双函数 σ:SS\sigma:S\to S 称为 S 上的一个 n 元置换 (permutation),记为:

        σ=(12nσ(1)σ(2)σ(n))\sigma= \begin{pmatrix} 1 & 2 & \cdots & n \\ \sigma(1) & \sigma(2) & \cdots & \sigma(n) \end{pmatrix} \quad

        # 置换群的构成

        # 置换乘法

        σ,τ\sigma,\tau 都是 S 上的 n 元置换,则 σ,τ\sigma,\tau 作为函数的复合 στ\sigma\circ\tauτσ\tau\circ\sigma 也是 n 元置换,称为 σ,τ\sigma,\tau乘积,记为 στ\sigma\tauτσ\tau\sigma

        • στ(i)=σ(τ(i))στ(i)=σ(τ(i))
        • τσ(i)=τ(σ(i))τσ(i)=τ(σ(i))

        # 恒等置换

        记为 e ,为 S 上的恒等函数。显然它是置换乘法的单位元

        # 逆置换

        对于某个置换 σ\sigma

        σ=(12nσ(1)σ(2)σ(n))\sigma= \begin{pmatrix} 1 & 2 & \cdots & n \\ \sigma(1) & \sigma(2) & \cdots & \sigma(n) \end{pmatrix} \quad

        σ\sigma​ 关于 (置换) 乘法的逆置换是:

        σ=(σ(1)σ(2)σ(n)12n)\sigma= \begin{pmatrix} \sigma(1) & \sigma(2) & \cdots & \sigma(n)\\ 1 & 2 & \cdots & n \end{pmatrix} \quad

        # 对称群

        SS 上的所有 n 元置换关于置换乘法构成群,这个群称 n 元对称群(symmetric group),并记为 SnS_n , SnS_n 的任意子群都称作 n 元置换群(permutation group)。

        # 轮换

        若 S 上的 n 元置换 σ\sigma 满足:

        σ(i1)=i2,σ(i2)=i3,,σ(ik1)=ik,σ(ik)=i1\sigma(i_1)=i_2,\ \ \ \sigma(i_2)=i_3,\ \cdots\ ,\ \sigma(i_{k-1})=i_k,\ \ \ \sigma(i_k)=i_1

        其中 ijS,j=1,,ki_j\in S,\ \ \ j=1,\cdots,k ;且满足其他元素不变,则称 σ\sigma 为 S 上的 k 阶轮换(cycle),记为:

        σ=(i1i2ik)\sigma=(i_1\ i_2\ \cdots\ i_k)

        特别地有:

        • 2 阶轮换也叫做对换
        • 1 阶轮换 (1)(1) 就是恒等置换

        σ=(i1i2ir)\sigma=(i_1\ i_2\ \cdots\ i_r)τ=(j1j2js)\tau=(j_1\ j_2\ \cdots\ j_s) 是两个轮换,如果对任意的 k=1,2,,rk=1,2,\cdots,r 及任意的 l=1,2,,sl=1,2,\cdots,s 都有 ikjli_k\neq j_l ,称 σ\sigmaτ\tau两个不相交的轮换σ\sigmaτ\tau 不相交

        # 轮换性质 1

        σSnσ\in S_n 是 n 阶置换,证明:σ(i1i2ik)σ1=(σ(i1)σ(i2)σ(ik))σ(i_1\ i_2\ \cdots\ i_k)σ^{-1}= (σ(i_1)\ σ(i_2 )\ \cdots\ σ(i_k ))

        轮换性质1证明

        # 轮换性质 2

        不相交轮换的可交换性

        任意两个不相交轮换的乘积可交换,即若 σ 和 τ 是两个不相交的轮换,则 στ=τσστ=τσ

        轮换性质2eg

        # 轮换性质 3

        置换的轮换表示

        每个置换都可表示为一些不相交轮换的乘积

        轮换性质3eg

        通常省略轮换表示中的一阶轮换,例如这里可以直接记为:σ=(15236)(78)σ=(1\ 5\ 2\ 3\ 6)(7\ 8)

        # 置换的对换表示

        任意置换都可表示成对换的乘积,因为任意轮换都可表示成对换的乘积

        σ=(i1i2ik)\sigma=(i_1\ i_2\ \cdots\ i_k)S={1,2,,n}S=\{1,2,\cdots,n\} 上的 k 阶轮换,则:

        σ=(i1i2)(i2i3)(ik1ik)\sigma=(i_1\ i_2)(i_2\ i_3)\cdots(i_{k-1}\ i_k)

        轮换性质4证明

        • 置换的轮换表示中的轮换是不交的,而对换表示式中的对换是允许有交
        • 轮换表示式在某种意义下(不考虑因子次序和 1 轮换个数)是惟一的,但对换表示式不惟一

        轮换性质4eg1

        轮换性质4eg2

        ⭐️一般地,k 阶轮换 (i1i2i3ik)(i_1\ i_2\ i_3\ \cdots i_k) 也等于 (i1ik)(i1ik1)(i1i2)(i_1\ i_k)(i_1\ i_{k-1} )\cdots(i_1\ i_2)

        # 奇偶性

        一个置换表为对换的乘积,所用的对换个数的奇偶性是惟一的。

        • 可表示成偶数个对换的乘积的置换称为偶置换 (even permutation)

        • 可表示成奇数个对换的乘积的置换称为奇置换 (odd permutation)

        • 任何两个偶置换的积是偶置换;

        • 两个奇置换的积是偶置换;

        • 一个偶置换与一个奇置换的积是奇置换;

        • 一个偶置换的逆置换仍然是偶置换;

        • 一个奇置换的逆仍然是奇置换;

        n>1n>1 时,在全体 n 元置换中,奇置换与偶置换各有 n!2\frac{n!}{2}

        # 交错群

        在 n 元对称群 S_n 中,全体偶置换构成 S_n 的子群称这个子群称为 n 元交错群 (alternative group)。

        交错群eg

        123
        f1f_1123
        f2f_2213
        f3f_3321
        f4f_4132
        f5f_5231
        f6f_6312

        S3S_3 中的偶置换包括 (1)(1)(表示成 0 个对换的乘积)、(123)=(12)(23)(1\ 2\ 3)=(1\ 2)(2\ 3)(132)=(13)(32)(1\ 3\ 2)=(1\ 3)(3\ 2) ,这些偶置换构成了 3 元交错群 A3={(1),(123),(132)}={f1,f5,f6}A_3 = \{(1), (1\ 2\ 3), (1\ 3\ 2)\} = \{f_1, f_5, f_6 \}

        \ No newline at end of file diff --git a/math/discrete-math/algebra/prime-max-ideal/index.html b/math/discrete-math/algebra/prime-max-ideal/index.html index ac69f6cb..6584ccfa 100644 --- a/math/discrete-math/algebra/prime-max-ideal/index.html +++ b/math/discrete-math/algebra/prime-max-ideal/index.html @@ -1 +1 @@ -素理想与极大理想 - 代数结构 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 素理想

        非交换环上也可以定义素理想、极大理想,但这里只考虑交换环上的素理想和极大理想!

        # 定义

        RR 是交换环,PPRR 的真理想。若对任意 a,bRa,b\in RabPab\in P 蕴涵 aPa\in PbPb\in P ,则称 PPRR素理想 (prime ideal)。

        素理想的概念 b 本质上来自数论中的基本结论:若 pp 为素数,则 p|ab\Rightarrow p|a\or p|b。

        # 整数环的素理想

        nn 是正整数,n\langle n\rangle 是整数环 Z\mathbb Z 的素理想当且仅当 nn 是素数。

        零理想 I={0}I=\{0\} 是整数环 Z\mathbb Z 的素理想,但是 Z\mathbb Z 不是素理想,因为素理想要求是真理想!

        # 素理想的充要条件

        RR 是有单位元 e0e≠0 的交换环,IIRR 的理想,则 IIRR 的素理想当且仅当 R/IR/I整环

        # 极大理想

        # 定义

        RR 是交换环,MMRR 的真理想。若对 RR 的任意包含 MM 的理想 NN ,必有 N=MN=MN=RN=R,则称 MMRR极大理想 (maximal ideal)。

        极大理想就是没有真包含它的非平凡理想

        # 极大理想的证明

        证明 IIRR 的极大理想的思路:(RR 是有单位元 ee 的交换环)

        假设 JJ 是理想且 IJI\sub J,...... 得到 eJe\in J,所以 J=RJ=R,所以 IIRR 的极大理想。

        # 整数环的极大理想

        pp 是正整数,p\langle p\rangle 是整数环 Z\mathbb Z 的极大理想当且仅当 pp 是素数,和素理想一致!

        # 极大理想的充要条件

        RR 是有单位元 ee 的交换环,IIRR 的理想,则:IIRR 的极大理想当且仅当 R/IR/I


        • 在交换环 RR 中,IIRR 的素理想当且仅当在 R/IR/Iaˉbˉ=0ˉ\bar{a}\bar{b}=\bar0 蕴涵 aˉ=0ˉ\bar a=\bar 0bˉ=0ˉ\bar b=\bar 0,当且仅当 R/IR/I 的每个元素不是零因子
        • 在有单位元交换环 RR 中,IIRR 的极大理想当且仅当每个真包含 II 的理想都包含 RR 的单位元(从而就包含 RR 的所有元素)

        # 推论

        RR 是有单位元的交换环,由于域都是整环,因此 RR 的每个极大理想都是 RR 的素理想。

        • 如果没有单位元,则极大理想不一定是素理想
          • 例如对于 R=2ZR=2\mathbb ZI=4ZI=4\mathbb ZIIRR 的极大理想,但 II 不是 RR 的素理想
        • 一个素理想(即使是非零素理想)也不一定是极大理想

        # 多项式环的极大理想

        证明 Zm[x]\mathbb Z_m[x] 的理想 ax2+bx+c\langle ax^2+bx+c\rangle (这里以二阶多项式为例)是极大理想思路:

        考察商环 Zm[x]/ax2+bx+c\mathbb Z_m[x]/\langle ax^2+bx+c\rangle 是否是,可以看其所有非零元素是否构成循环群,若构成循环群,则每个非零元素都可逆,则商环是域。

        # 高斯整环的极大理想

        以下结论包括了高斯整环中的所有极大理想:

        • 高斯整环 Z[i]\mathbb Z[i] 的主理想 a+bi\langle a+bi\rangle 是极大理想当且仅当 a2+b2a^2+b^2 是素数,且高斯整环 Z[i]\mathbb Z[i] 关于极大理想 a+bi\langle a+bi\rangle 的商环 Z[i]/a+bi\mathbb Z[i]/\langle a+bi\rangle 和模 a2+b2a^2+b^2 剩余类环 Za2+b2\mathbb Z_{a^2+b^2} 同构。

        a2+b2a^2+b^2 是素数,则一定有 a2+b23mod4a^2+b^2\equiv3\bmod4

        • 高斯整环 Z[i]\mathbb Z[i] 的主理想 p\langle p\rangle 是极大理想当且仅当 pp 是素数并且 p3mod4p\equiv3\bmod4,和整数环中一致。且高斯整环 Z[i]\mathbb Z[i] 关于极大理想 p\langle p\rangle 的商环 Z[i]/p\mathbb Z[i]/\langle p\rangle 和模 p2p^2 剩余类环 Zp2\mathbb Z_{p^2} 同构。

        • 高斯整环中,1+i\langle1+i\rangle1i\langle1-i\rangle 是极大理想。

        \ No newline at end of file +素理想与极大理想 - 代数结构 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 素理想

        非交换环上也可以定义素理想、极大理想,但这里只考虑交换环上的素理想和极大理想!

        # 定义

        RR 是交换环,PPRR 的真理想。若对任意 a,bRa,b\in RabPab\in P 蕴涵 aPa\in PbPb\in P ,则称 PPRR素理想 (prime ideal)。

        素理想的概念 b 本质上来自数论中的基本结论:若 pp 为素数,则 p|ab\Rightarrow p|a\or p|b。

        # 整数环的素理想

        nn 是正整数,n\langle n\rangle 是整数环 Z\mathbb Z 的素理想当且仅当 nn 是素数。

        零理想 I={0}I=\{0\} 是整数环 Z\mathbb Z 的素理想,但是 Z\mathbb Z 不是素理想,因为素理想要求是真理想!

        # 素理想的充要条件

        RR 是有单位元 e0e≠0 的交换环,IIRR 的理想,则 IIRR 的素理想当且仅当 R/IR/I整环

        # 极大理想

        # 定义

        RR 是交换环,MMRR 的真理想。若对 RR 的任意包含 MM 的理想 NN ,必有 N=MN=MN=RN=R,则称 MMRR极大理想 (maximal ideal)。

        极大理想就是没有真包含它的非平凡理想

        # 极大理想的证明

        证明 IIRR 的极大理想的思路:(RR 是有单位元 ee 的交换环)

        假设 JJ 是理想且 IJI\sub J,...... 得到 eJe\in J,所以 J=RJ=R,所以 IIRR 的极大理想。

        # 整数环的极大理想

        pp 是正整数,p\langle p\rangle 是整数环 Z\mathbb Z 的极大理想当且仅当 pp 是素数,和素理想一致!

        # 极大理想的充要条件

        RR 是有单位元 ee 的交换环,IIRR 的理想,则:IIRR 的极大理想当且仅当 R/IR/I


        • 在交换环 RR 中,IIRR 的素理想当且仅当在 R/IR/Iaˉbˉ=0ˉ\bar{a}\bar{b}=\bar0 蕴涵 aˉ=0ˉ\bar a=\bar 0bˉ=0ˉ\bar b=\bar 0,当且仅当 R/IR/I 的每个元素不是零因子
        • 在有单位元交换环 RR 中,IIRR 的极大理想当且仅当每个真包含 II 的理想都包含 RR 的单位元(从而就包含 RR 的所有元素)

        # 推论

        RR 是有单位元的交换环,由于域都是整环,因此 RR 的每个极大理想都是 RR 的素理想。

        • 如果没有单位元,则极大理想不一定是素理想
          • 例如对于 R=2ZR=2\mathbb ZI=4ZI=4\mathbb ZIIRR 的极大理想,但 II 不是 RR 的素理想
        • 一个素理想(即使是非零素理想)也不一定是极大理想

        # 多项式环的极大理想

        证明 Zm[x]\mathbb Z_m[x] 的理想 ax2+bx+c\langle ax^2+bx+c\rangle (这里以二阶多项式为例)是极大理想思路:

        考察商环 Zm[x]/ax2+bx+c\mathbb Z_m[x]/\langle ax^2+bx+c\rangle 是否是,可以看其所有非零元素是否构成循环群,若构成循环群,则每个非零元素都可逆,则商环是域。

        # 高斯整环的极大理想

        以下结论包括了高斯整环中的所有极大理想:

        • 高斯整环 Z[i]\mathbb Z[i] 的主理想 a+bi\langle a+bi\rangle 是极大理想当且仅当 a2+b2a^2+b^2 是素数,且高斯整环 Z[i]\mathbb Z[i] 关于极大理想 a+bi\langle a+bi\rangle 的商环 Z[i]/a+bi\mathbb Z[i]/\langle a+bi\rangle 和模 a2+b2a^2+b^2 剩余类环 Za2+b2\mathbb Z_{a^2+b^2} 同构。

        a2+b2a^2+b^2 是素数,则一定有 a2+b23mod4a^2+b^2\equiv3\bmod4

        • 高斯整环 Z[i]\mathbb Z[i] 的主理想 p\langle p\rangle 是极大理想当且仅当 pp 是素数并且 p3mod4p\equiv3\bmod4,和整数环中一致。且高斯整环 Z[i]\mathbb Z[i] 关于极大理想 p\langle p\rangle 的商环 Z[i]/p\mathbb Z[i]/\langle p\rangle 和模 p2p^2 剩余类环 Zp2\mathbb Z_{p^2} 同构。

        • 高斯整环中,1+i\langle1+i\rangle1i\langle1-i\rangle 是极大理想。

        \ No newline at end of file diff --git a/math/discrete-math/algebra/ring/index.html b/math/discrete-math/algebra/ring/index.html index 8543dcf5..83d6f146 100644 --- a/math/discrete-math/algebra/ring/index.html +++ b/math/discrete-math/algebra/ring/index.html @@ -1 +1 @@ -环 - 代数结构 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 环的相关定义

        RR 是非空集合,如果在 RR 上定义了两个二元运算 “++”(称为加法)和 “ \cdot ”(称为乘法),且满足:

        • (1) RR 关于加法 ++ 构成交换群(即加群)
        • (2) RR 关于乘法 \cdot 构成半群,即乘法满足结合律:a,b,cR,a(bc)=(ab)c\forall a,b,c\in R,\ \ a\cdot(b\cdot c)=(a\cdot b)\cdot c
        • (3) 乘法 \cdot 对加法有左右分配律

        a,b,cRa(b+c)=ab+ac(b+c)a=ba+ca\forall a,b,c\in R\\ a\cdot(b+c)=a\cdot b+a\cdot c\\ (b+c)\cdot a=b\cdot a+c\cdot a

        则称代数 (R,+,)(R,+,\ \cdot\ ) 是环(ring)。在通过上下文能明确运算时,通常直接称 RR 为环。

        有教材要求环乘法必须有单位元,也有教材要求环乘法必须满足交换律等

        对环 (R,+,)(R,+,\ \cdot\ )(R,+)(R\ ,+) 是交换群,称为环 RR 的加法群,其单位元通常用 0 表示,称为环 RR 的零元,元素 aa 的加法逆元记为 a-a ,称为 aa 的负元。

        通过负元引入环的减法,即 aba-b 定义为 a+(b)a+(-b)

        • 环乘法是否满足交换律
          • 若环 RR 的乘法也满足交换律,则称为交换环 (commutative ring)
        • 是否有单位元
          • 若环 RR 的乘法有单位元,则称为有单位元环,环乘法的单位元通常记为 ee11 ,并称为环 RR 的单位元
        • 环乘法当环 RR 有单位元 ee 时,考虑环的元素是否有逆元
          • 若对 aRa\in R ,存在 bRb\in R ,使得 ab=ba=eab=ba=e ,则称 aa 是可逆元,或称为环 RR单位 (unit),并称 bbaa 的逆元。当然这时 bb 也是可逆元,且 aabb 的逆元
          • 所有可逆元关于环乘法构成群,称为 RR单位群 (group of units),记为 U(R)U(R)
        • 环乘法是否有零因子
          • 对环 RR 的两个非零元素 a,ba,b ,若 ab=0a\cdot b=0 ,则称 aa左零因子 (left zero-divisor),b 是右零因子 (right zero-divisor)。左零因子和右零因子统称为零因子
          • 若环 RR 的所有非零元素都不是左零因子或右零因子,则称 RR 为无零因子环。

        当集合 RR 只有一个元素 00 ,定义 RR 的加法运算和乘法运算:0+0=0,00=00+0=0,\ 0\cdot 0=0,则 ({0},+,)(\{0\},+,\ \cdot\ ) 构成环,这个环称为零环。在这个环中,零元也是单位元,而零元也是可逆元

        零环过于简单,通常在对环进行讨论时都将零环排除在外。因此在提到环时,总是默认环至少有两个元素,这时零元不可能是单位元,也不可能是可逆元。

        # 整环

        • 有单位元
        • 无零因子
        • (乘法) 可交换

        的环称为整环 (integral domain)


        证明一个环是整环:

        \forall a,b\in R,\ \ ab=0\Rightarrow a=0\or b=0

        # 除环

        • 有单位元
        • 至少有两个元素
        • 每个非零元都可逆 (乘法)
        • (无零因子,可逆元都不是零因子)

        的环称为除环 (division ring)

        #

        交换除环称为 (field) (比较常用),也即需满足:

        • 有单位元
        • 至少有两个元素
        • 每个非零元都可逆 (乘法)
        • (乘法) 可交换
        • (无零因子,可逆元都不是零因子)

        可逆元都不是零因子(为什么?),因此域都是整环!

        非交换除环也称 (skew field) (比较少用)

        # 常用环

        # 整数环及其子环

        • 整数集 Z\mathbb Z 关于普通加法 ++ 和普通乘法 \cdot 构成环,称为整数环

          • 整数环 Z\mathbb Z交换环,(乘法) 零元是 0,单位元是 1

          • 只有 1 和 -1 是可逆元,因此整数环的单位群是 ({1,1},)(\{1, -1\},\ \cdot\ )

          • 整数环的每个非零整数都不是零因子,因此整数环是整环

        • 固定整数 dd ,集合 dZ={kdkZ}d\mathbb Z=\{kd|k\in \mathbb Z\} 关于普通加法 ++ 和普通乘法 \cdot 也构成交换环

          • d1d≠1 时,环 dZd\mathbb Z 没有单位元。显然环 dZd\mathbb Z 也是无零因子环

          • 具体来说,所有偶数构成的集合 2Z2\mathbb Z 关于普通加法和普通乘法构成无单位元、无零因子的交换环。

        # 模 m 剩余类环及其子环

        固定整数 m2m≥2 ,模 mm 剩余类 Zm={0,1,,m1}\mathbb Z_m=\{\overline{0},\overline{1},\cdots,\overline{m-1}\} 关于模 mmm⊕_m 和模 mmm⊗_m 构成环,称为模 mm 剩余类环 (residue class ring)

        • 0\overline 0 表示整除 mm 余 0 的所有整数构成的集合,Zm\mathbb Z_m 的元素是集合

        • mm 剩余类环 Z\mathbb Z 是有单位元交换环,零元是 0,有单位元 1

        • 对于 aZma\in \mathbb Z_m ,如果 aamm 互质,则 aa 关于模 mm 乘有逆元,因此单位群是 U(m)U(m)

        • mm 是质数 pp 时,Zp={0}U(p)\mathbb Z_p=\{0\}\cup U(p) ,关于 p⊕_pp⊗_p 构成有单位元、每个非零元都可逆的交换环,也即这时 Zp\mathbb Z_p 是域

        • mm 不是质数时,若 m=kd(2km,2km)m=kd(2≤k≤m,2≤k≤m) ,则 kkdd 都是模 mm 剩余类环的零因子,这时就不是无零因子环,当然也不是整环

        # 有理数域

        有理数集 QQ 关于普通加法 ++ 和普通乘法 \cdot 构成环

        • 有单位元 1,是交换环,而且每个非零有理数 rr 都有逆 1/r1/r

        • 单位群是所有非零有理数集 Q\mathbb Q^* 关于普通乘法构成的群 (Q,)(\mathbb Q^*,\ \cdot\ )

        • 因此通常直接称有理数集 Q\mathbb Q 为有理数域

        实数集 R\mathbb R 和复数集 C\mathbb C 关于普通加法 ++ 和普通乘法 \cdot 也都构成域,分别称为实数域 R\mathbb R 和复数域 C\mathbb C

        • 单位群分别是所有非零实数集 R\mathbb R^* 和非零复数集 C\mathbb C^* 关于普通乘法构成的群

        • 复数域 C\mathbb C 的零元是实数 0,单位元是实数 1,而复数乘法的逆为:(a+bi)1=abia2+b2(a+bi)^{-1}=\frac{a-bi}{a^2+b^2}

        # 高斯整环

        高斯整环

        # 类高斯整环

        看起来和高斯整环很像,我乱起的名字,不是很重要

        类高斯整环1

        类高斯整环2

        # 全矩阵环

        全矩阵环

        # 多项式环

        多项式环1多项式环2

        例子

        多项式环举例

        # 理想

        # 定义

        RR 是环,IIRR 的非空子集。若 I 满足:

        1. r1,r2I\forall r_1,r_2\in I,有 r1r2Ir_1-r_2\in I
        2. rI,sR,rs,srI\forall r\in I,\ \forall s\in R,\ \ rs,sr\in I

        则称 II 为环 RR理想 (ideal),记为 IRI⊲R,若 IIRR 的真子集,则称 IIRR真理想 (proper ideal)。

        理想是子环:显然如果 IIRR 的理想,则 II 必定是 RR 的子环;

        ⚠️ 当然子环不一定是理想 ⚠️

        RR 的单个零元构成的集合 {0}\{0\}(称为零理想)和 RR 本身都是 RR 的理想,这两个理想称为 RR平凡理想。因此 RR 的非平凡理想就是非零真理想

        # 常用理想

        整数环 ZZ 的所有理想是 dZ={dzzZ},d=0,1,dZ=\{dz|z\in Z\},\ d=0,1,\cdots

        mm 剩余类环 ZmZ_m 的所有理想是 dZm={dzzZm},d=0,1,,m1dZ_m=\{dz|z\in Z_m\},\ d=0, 1, \cdots, m-1


        证明

        理想是子环,而 ZZ 的每个子环都具有形式 dZdZ ,而且对任意子环 dZdZ ,对任意 dzdZdz\in dZ ,以及 sZs\in Z ,显然有 dzs,sdzdZdzs, sdz\in dZ ,因此每个子环 dZdZ 都是 ZZ 的理想

        类似可得到模 mm 剩余类环 ZmZ_m 的所有理想是 dZm,d=0,1,,m1dZ_m, d=0, 1, \cdots, m-1

        # 运算

        #

        RR 是环,I,JI,J 都是 RR 的理想,I+JI+J 称为理想 IIJJ

        I+J={a+bRaI,bJ}I+J=\{a+b\in R|a\in I, b\in J\}

        #

        IJI\cap J 称为 IIJJ

        IJ={xRxI,xJ}I\cap J=\{x\in R|x\in I,x\in J\}

        #

        IJIJ 称为理想 IIJJ

        IJ={a1b1+a2b2++anbnnN,akI,bkJ}IJ=\{a_1b_1+a_2b_2+\cdots+a_nb_n|n\in\mathbb N^*,a_k\in I,b_k\in J\}

        # 和交积保持理想

        IIJJ 都是环 RR 的理想,则 I+JI+JIJI\cap JIJIJ 也是环 RR 的理想

        环的任意有限多个理想的和仍是理想,而任意有限或无限多个理想的交仍是理想

        # 主理想

        # 定义

        RR 是环,aaRR 的元素,记所有包含 aa 的理想构成的集合为:

        ={IRaI}\sum=\{I\lhd R|a\in I\}

        至少有 RR\in\sum ,所以 \sum 非空。令:

        a=II\langle a\rangle=\bigcap_{I\in\sum}I

        a\langle a\rangle 是理想,而且是包含 aa 的最小理想。这个理想称由 aa 生成的主理想(principal ideal),aa 为其生成元

        # 定理结论

        RR 的由 aa 生成的主理想满足:

        a={(i=1nxiayi)+xa+ay+maxi,yi,x,yR,nZ+,mZ}\langle a\rangle=\big\{ \big(\sum_{i=1}^{n}x_iay_i\big) +xa+ay+ma\big|x_i,y_i,x,y\in R,n\in\mathbb Z^+,m\in\mathbb Z \big\}

        RR 有单位元,则:

        \langle a\rangle=\big\{ \big(\sum_{i=1}^{n}x_iay_i\big) \big|x_i,y_i\in R \big\}\

        RR交换环,则:

        a={xa+maxR,mZ}\langle a\rangle=\big\{ xa+ma\big|x\in R,m\in\mathbb Z \big\}

        ⭐️若 RR有单位元的交换环,则:

        a=aR={arrR}\langle a\rangle=aR=\big\{ ar\big|r\in R \big\}

        # 常用理想的主理想

        • 整数环 Z\mathbb Z 的每个理想 dZd\mathbb Z 都是主理想,即是 d\langle d\rangle
        • mm 剩余类环 Zm\mathbb Z_m 的每个理想 dZmd\mathbb Z_m 也都是主理想,即也是 d\langle d\rangle

        RR 是环,a1,a2,,anRa_1,a_2,\cdots,a_n\in R,则:

        a1,a2,,an=a1+a2++an\langle a_1,a_2,\cdots,a_n\rangle=\langle a_1\rangle+\langle a_2\rangle+\cdots+\langle a_n\rangle

        也是 RR 的理想,并且是包含 a1,a2,,ana_1,a_2,\cdots,a_n 的最小理想!称为由 a1,a2,,ana_1,a_2,\cdots,a_n 生成的理想。

        整数环 Z\mathbb Za1,a2a_1,a_2 生成的理想是gcd(a1,a2)\langle\gcd(a_1,a_2)\rangle

        # 商环

        环关于理想的商环

        # 定义

        RR 是环,II(R,+,)(R,+,\ \cdot\ ) 的一个理想,则 (I,+)(I,+)(R,+)(R,+) 的正规子群,它所有(关于加群)的陪集构成集合 R/I{x+IxR}R/I=\{x+I|x\in R\} 、记陪集 x+Ix+Ix\overline x

        # 运算的定义

        # 加法

        在集合 R/IR/I 上可定义加法 ++

        x,yR/I,x+y=x+y(x+I)+(y+I)=(x+y)+I\forall \overline x,\overline y\in R/I,\\ \overline{x}+\overline{y}=\overline{x+y}\Longleftrightarrow (x+I)+(y+I)=(x+y)+I

        这里 x+y\overline{x}+\overline{y} 中的 ++ 是要定义的加法,即后面所说商环中的加法,而 x+y\overline{x+y} 中的 ++ 是环 RR 中的加法。

        根据正规子群的性质,该定义是合适的。

        # 乘法

        在集合 R/IR/I 上可定义乘法 ×\times

        x,yR/I,xy=xy(x+I)(y+I)=(xy)+I\forall \overline x,\overline y\in R/I,\\ \overline{x}\cdot\overline{y}=\overline{x\cdot y}\Longleftrightarrow (x+I)\cdot(y+I)=(x\cdot y)+I

        这里 xy\overline{x}\cdot\overline{y} 中的 $\cdot $ 是要定义的乘法,即后面所说商环中的乘法,而 xy\overline{x\cdot y} 中的 $\cdot $ 是环 RR 中的乘法。


        由于这定义在等价类(陪集)上,因此需要证明该定义是合适的,即与选择的代表无关,即:

        x1=x2y1=y2x1y1=x2y2\overline{x_1}=\overline{x_2}\ \land\overline{y_1}=\overline{y_2}\\ \Downarrow\\ \overline{x_1\cdot y_1}=\overline{x_2\cdot y_2}

        商环乘法合适定义证明

        # 构成环

        商环构成环证明1

        商环构成环证明2

        # 商环定义

        RR 是环,II 是理想,则环的加群关于理想的商群 R/IR/I构成环(基于上面定义的加法和乘法),称为环 RR 关于理想 II 的商环(quotient ring),仍记为 R/IR/I

        R/I={x+IxR}R/I=\{x+I|x\in R\}

        基本性质:

        • a=a+I\overline{a}=a+I
        • 商环的(加法)零元:0=0+I\overline{0}=0+I
        • 商环的(加法)单位元:e=e+I\overline{e}=e+I

        # 环同态

        环的同态就是与环的加法与乘法都可交换的函数

        (R,+,)(R,+,\cdot)(R,,)(R',\oplus,\otimes) 是环,φ:RR\varphi:R\to R'RRRR' 的函数,若对 a,bR\forall a,b\in R 有:

        φ(a+b)=φ(a)φ(b)φ(ab)=φ(a)φ(b)\varphi(a+b)=\varphi(a)\oplus\varphi(b)\\ \varphi(a\cdot b)=\varphi(a)\otimes\varphi(b)

        则称 φ\varphiRRRR'同态

        • φ\varphi 是单函数,则称为单同态
        • φ\varphi 是满函数,则称为满同态
        • φ\varphi 是双函数,则称为同构,记为:RRR\cong R'

        RRRR' 是两个环,定义函数 φ:RRφ:R→R^′aR,φ(a)=0∀a\in R,φ(a)= 0,这里 0 是 RR' 的零元,则容易验证 φφ 是同态,这个同态称为零同态 (zero homomorphism)

        RR 是环,IIRR 的理想,则很自然地有同态 ρ:RR/Iρ:R→R/IaR,ρ(a)=a∀a\in R,ρ(a)=\overline a,这里 a\overline aaa 所在的等价类,即 a+Ia+I 。这个同态称为商环 R/IR/I自然同态 (natural morphism)

        零同态、自然同态都是满同态。

        自然同态一个例子:

        • 整数环 Z\mathbb Z 到模 mm 剩余类环 Zm\mathbb Z_m 有很自然的满同态 φ:ZZmφ:\mathbb Z→\mathbb Z_mzZ,φ(z)=zmodm∀z\in \mathbb Z, φ(z)=z\bmod m

        零同态全部映射到零元,单同态给出子代数,满同态给出商代数

        环同态保持:

        • 加法单位元
        • 子环

        不保持:

        • 理想

        # Zm\mathbb{Z}_mZn\mathbb{Z}_n 的同态

        zm到zn的同态符号约定

        φ:ZmZn\varphi:\mathbb{Z}_m\to\mathbb{Z}_nZm\mathbb{Z}_mZn\mathbb{Z}_n 的同态,当且仅当:

        • φ(1)=[a]\varphi(\overline1)=[a];(环同态由该值决定)
        • φ(x)=x[a]\varphi(\overline{x})=x[a];(乘法陪集)
        • m[a]=[0]m[a]=[0]
        • [a]2=[a][a]^2=[a];([a][a]Zm\mathbb{Z}_m 的幂等元)

        # 环扩张

        环的扩张定理,也称为挖补定理

        环扩张定理

        如图,RRSS' 是 2 个没有公共元素的环,存在 RRSS' 的单同态:

        ψ:RS\psi:R\to S'

        并且:ψ(R)=S\psi(R)=S

        那么 RR 可以扩张为 RR' ,满足:

        • RR=SSR'-R=S'-S
        • SRS'\cong R'

        RR 是一个没有单位元的环,则存在一个有单位元的环 RR' ,使得 RRRR' 的子环。

        # 特征

        RR 是环。若存在最小正整数 nn 使得对所有 aRa\in Rna=0na=0 ,则称 nn 为环 RR特征。如果不存在这样的正整数,则称 RR 的特征为 0。环 R 的特征记为 Char RR

        数域 Z,Q,R,C\mathbb Z, \mathbb Q, \mathbb R, \mathbb C 的特征都是 0,而模 mm 剩余类环 Zm\mathbb Z_m 的特征是 mm

        【定理】设 RR 是有单位元 ee 的环。若 ee 关于加法的阶为无穷大,则 Char R=0R=0,否则 Char R=eR=|e|,这里 e|e| 在环的加群中的阶。

        证明 若 ee 的加法的阶为无穷大,则不存在正整数 nn 使得 ne=0ne=0 ,从而 Char R=0R= 0,否则,若 e=n|e|=n ,则 ne=0ne=0 ,从而对任意 aRa\in Rna=n(ea)=(ne)a=0na=n(ea)=(ne)a=0

        整环的特征是 0 或是一个素数,进而域的特征也只能是 0 或是素数。

        根据 RR 的特征构造 Z\mathbb ZRR 的同态

        【定理】设 RR 是有单位元 ee 的环,定义函数 φ:ZRφ:\mathbb Z→RnZ,φ(n)=ne∀n\in \mathbb Z, φ(n)=ne,则 φφ 是环 Z\mathbb ZRR 的同态。

        【推论】设 RR 是有单位元 ee 的环。

        1. 如果 RR 的特征为 n>0n>0,则 RR 包含一个与 Zn\mathbb Z_n 同构的子环;

        2. 如果 RR 的特征为 0 ,则 RR 包含一个与 Z\mathbb Z 同构的子环。

          • 所以特征为 0 的环一定是无穷环

        ⭐️每个有限域的阶必为素数的幂,即有限域的阶可表示为 pnp^n ( pp 是素数、nn 是正整数),该有限域通常称为 Galois 域 (Galois Fields),记为 GF(pn)GF(p^n)

        【推论】设 FF 是域。

        1. 如果 FF 的特征为 0 ,则 FF 包含一个与有理数域同构的子域;

        2. 如果 RR 的特征是素数 pp ,则 FF 包含一个与模 pp 剩余类环 Zp\mathbb Z_p 同构的子域。

        # 素域

        若域 FF 不包含任何真子域,则称 F 是素域 (prime field)。

        特征为素数 pp交换环中:

        (a±b)pn=apn+bpn(a\pm b)^{p^n}=a^{p^n}+b^{p^n}


        证明思路

        交换环中也有二项式定理,以此证明 (a±b)p=ap+bp(a\pm b)^{p}=a^{p}+b^{p} ,然后数学归纳法即可。

        \ No newline at end of file +环 - 代数结构 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 环的相关定义

        RR 是非空集合,如果在 RR 上定义了两个二元运算 “++”(称为加法)和 “ \cdot ”(称为乘法),且满足:

        • (1) RR 关于加法 ++ 构成交换群(即加群)
        • (2) RR 关于乘法 \cdot 构成半群,即乘法满足结合律:a,b,cR,a(bc)=(ab)c\forall a,b,c\in R,\ \ a\cdot(b\cdot c)=(a\cdot b)\cdot c
        • (3) 乘法 \cdot 对加法有左右分配律

        a,b,cRa(b+c)=ab+ac(b+c)a=ba+ca\forall a,b,c\in R\\ a\cdot(b+c)=a\cdot b+a\cdot c\\ (b+c)\cdot a=b\cdot a+c\cdot a

        则称代数 (R,+,)(R,+,\ \cdot\ ) 是环(ring)。在通过上下文能明确运算时,通常直接称 RR 为环。

        有教材要求环乘法必须有单位元,也有教材要求环乘法必须满足交换律等

        对环 (R,+,)(R,+,\ \cdot\ )(R,+)(R\ ,+) 是交换群,称为环 RR 的加法群,其单位元通常用 0 表示,称为环 RR 的零元,元素 aa 的加法逆元记为 a-a ,称为 aa 的负元。

        通过负元引入环的减法,即 aba-b 定义为 a+(b)a+(-b)

        • 环乘法是否满足交换律
          • 若环 RR 的乘法也满足交换律,则称为交换环 (commutative ring)
        • 是否有单位元
          • 若环 RR 的乘法有单位元,则称为有单位元环,环乘法的单位元通常记为 ee11 ,并称为环 RR 的单位元
        • 环乘法当环 RR 有单位元 ee 时,考虑环的元素是否有逆元
          • 若对 aRa\in R ,存在 bRb\in R ,使得 ab=ba=eab=ba=e ,则称 aa 是可逆元,或称为环 RR单位 (unit),并称 bbaa 的逆元。当然这时 bb 也是可逆元,且 aabb 的逆元
          • 所有可逆元关于环乘法构成群,称为 RR单位群 (group of units),记为 U(R)U(R)
        • 环乘法是否有零因子
          • 对环 RR 的两个非零元素 a,ba,b ,若 ab=0a\cdot b=0 ,则称 aa左零因子 (left zero-divisor),b 是右零因子 (right zero-divisor)。左零因子和右零因子统称为零因子
          • 若环 RR 的所有非零元素都不是左零因子或右零因子,则称 RR 为无零因子环。

        当集合 RR 只有一个元素 00 ,定义 RR 的加法运算和乘法运算:0+0=0,00=00+0=0,\ 0\cdot 0=0,则 ({0},+,)(\{0\},+,\ \cdot\ ) 构成环,这个环称为零环。在这个环中,零元也是单位元,而零元也是可逆元

        零环过于简单,通常在对环进行讨论时都将零环排除在外。因此在提到环时,总是默认环至少有两个元素,这时零元不可能是单位元,也不可能是可逆元。

        # 整环

        • 有单位元
        • 无零因子
        • (乘法) 可交换

        的环称为整环 (integral domain)


        证明一个环是整环:

        \forall a,b\in R,\ \ ab=0\Rightarrow a=0\or b=0

        # 除环

        • 有单位元
        • 至少有两个元素
        • 每个非零元都可逆 (乘法)
        • (无零因子,可逆元都不是零因子)

        的环称为除环 (division ring)

        #

        交换除环称为 (field) (比较常用),也即需满足:

        • 有单位元
        • 至少有两个元素
        • 每个非零元都可逆 (乘法)
        • (乘法) 可交换
        • (无零因子,可逆元都不是零因子)

        可逆元都不是零因子(为什么?),因此域都是整环!

        非交换除环也称 (skew field) (比较少用)

        # 常用环

        # 整数环及其子环

        • 整数集 Z\mathbb Z 关于普通加法 ++ 和普通乘法 \cdot 构成环,称为整数环

          • 整数环 Z\mathbb Z交换环,(乘法) 零元是 0,单位元是 1

          • 只有 1 和 -1 是可逆元,因此整数环的单位群是 ({1,1},)(\{1, -1\},\ \cdot\ )

          • 整数环的每个非零整数都不是零因子,因此整数环是整环

        • 固定整数 dd ,集合 dZ={kdkZ}d\mathbb Z=\{kd|k\in \mathbb Z\} 关于普通加法 ++ 和普通乘法 \cdot 也构成交换环

          • d1d≠1 时,环 dZd\mathbb Z 没有单位元。显然环 dZd\mathbb Z 也是无零因子环

          • 具体来说,所有偶数构成的集合 2Z2\mathbb Z 关于普通加法和普通乘法构成无单位元、无零因子的交换环。

        # 模 m 剩余类环及其子环

        固定整数 m2m≥2 ,模 mm 剩余类 Zm={0,1,,m1}\mathbb Z_m=\{\overline{0},\overline{1},\cdots,\overline{m-1}\} 关于模 mmm⊕_m 和模 mmm⊗_m 构成环,称为模 mm 剩余类环 (residue class ring)

        • 0\overline 0 表示整除 mm 余 0 的所有整数构成的集合,Zm\mathbb Z_m 的元素是集合

        • mm 剩余类环 Z\mathbb Z 是有单位元交换环,零元是 0,有单位元 1

        • 对于 aZma\in \mathbb Z_m ,如果 aamm 互质,则 aa 关于模 mm 乘有逆元,因此单位群是 U(m)U(m)

        • mm 是质数 pp 时,Zp={0}U(p)\mathbb Z_p=\{0\}\cup U(p) ,关于 p⊕_pp⊗_p 构成有单位元、每个非零元都可逆的交换环,也即这时 Zp\mathbb Z_p 是域

        • mm 不是质数时,若 m=kd(2km,2km)m=kd(2≤k≤m,2≤k≤m) ,则 kkdd 都是模 mm 剩余类环的零因子,这时就不是无零因子环,当然也不是整环

        # 有理数域

        有理数集 QQ 关于普通加法 ++ 和普通乘法 \cdot 构成环

        • 有单位元 1,是交换环,而且每个非零有理数 rr 都有逆 1/r1/r

        • 单位群是所有非零有理数集 Q\mathbb Q^* 关于普通乘法构成的群 (Q,)(\mathbb Q^*,\ \cdot\ )

        • 因此通常直接称有理数集 Q\mathbb Q 为有理数域

        实数集 R\mathbb R 和复数集 C\mathbb C 关于普通加法 ++ 和普通乘法 \cdot 也都构成域,分别称为实数域 R\mathbb R 和复数域 C\mathbb C

        • 单位群分别是所有非零实数集 R\mathbb R^* 和非零复数集 C\mathbb C^* 关于普通乘法构成的群

        • 复数域 C\mathbb C 的零元是实数 0,单位元是实数 1,而复数乘法的逆为:(a+bi)1=abia2+b2(a+bi)^{-1}=\frac{a-bi}{a^2+b^2}

        # 高斯整环

        高斯整环

        # 类高斯整环

        看起来和高斯整环很像,我乱起的名字,不是很重要

        类高斯整环1

        类高斯整环2

        # 全矩阵环

        全矩阵环

        # 多项式环

        多项式环1多项式环2

        例子

        多项式环举例

        # 理想

        # 定义

        RR 是环,IIRR 的非空子集。若 I 满足:

        1. r1,r2I\forall r_1,r_2\in I,有 r1r2Ir_1-r_2\in I
        2. rI,sR,rs,srI\forall r\in I,\ \forall s\in R,\ \ rs,sr\in I

        则称 II 为环 RR理想 (ideal),记为 IRI⊲R,若 IIRR 的真子集,则称 IIRR真理想 (proper ideal)。

        理想是子环:显然如果 IIRR 的理想,则 II 必定是 RR 的子环;

        ⚠️ 当然子环不一定是理想 ⚠️

        RR 的单个零元构成的集合 {0}\{0\}(称为零理想)和 RR 本身都是 RR 的理想,这两个理想称为 RR平凡理想。因此 RR 的非平凡理想就是非零真理想

        # 常用理想

        整数环 ZZ 的所有理想是 dZ={dzzZ},d=0,1,dZ=\{dz|z\in Z\},\ d=0,1,\cdots

        mm 剩余类环 ZmZ_m 的所有理想是 dZm={dzzZm},d=0,1,,m1dZ_m=\{dz|z\in Z_m\},\ d=0, 1, \cdots, m-1


        证明

        理想是子环,而 ZZ 的每个子环都具有形式 dZdZ ,而且对任意子环 dZdZ ,对任意 dzdZdz\in dZ ,以及 sZs\in Z ,显然有 dzs,sdzdZdzs, sdz\in dZ ,因此每个子环 dZdZ 都是 ZZ 的理想

        类似可得到模 mm 剩余类环 ZmZ_m 的所有理想是 dZm,d=0,1,,m1dZ_m, d=0, 1, \cdots, m-1

        # 运算

        #

        RR 是环,I,JI,J 都是 RR 的理想,I+JI+J 称为理想 IIJJ

        I+J={a+bRaI,bJ}I+J=\{a+b\in R|a\in I, b\in J\}

        #

        IJI\cap J 称为 IIJJ

        IJ={xRxI,xJ}I\cap J=\{x\in R|x\in I,x\in J\}

        #

        IJIJ 称为理想 IIJJ

        IJ={a1b1+a2b2++anbnnN,akI,bkJ}IJ=\{a_1b_1+a_2b_2+\cdots+a_nb_n|n\in\mathbb N^*,a_k\in I,b_k\in J\}

        # 和交积保持理想

        IIJJ 都是环 RR 的理想,则 I+JI+JIJI\cap JIJIJ 也是环 RR 的理想

        环的任意有限多个理想的和仍是理想,而任意有限或无限多个理想的交仍是理想

        # 主理想

        # 定义

        RR 是环,aaRR 的元素,记所有包含 aa 的理想构成的集合为:

        ={IRaI}\sum=\{I\lhd R|a\in I\}

        至少有 RR\in\sum ,所以 \sum 非空。令:

        a=II\langle a\rangle=\bigcap_{I\in\sum}I

        a\langle a\rangle 是理想,而且是包含 aa 的最小理想。这个理想称由 aa 生成的主理想(principal ideal),aa 为其生成元

        # 定理结论

        RR 的由 aa 生成的主理想满足:

        a={(i=1nxiayi)+xa+ay+maxi,yi,x,yR,nZ+,mZ}\langle a\rangle=\big\{ \big(\sum_{i=1}^{n}x_iay_i\big) +xa+ay+ma\big|x_i,y_i,x,y\in R,n\in\mathbb Z^+,m\in\mathbb Z \big\}

        RR 有单位元,则:

        \langle a\rangle=\big\{ \big(\sum_{i=1}^{n}x_iay_i\big) \big|x_i,y_i\in R \big\}\

        RR交换环,则:

        a={xa+maxR,mZ}\langle a\rangle=\big\{ xa+ma\big|x\in R,m\in\mathbb Z \big\}

        ⭐️若 RR有单位元的交换环,则:

        a=aR={arrR}\langle a\rangle=aR=\big\{ ar\big|r\in R \big\}

        # 常用理想的主理想

        • 整数环 Z\mathbb Z 的每个理想 dZd\mathbb Z 都是主理想,即是 d\langle d\rangle
        • mm 剩余类环 Zm\mathbb Z_m 的每个理想 dZmd\mathbb Z_m 也都是主理想,即也是 d\langle d\rangle

        RR 是环,a1,a2,,anRa_1,a_2,\cdots,a_n\in R,则:

        a1,a2,,an=a1+a2++an\langle a_1,a_2,\cdots,a_n\rangle=\langle a_1\rangle+\langle a_2\rangle+\cdots+\langle a_n\rangle

        也是 RR 的理想,并且是包含 a1,a2,,ana_1,a_2,\cdots,a_n 的最小理想!称为由 a1,a2,,ana_1,a_2,\cdots,a_n 生成的理想。

        整数环 Z\mathbb Za1,a2a_1,a_2 生成的理想是gcd(a1,a2)\langle\gcd(a_1,a_2)\rangle

        # 商环

        环关于理想的商环

        # 定义

        RR 是环,II(R,+,)(R,+,\ \cdot\ ) 的一个理想,则 (I,+)(I,+)(R,+)(R,+) 的正规子群,它所有(关于加群)的陪集构成集合 R/I{x+IxR}R/I=\{x+I|x\in R\} 、记陪集 x+Ix+Ix\overline x

        # 运算的定义

        # 加法

        在集合 R/IR/I 上可定义加法 ++

        x,yR/I,x+y=x+y(x+I)+(y+I)=(x+y)+I\forall \overline x,\overline y\in R/I,\\ \overline{x}+\overline{y}=\overline{x+y}\Longleftrightarrow (x+I)+(y+I)=(x+y)+I

        这里 x+y\overline{x}+\overline{y} 中的 ++ 是要定义的加法,即后面所说商环中的加法,而 x+y\overline{x+y} 中的 ++ 是环 RR 中的加法。

        根据正规子群的性质,该定义是合适的。

        # 乘法

        在集合 R/IR/I 上可定义乘法 ×\times

        x,yR/I,xy=xy(x+I)(y+I)=(xy)+I\forall \overline x,\overline y\in R/I,\\ \overline{x}\cdot\overline{y}=\overline{x\cdot y}\Longleftrightarrow (x+I)\cdot(y+I)=(x\cdot y)+I

        这里 xy\overline{x}\cdot\overline{y} 中的 $\cdot $ 是要定义的乘法,即后面所说商环中的乘法,而 xy\overline{x\cdot y} 中的 $\cdot $ 是环 RR 中的乘法。


        由于这定义在等价类(陪集)上,因此需要证明该定义是合适的,即与选择的代表无关,即:

        x1=x2y1=y2x1y1=x2y2\overline{x_1}=\overline{x_2}\ \land\overline{y_1}=\overline{y_2}\\ \Downarrow\\ \overline{x_1\cdot y_1}=\overline{x_2\cdot y_2}

        商环乘法合适定义证明

        # 构成环

        商环构成环证明1

        商环构成环证明2

        # 商环定义

        RR 是环,II 是理想,则环的加群关于理想的商群 R/IR/I构成环(基于上面定义的加法和乘法),称为环 RR 关于理想 II 的商环(quotient ring),仍记为 R/IR/I

        R/I={x+IxR}R/I=\{x+I|x\in R\}

        基本性质:

        • a=a+I\overline{a}=a+I
        • 商环的(加法)零元:0=0+I\overline{0}=0+I
        • 商环的(加法)单位元:e=e+I\overline{e}=e+I

        # 环同态

        环的同态就是与环的加法与乘法都可交换的函数

        (R,+,)(R,+,\cdot)(R,,)(R',\oplus,\otimes) 是环,φ:RR\varphi:R\to R'RRRR' 的函数,若对 a,bR\forall a,b\in R 有:

        φ(a+b)=φ(a)φ(b)φ(ab)=φ(a)φ(b)\varphi(a+b)=\varphi(a)\oplus\varphi(b)\\ \varphi(a\cdot b)=\varphi(a)\otimes\varphi(b)

        则称 φ\varphiRRRR'同态

        • φ\varphi 是单函数,则称为单同态
        • φ\varphi 是满函数,则称为满同态
        • φ\varphi 是双函数,则称为同构,记为:RRR\cong R'

        RRRR' 是两个环,定义函数 φ:RRφ:R→R^′aR,φ(a)=0∀a\in R,φ(a)= 0,这里 0 是 RR' 的零元,则容易验证 φφ 是同态,这个同态称为零同态 (zero homomorphism)

        RR 是环,IIRR 的理想,则很自然地有同态 ρ:RR/Iρ:R→R/IaR,ρ(a)=a∀a\in R,ρ(a)=\overline a,这里 a\overline aaa 所在的等价类,即 a+Ia+I 。这个同态称为商环 R/IR/I自然同态 (natural morphism)

        零同态、自然同态都是满同态。

        自然同态一个例子:

        • 整数环 Z\mathbb Z 到模 mm 剩余类环 Zm\mathbb Z_m 有很自然的满同态 φ:ZZmφ:\mathbb Z→\mathbb Z_mzZ,φ(z)=zmodm∀z\in \mathbb Z, φ(z)=z\bmod m

        零同态全部映射到零元,单同态给出子代数,满同态给出商代数

        环同态保持:

        • 加法单位元
        • 子环

        不保持:

        • 理想

        # Zm\mathbb{Z}_mZn\mathbb{Z}_n 的同态

        zm到zn的同态符号约定

        φ:ZmZn\varphi:\mathbb{Z}_m\to\mathbb{Z}_nZm\mathbb{Z}_mZn\mathbb{Z}_n 的同态,当且仅当:

        • φ(1)=[a]\varphi(\overline1)=[a];(环同态由该值决定)
        • φ(x)=x[a]\varphi(\overline{x})=x[a];(乘法陪集)
        • m[a]=[0]m[a]=[0]
        • [a]2=[a][a]^2=[a];([a][a]Zm\mathbb{Z}_m 的幂等元)

        # 环扩张

        环的扩张定理,也称为挖补定理

        环扩张定理

        如图,RRSS' 是 2 个没有公共元素的环,存在 RRSS' 的单同态:

        ψ:RS\psi:R\to S'

        并且:ψ(R)=S\psi(R)=S

        那么 RR 可以扩张为 RR' ,满足:

        • RR=SSR'-R=S'-S
        • SRS'\cong R'

        RR 是一个没有单位元的环,则存在一个有单位元的环 RR' ,使得 RRRR' 的子环。

        # 特征

        RR 是环。若存在最小正整数 nn 使得对所有 aRa\in Rna=0na=0 ,则称 nn 为环 RR特征。如果不存在这样的正整数,则称 RR 的特征为 0。环 R 的特征记为 Char RR

        数域 Z,Q,R,C\mathbb Z, \mathbb Q, \mathbb R, \mathbb C 的特征都是 0,而模 mm 剩余类环 Zm\mathbb Z_m 的特征是 mm

        【定理】设 RR 是有单位元 ee 的环。若 ee 关于加法的阶为无穷大,则 Char R=0R=0,否则 Char R=eR=|e|,这里 e|e| 在环的加群中的阶。

        证明 若 ee 的加法的阶为无穷大,则不存在正整数 nn 使得 ne=0ne=0 ,从而 Char R=0R= 0,否则,若 e=n|e|=n ,则 ne=0ne=0 ,从而对任意 aRa\in Rna=n(ea)=(ne)a=0na=n(ea)=(ne)a=0

        整环的特征是 0 或是一个素数,进而域的特征也只能是 0 或是素数。

        根据 RR 的特征构造 Z\mathbb ZRR 的同态

        【定理】设 RR 是有单位元 ee 的环,定义函数 φ:ZRφ:\mathbb Z→RnZ,φ(n)=ne∀n\in \mathbb Z, φ(n)=ne,则 φφ 是环 Z\mathbb ZRR 的同态。

        【推论】设 RR 是有单位元 ee 的环。

        1. 如果 RR 的特征为 n>0n>0,则 RR 包含一个与 Zn\mathbb Z_n 同构的子环;

        2. 如果 RR 的特征为 0 ,则 RR 包含一个与 Z\mathbb Z 同构的子环。

          • 所以特征为 0 的环一定是无穷环

        ⭐️每个有限域的阶必为素数的幂,即有限域的阶可表示为 pnp^n ( pp 是素数、nn 是正整数),该有限域通常称为 Galois 域 (Galois Fields),记为 GF(pn)GF(p^n)

        【推论】设 FF 是域。

        1. 如果 FF 的特征为 0 ,则 FF 包含一个与有理数域同构的子域;

        2. 如果 RR 的特征是素数 pp ,则 FF 包含一个与模 pp 剩余类环 Zp\mathbb Z_p 同构的子域。

        # 素域

        若域 FF 不包含任何真子域,则称 F 是素域 (prime field)。

        特征为素数 pp交换环中:

        (a±b)pn=apn+bpn(a\pm b)^{p^n}=a^{p^n}+b^{p^n}


        证明思路

        交换环中也有二项式定理,以此证明 (a±b)p=ap+bp(a\pm b)^{p}=a^{p}+b^{p} ,然后数学归纳法即可。

        \ No newline at end of file diff --git a/math/discrete-math/discrete-math-basics/dmb/index.html b/math/discrete-math/discrete-math-basics/dmb/index.html index bf7bb38e..a5babdb6 100644 --- a/math/discrete-math/discrete-math-basics/dmb/index.html +++ b/math/discrete-math/discrete-math-basics/dmb/index.html @@ -1 +1 @@ -离散数学基础 - 整理 - 离散数学基础 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +离散数学基础 - 整理 - 离散数学基础 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git a/math/discrete-math/discrete-math-basics/dmbs/algebra-system/index.html b/math/discrete-math/discrete-math-basics/dmbs/algebra-system/index.html index bf496330..c1236814 100644 --- a/math/discrete-math/discrete-math-basics/dmbs/algebra-system/index.html +++ b/math/discrete-math/discrete-math-basics/dmbs/algebra-system/index.html @@ -1 +1 @@ -离散数学基础:代数系统 - 离散数学基础 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 思维导图

        代数系统

        # 二元运算

        封闭性

        # 定义

        f:S×SS是集合S的运算,则称集合S对运算f封闭\begin{aligned}& f:S\times S\to S 是集合S的运算,\\& 则称集合S对运算f封闭 \end{aligned}

        TSS的子集,若t1,t2T(f(t1,t2)T)则称集合TS的运算f封闭\begin{aligned}& T\subseteq S是S的子集,若\forall t_1,t_2\in T(f(t_1,t_2)\in T),\\& 则称集合T对S的运算f封闭 \end{aligned}

        # 经典例子

        自然数集、整数集和实数集对加法、乘法运算封闭\begin{aligned}& 自然数集、整数集和实数集对加法、乘法运算封闭 \end{aligned}

        自然数集和整数集对实数集的加法、乘法运算封闭\begin{aligned}& 自然数集和整数集对实数集的加法、乘法运算封闭 \end{aligned}

        自然数集对整数集的加法、乘法运算封闭\begin{aligned}& 自然数集对整数集的加法、乘法运算封闭 \end{aligned}

        自然数集对实数集和整数集的减法运算不封闭\begin{aligned}& 自然数集对实数集和整数集的减法运算不封闭 \end{aligned}

        自然数集、整数集和实数集对除法运算不封闭\begin{aligned}& 自然数集、整数集和实数集对除法运算不封闭 \end{aligned}

        关系复合P(A×B)×P(C×D)P(A×D)的函数一般情况下不是某个集合的运算关系复合是运算A=B=C=D关系复合P(A×A)×P(A×A)P(A×A)的函数\begin{aligned}& 关系复合\circ 是P(A\times B)\times P(C\times D)\to P(A\times D)的函数\\& 一般情况下\circ 不是某个集合的运算\\& 关系复合\circ 是运算\\ \Leftrightarrow & A=B=C=D\\ \Leftrightarrow & 关系复合\circ 是P(A\times A)\times P(A\times A)\to P(A\times A)的函数 \end{aligned}

        函数族AA关于关系复合运算封闭,AAP(A×A)\begin{aligned}& 函数族A^A关于关系复合运算\circ 封闭,A^A\subseteq P(A\times A) \end{aligned}

        # 特殊元

        # 左单位元

        el是运算的左单位元xS,elx=x\begin{aligned}& e_l是运算\circ的左单位元\Leftrightarrow\forall x\in S, e_l\circ x=x \end{aligned}

        # 右单位元

        er是运算的右单位元xS,xer=x\begin{aligned}& e_r是运算\circ的右单位元\Leftrightarrow\forall x\in S, x\circ e_r=x \end{aligned}

        # 单位元

        (幺元)

        e是运算的单位元xS,xe=ex=x\begin{aligned}& e是运算\circ的单位元\Leftrightarrow\forall x\in S, x\circ e=e\circ x=x \end{aligned}

        • 左逆元
          (左逆)

        ylx(关于运算)的左逆元ylx=ex左可逆\begin{aligned}& y_l 是x(关于运算\circ)的左逆元\\\Leftrightarrow& y_l\circ x=e\\\Leftrightarrow& x左可逆 \end{aligned}

        • 右逆元
          (右逆)

        yrx(关于运算)的右逆元xyr=ex右可逆\begin{aligned}& y_r 是x(关于运算\circ)的右逆元\\\Leftrightarrow& x\circ y_r=e\\\Leftrightarrow&x右可逆 \end{aligned}

        • 逆元
          (逆)

        yx(关于运算)的逆元y既是x的左逆又是x的右逆x可逆\begin{aligned}& y是x(关于运算\circ)的逆元\\\Leftrightarrow& y既是x的左逆又是x的右逆\\\Leftrightarrow& x可逆 \end{aligned}

        • 性质

          • 单位元若存在一定是唯一的
          • 若满足结合律,且单位元存在,则可逆元素有唯一逆元

        # 左零元

        θl是运算的左单位元xS,θlx=θl\begin{aligned}& \theta_l 是运算\circ的左单位元\Leftrightarrow\forall x\in S, \theta_l \circ x=\theta_l \end{aligned}

        # 左零元

        θr是运算的左单位元xS,xθr=θr\begin{aligned}& \theta_r 是运算\circ的左单位元\Leftrightarrow\forall x\in S, x\circ\theta_r =\theta_r \end{aligned}

        # 零元

        θ是运算的左单位元xS,xθ=θx=θ\begin{aligned}& \theta 是运算\circ的左单位元\Leftrightarrow\forall x\in S, x\circ\theta=\theta\circ x =\theta \end{aligned}

        # 幂等元

        xx=x\begin{aligned}& x\circ x=x \end{aligned}

        # 二元运算性质

        # 交换律

        • 描述

        运算S上是可交换的,或满足交换律的\begin{aligned}& 运算\circ 在S上是可交换的,或满足交换律的 \end{aligned}

        • 定义

        x,yS,xy=yx\begin{aligned}& \forall x,y\in S,\ \ \ x\circ y=y\circ x \end{aligned}

        # 结合律

        • 描述

        运算S上是可结合的,或满足结合律的\begin{aligned}& 运算\circ 在S上是可结合的,或满足结合律的 \end{aligned}

        • 定义

        x,y,zS,(xy)z=x(yx)\begin{aligned}& \forall x,y,z\in S,\ \ \ (x\circ y)\circ z=x\circ (y\circ x) \end{aligned}

        • 满足结合律的运算的指数运算 (幂运算)

          • 定义

        xn=xx...x\begin{aligned}& x^n=x\circ x\circ...\circ x \end{aligned}

        xn={xxn1xn=1n>1\begin{aligned} x^n=\left\{\begin{aligned} &x\\ &x^{n-1}\circ x \end{aligned}\right. && \begin{aligned} &n=1\\ &n\gt 1 \end{aligned} \end{aligned}

        • 性质

        xnxm=xn+m\begin{aligned}& x^nx^m=x^{n+m} \end{aligned}

        (xn)xm=xnm\begin{aligned}& (x^n)x^m=x^{nm} \end{aligned}

        # 幂等律

        • 定义

        S的运算满足幂等律xS,xx=xS的所有元素都是幂等元\begin{aligned}& S的运算\circ 满足幂等律\\\Leftrightarrow& \forall x\in S, x\circ x=x\\\Leftrightarrow& S的所有元素都是幂等元 \end{aligned}

        # 分配律

        • 描述

        运算对运算是可分配的,或满足分配律的\begin{aligned}& 运算*对运算\circ 是可分配的,或满足分配律的 \end{aligned}

        • 定义

        x,y,zS,x(yz)=(xy)(xz)(yz)x=(yx)(zx)\begin{aligned} \forall x,y,z\in S\ ,\ & x*(y\circ z)=(x*y)\circ(x*z)\\& (y\circ z)*x=(y*x)\circ(z*x) \end{aligned}

        # 吸收律

        • 描述

        运算对运算是可分配的,或满足分配律的\begin{aligned}& 运算*对运算\circ 是可分配的,或满足分配律的 \end{aligned}

        • 定义

        x,yS,x(xy)=xx(xy)=x\begin{aligned} \forall x,y\in S\ ,\ &x*(x\circ y)=x\\&x\circ(x*y)=x \end{aligned}

        • 典例

        逻辑与、或满足吸收律\begin{aligned}& 逻辑与、或满足吸收律 \end{aligned}

        集合交、并满足吸收律\begin{aligned}& 集合交、并满足吸收律 \end{aligned}

        # 消去律

        • 定义

        x,y,zS,x不是零元xy=xzy=zyx=zxy=z\begin{aligned}& \forall x,y,z\in S\ ,\ x不是零元\\& x\circ y=x\circ z\to y=z\\& y\circ x=z\circ x\to y=z \end{aligned}

        • 典例

        数集加法、乘法满足消去律\begin{aligned}& 数集加法、乘法满足消去律 \end{aligned}

        集合交、并不满足消去律\begin{aligned}& 集合交、并不满足消去律 \end{aligned}

        #

        # 定义

        • 一个群包含

        一个二元运算\begin{aligned}& 一个二元运算\circ \end{aligned}

        满足结合律\begin{aligned}& \circ 满足结合律 \end{aligned}

        一个零元运算即单位元e\begin{aligned}& 一个零元运算即单位元e \end{aligned}

        e的单位元\begin{aligned}& e是\circ 的单位元 \end{aligned}

        一个一元运算()1\begin{aligned}& 一个一元运算(-)^{-1} \end{aligned}

        ()1给出每个元素关于的逆元\begin{aligned}& (-)^{-1}给出每个元素关于\circ 的逆元 \end{aligned}

        • 一个独异点包含

        一个二元运算\begin{aligned}& 一个二元运算\circ \end{aligned}

        满足结合律\begin{aligned}& \circ 满足结合律 \end{aligned}

        一个零元运算即单位元e\begin{aligned}& 一个零元运算即单位元e \end{aligned}

        e的单位元\begin{aligned}& e是\circ 的单位元 \end{aligned}

        • 一个半
          群包含

        一个二元运算\begin{aligned}& 一个二元运算\circ \end{aligned}

        满足结合律\begin{aligned}& \circ 满足结合律 \end{aligned}

        # 性质

        • 群有唯一单位元,没有零元,且所有元素有唯一逆元
        • 群的二元运算满足消去律

        # 阿贝尔群

        (可交换群)

        运算满足交换律的群运算满足交换律的群

        # 群的阶

        • 群的元素个数

        # 群元素的阶

        定义群元素an次幂:an={ean1a(a1)nn=0n>0n<0a的阶a是使ak=e成立的最小正整数k,当k不存在时a是无限阶元\begin{aligned}& 定义群元素a的n次幂:\\& \begin{aligned} a^n=\left\{\begin{aligned} &e\\ &a^{n-1}a\\ &(a^{-1})^{|n|} \end{aligned}\right. &&&& \begin{aligned} &n=0\\ &n\gt 0\\ &n\lt 0 \end{aligned} \end{aligned}\\& a的阶|a|是使a^k=e成立的最小正整数k,当k不存在时a是无限阶元 \end{aligned}

        • 性质

        am=enm,mZ\begin{aligned}& a^m=e\iff n|m,\ \ m\in \mathbb Z \end{aligned}

        (a1)k=(ak)1,kN\begin{aligned}& (a^{-1})^k=(a^k)^{-1},\ k\in\mathbb N \end{aligned}

        a1=a\begin{aligned}& |a^{-1}|=|a| \end{aligned}

        a=amgcd(a,m),mZ\begin{aligned}& |a|=|a^m|gcd(|a|,m),\ \ m\in \mathbb Z \end{aligned}

        # 模 m 单位群

        # 定义

        Zm={0,1,2,...,m1}U(m)Zm中所有与m互质的数构成的集合:Um={aZmgcd(a,m)=1}m乘运算Um构成群\begin{aligned}& \mathbb{Z}_m=\{0,1,2,...,m-1\}\\& U(m)是\mathbb{Z}_m中所有与m互质的数构成的集合:\\& U_m=\{a\in\mathbb{Z}_m|gcd(a,m)=1\}\\& 模m乘运算\otimes与U_m构成群 \end{aligned}

        # 正规子群

        or
        不变子群

        # 定义

        HG的子群,aG,Ha=aH\begin{aligned}& H是G的子群, \forall a\in G,\ Ha=aH \end{aligned}

        # 平凡正规子群

        G的单位元子群{e}G的单位元子群\{e\}

        G本身G本身

        # 单群

        G的正规子群只有平凡正规子群,且G{e}\begin{aligned}& G的正规子群只有平凡正规子群,且G\neq \{e\} \end{aligned}

        # 正规子群

        经典例子

        • 交换群

        A交换群每个元素左陪集等于右陪集故交换群每个子群都是正规子群\begin{aligned}& A交换群每个元素左陪集等于右陪集\\& 故交换群每个子群都是正规子群 \end{aligned}

        • 只有 2 个
          相异陪集

        待补待补

        # 商群

        # 描述

        G关于正规子群H的商群G关于正规子群H的商群

        # 定义

        NG的正规子群,N的全体陪集构成集合G/N定义G/N上的运算a,bG,NaNb=Nab从而G/N关于运算构成群,单位元为N\begin{aligned}& N是G的正规子群,N的全体陪集构成集合G/N\\& 定义G/N上的运算\circ 为\forall a,b\in G,\ \ Na\circ Nb=Nab\\& 从而G/N关于运算\circ构成群,单位元为N \end{aligned}

        # 群同态

        # 描述

        GG的一个同态f是一个函数f:GG群G到G'的一个同态f是一个函数f:G\to G'

        # 定义

        a,bG,f(ab)=f(a)f(b)\begin{aligned}& \forall a,b\in G,\ \ \ f(ab)=f(a)f(b) \end{aligned}

        # 自同态

        G到自己的同态f:GG群G到自己的同态f:G\to G

        # 满同态

        f是满函数f是满函数

        # 单同态

        f是单函数f是单函数

        # 同构

        f是双函数f是双函数

        GGG\cong G'

        \ No newline at end of file +离散数学基础:代数系统 - 离散数学基础 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 思维导图

        代数系统

        # 二元运算

        封闭性

        # 定义

        f:S×SS是集合S的运算,则称集合S对运算f封闭\begin{aligned}& f:S\times S\to S 是集合S的运算,\\& 则称集合S对运算f封闭 \end{aligned}

        TSS的子集,若t1,t2T(f(t1,t2)T)则称集合TS的运算f封闭\begin{aligned}& T\subseteq S是S的子集,若\forall t_1,t_2\in T(f(t_1,t_2)\in T),\\& 则称集合T对S的运算f封闭 \end{aligned}

        # 经典例子

        自然数集、整数集和实数集对加法、乘法运算封闭\begin{aligned}& 自然数集、整数集和实数集对加法、乘法运算封闭 \end{aligned}

        自然数集和整数集对实数集的加法、乘法运算封闭\begin{aligned}& 自然数集和整数集对实数集的加法、乘法运算封闭 \end{aligned}

        自然数集对整数集的加法、乘法运算封闭\begin{aligned}& 自然数集对整数集的加法、乘法运算封闭 \end{aligned}

        自然数集对实数集和整数集的减法运算不封闭\begin{aligned}& 自然数集对实数集和整数集的减法运算不封闭 \end{aligned}

        自然数集、整数集和实数集对除法运算不封闭\begin{aligned}& 自然数集、整数集和实数集对除法运算不封闭 \end{aligned}

        关系复合P(A×B)×P(C×D)P(A×D)的函数一般情况下不是某个集合的运算关系复合是运算A=B=C=D关系复合P(A×A)×P(A×A)P(A×A)的函数\begin{aligned}& 关系复合\circ 是P(A\times B)\times P(C\times D)\to P(A\times D)的函数\\& 一般情况下\circ 不是某个集合的运算\\& 关系复合\circ 是运算\\ \Leftrightarrow & A=B=C=D\\ \Leftrightarrow & 关系复合\circ 是P(A\times A)\times P(A\times A)\to P(A\times A)的函数 \end{aligned}

        函数族AA关于关系复合运算封闭,AAP(A×A)\begin{aligned}& 函数族A^A关于关系复合运算\circ 封闭,A^A\subseteq P(A\times A) \end{aligned}

        # 特殊元

        # 左单位元

        el是运算的左单位元xS,elx=x\begin{aligned}& e_l是运算\circ的左单位元\Leftrightarrow\forall x\in S, e_l\circ x=x \end{aligned}

        # 右单位元

        er是运算的右单位元xS,xer=x\begin{aligned}& e_r是运算\circ的右单位元\Leftrightarrow\forall x\in S, x\circ e_r=x \end{aligned}

        # 单位元

        (幺元)

        e是运算的单位元xS,xe=ex=x\begin{aligned}& e是运算\circ的单位元\Leftrightarrow\forall x\in S, x\circ e=e\circ x=x \end{aligned}

        • 左逆元
          (左逆)

        ylx(关于运算)的左逆元ylx=ex左可逆\begin{aligned}& y_l 是x(关于运算\circ)的左逆元\\\Leftrightarrow& y_l\circ x=e\\\Leftrightarrow& x左可逆 \end{aligned}

        • 右逆元
          (右逆)

        yrx(关于运算)的右逆元xyr=ex右可逆\begin{aligned}& y_r 是x(关于运算\circ)的右逆元\\\Leftrightarrow& x\circ y_r=e\\\Leftrightarrow&x右可逆 \end{aligned}

        • 逆元
          (逆)

        yx(关于运算)的逆元y既是x的左逆又是x的右逆x可逆\begin{aligned}& y是x(关于运算\circ)的逆元\\\Leftrightarrow& y既是x的左逆又是x的右逆\\\Leftrightarrow& x可逆 \end{aligned}

        • 性质

          • 单位元若存在一定是唯一的
          • 若满足结合律,且单位元存在,则可逆元素有唯一逆元

        # 左零元

        θl是运算的左单位元xS,θlx=θl\begin{aligned}& \theta_l 是运算\circ的左单位元\Leftrightarrow\forall x\in S, \theta_l \circ x=\theta_l \end{aligned}

        # 左零元

        θr是运算的左单位元xS,xθr=θr\begin{aligned}& \theta_r 是运算\circ的左单位元\Leftrightarrow\forall x\in S, x\circ\theta_r =\theta_r \end{aligned}

        # 零元

        θ是运算的左单位元xS,xθ=θx=θ\begin{aligned}& \theta 是运算\circ的左单位元\Leftrightarrow\forall x\in S, x\circ\theta=\theta\circ x =\theta \end{aligned}

        # 幂等元

        xx=x\begin{aligned}& x\circ x=x \end{aligned}

        # 二元运算性质

        # 交换律

        • 描述

        运算S上是可交换的,或满足交换律的\begin{aligned}& 运算\circ 在S上是可交换的,或满足交换律的 \end{aligned}

        • 定义

        x,yS,xy=yx\begin{aligned}& \forall x,y\in S,\ \ \ x\circ y=y\circ x \end{aligned}

        # 结合律

        • 描述

        运算S上是可结合的,或满足结合律的\begin{aligned}& 运算\circ 在S上是可结合的,或满足结合律的 \end{aligned}

        • 定义

        x,y,zS,(xy)z=x(yx)\begin{aligned}& \forall x,y,z\in S,\ \ \ (x\circ y)\circ z=x\circ (y\circ x) \end{aligned}

        • 满足结合律的运算的指数运算 (幂运算)

          • 定义

        xn=xx...x\begin{aligned}& x^n=x\circ x\circ...\circ x \end{aligned}

        xn={xxn1xn=1n>1\begin{aligned} x^n=\left\{\begin{aligned} &x\\ &x^{n-1}\circ x \end{aligned}\right. && \begin{aligned} &n=1\\ &n\gt 1 \end{aligned} \end{aligned}

        • 性质

        xnxm=xn+m\begin{aligned}& x^nx^m=x^{n+m} \end{aligned}

        (xn)xm=xnm\begin{aligned}& (x^n)x^m=x^{nm} \end{aligned}

        # 幂等律

        • 定义

        S的运算满足幂等律xS,xx=xS的所有元素都是幂等元\begin{aligned}& S的运算\circ 满足幂等律\\\Leftrightarrow& \forall x\in S, x\circ x=x\\\Leftrightarrow& S的所有元素都是幂等元 \end{aligned}

        # 分配律

        • 描述

        运算对运算是可分配的,或满足分配律的\begin{aligned}& 运算*对运算\circ 是可分配的,或满足分配律的 \end{aligned}

        • 定义

        x,y,zS,x(yz)=(xy)(xz)(yz)x=(yx)(zx)\begin{aligned} \forall x,y,z\in S\ ,\ & x*(y\circ z)=(x*y)\circ(x*z)\\& (y\circ z)*x=(y*x)\circ(z*x) \end{aligned}

        # 吸收律

        • 描述

        运算对运算是可分配的,或满足分配律的\begin{aligned}& 运算*对运算\circ 是可分配的,或满足分配律的 \end{aligned}

        • 定义

        x,yS,x(xy)=xx(xy)=x\begin{aligned} \forall x,y\in S\ ,\ &x*(x\circ y)=x\\&x\circ(x*y)=x \end{aligned}

        • 典例

        逻辑与、或满足吸收律\begin{aligned}& 逻辑与、或满足吸收律 \end{aligned}

        集合交、并满足吸收律\begin{aligned}& 集合交、并满足吸收律 \end{aligned}

        # 消去律

        • 定义

        x,y,zS,x不是零元xy=xzy=zyx=zxy=z\begin{aligned}& \forall x,y,z\in S\ ,\ x不是零元\\& x\circ y=x\circ z\to y=z\\& y\circ x=z\circ x\to y=z \end{aligned}

        • 典例

        数集加法、乘法满足消去律\begin{aligned}& 数集加法、乘法满足消去律 \end{aligned}

        集合交、并不满足消去律\begin{aligned}& 集合交、并不满足消去律 \end{aligned}

        #

        # 定义

        • 一个群包含

        一个二元运算\begin{aligned}& 一个二元运算\circ \end{aligned}

        满足结合律\begin{aligned}& \circ 满足结合律 \end{aligned}

        一个零元运算即单位元e\begin{aligned}& 一个零元运算即单位元e \end{aligned}

        e的单位元\begin{aligned}& e是\circ 的单位元 \end{aligned}

        一个一元运算()1\begin{aligned}& 一个一元运算(-)^{-1} \end{aligned}

        ()1给出每个元素关于的逆元\begin{aligned}& (-)^{-1}给出每个元素关于\circ 的逆元 \end{aligned}

        • 一个独异点包含

        一个二元运算\begin{aligned}& 一个二元运算\circ \end{aligned}

        满足结合律\begin{aligned}& \circ 满足结合律 \end{aligned}

        一个零元运算即单位元e\begin{aligned}& 一个零元运算即单位元e \end{aligned}

        e的单位元\begin{aligned}& e是\circ 的单位元 \end{aligned}

        • 一个半
          群包含

        一个二元运算\begin{aligned}& 一个二元运算\circ \end{aligned}

        满足结合律\begin{aligned}& \circ 满足结合律 \end{aligned}

        # 性质

        • 群有唯一单位元,没有零元,且所有元素有唯一逆元
        • 群的二元运算满足消去律

        # 阿贝尔群

        (可交换群)

        运算满足交换律的群运算满足交换律的群

        # 群的阶

        • 群的元素个数

        # 群元素的阶

        定义群元素an次幂:an={ean1a(a1)nn=0n>0n<0a的阶a是使ak=e成立的最小正整数k,当k不存在时a是无限阶元\begin{aligned}& 定义群元素a的n次幂:\\& \begin{aligned} a^n=\left\{\begin{aligned} &e\\ &a^{n-1}a\\ &(a^{-1})^{|n|} \end{aligned}\right. &&&& \begin{aligned} &n=0\\ &n\gt 0\\ &n\lt 0 \end{aligned} \end{aligned}\\& a的阶|a|是使a^k=e成立的最小正整数k,当k不存在时a是无限阶元 \end{aligned}

        • 性质

        am=enm,mZ\begin{aligned}& a^m=e\iff n|m,\ \ m\in \mathbb Z \end{aligned}

        (a1)k=(ak)1,kN\begin{aligned}& (a^{-1})^k=(a^k)^{-1},\ k\in\mathbb N \end{aligned}

        a1=a\begin{aligned}& |a^{-1}|=|a| \end{aligned}

        a=amgcd(a,m),mZ\begin{aligned}& |a|=|a^m|gcd(|a|,m),\ \ m\in \mathbb Z \end{aligned}

        # 模 m 单位群

        # 定义

        Zm={0,1,2,...,m1}U(m)Zm中所有与m互质的数构成的集合:Um={aZmgcd(a,m)=1}m乘运算Um构成群\begin{aligned}& \mathbb{Z}_m=\{0,1,2,...,m-1\}\\& U(m)是\mathbb{Z}_m中所有与m互质的数构成的集合:\\& U_m=\{a\in\mathbb{Z}_m|gcd(a,m)=1\}\\& 模m乘运算\otimes与U_m构成群 \end{aligned}

        # 正规子群

        or
        不变子群

        # 定义

        HG的子群,aG,Ha=aH\begin{aligned}& H是G的子群, \forall a\in G,\ Ha=aH \end{aligned}

        # 平凡正规子群

        G的单位元子群{e}G的单位元子群\{e\}

        G本身G本身

        # 单群

        G的正规子群只有平凡正规子群,且G{e}\begin{aligned}& G的正规子群只有平凡正规子群,且G\neq \{e\} \end{aligned}

        # 正规子群

        经典例子

        • 交换群

        A交换群每个元素左陪集等于右陪集故交换群每个子群都是正规子群\begin{aligned}& A交换群每个元素左陪集等于右陪集\\& 故交换群每个子群都是正规子群 \end{aligned}

        • 只有 2 个
          相异陪集

        待补待补

        # 商群

        # 描述

        G关于正规子群H的商群G关于正规子群H的商群

        # 定义

        NG的正规子群,N的全体陪集构成集合G/N定义G/N上的运算a,bG,NaNb=Nab从而G/N关于运算构成群,单位元为N\begin{aligned}& N是G的正规子群,N的全体陪集构成集合G/N\\& 定义G/N上的运算\circ 为\forall a,b\in G,\ \ Na\circ Nb=Nab\\& 从而G/N关于运算\circ构成群,单位元为N \end{aligned}

        # 群同态

        # 描述

        GG的一个同态f是一个函数f:GG群G到G'的一个同态f是一个函数f:G\to G'

        # 定义

        a,bG,f(ab)=f(a)f(b)\begin{aligned}& \forall a,b\in G,\ \ \ f(ab)=f(a)f(b) \end{aligned}

        # 自同态

        G到自己的同态f:GG群G到自己的同态f:G\to G

        # 满同态

        f是满函数f是满函数

        # 单同态

        f是单函数f是单函数

        # 同构

        f是双函数f是双函数

        GGG\cong G'

        \ No newline at end of file diff --git a/math/discrete-math/discrete-math-basics/dmbs/counting/index.html b/math/discrete-math/discrete-math-basics/dmbs/counting/index.html index afa98c64..6ddde670 100644 --- a/math/discrete-math/discrete-math-basics/dmbs/counting/index.html +++ b/math/discrete-math/discrete-math-basics/dmbs/counting/index.html @@ -1 +1 @@ -离散数学基础:排列组合计数问题 - 离散数学基础 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 思维导图

        排列组合计数问题

        #

        P(n,r)=n!(nr)!P(n,r)=\frac{n!}{(n-r)!}

        n个不同物体不允许重复地选r个物体的排列数n个不同物体不允许重复地选r个物体的排列数

        n元素的字符集上长度为r且不含重复字符的字符串数n元素的字符集上长度为r且不含重复字符的字符串数

        #

        nrn^r

        n类不同物体不允许重复地选r个物体的排列数n类不同物体不允许重复地选r个物体的排列数

        n元素的字符集上长度为r且可含重复字符的字符串数n元素的字符集上长度为r且可含重复字符的字符串数

        #

        C(n,r)=n!r!(nr)!C(n,r)=\frac{n!}{r!(n-r)!}

        n个不同物体不允许重复地选r个物体的组合数n个不同物体不允许重复地选r个物体的组合数

        r个无区别的物体放到n个无区别的容器里将r个无区别的物体放到n个无区别的容器里

        长度为n且有r1(0)的二进制串数长度为n且有r个1(或0)的二进制串数

        n元素集合的r元素子集的个数n元素集合的r元素子集的个数

        #

        C(n1+r,r)=(n1+r)!r!(n1)!C(n-1+r,r)=\frac{(n-1+r)!}{r!(n-1)!}

        n类不同物体允许重复地选r个物体的组合数n类不同物体允许重复地选r个物体的组合数

        r个无区别的物体放到n个可区别的容器里将r个无区别的物体放到n个可区别的容器里

        不定方程x1+x2+...+xn=r的非负整数解的个数\begin{aligned} &不定方程x_1 +x_2 +...+x_n =r\\ &的非负整数解的个数 \end{aligned}

        #

        i=1n(rj=1i1mjmi)=(rm1)(rm1m2)...(rm1m2...mn1mn)=r!m1!m2!...m3!\begin{aligned} &\prod_{i=1}^{n} \begin{pmatrix}r-\sum_{j=1}^{i-1}m_j\\m_i\end{pmatrix}\\ =& \begin{pmatrix}r\\m_1\end{pmatrix} \begin{pmatrix}r-m_1\\m_2\end{pmatrix} ... \begin{pmatrix}r-m_1-m_2-...-m_{n-1}\\m_n\end{pmatrix} \\ =&\frac{r!}{m_1!m_2!...m_3!} \end{aligned}

        分别有m1,m2,...,mn个的n类不同物体的m1+m2+...+mn=r排列数\begin{aligned} &分别有m_1,m_2,...,m_n个的n类不同物体的\\ &m_1+m_2+...+m_n=r排列数 \end{aligned}

        r个无区别的物体放到n个可区别的容器里将r个无区别的物体放到n个可区别的容器里

        m1+m2+...+mn=r个可区别的物体放到n个可区别的容器使第i个容器恰有mi个物体的方法数\begin{aligned} &m_1+m_2+...+m_n=r个可区别的物体\\ &放到n个可区别的容器使第i个容器\\ &恰有m_i个物体的方法数 \end{aligned}

        #

        不定方程x1+x2+...+xn=r的满足ximini的非负整数解个数等于不定方程(x1min1)+(x2min2)+...+(xnminn)=ri=1nminix1+x2+...+xn=ri=1nmini的非负整数解的个数\begin{aligned} &不定方程x_1 +x_2 +...+x_n =r的满足x_i\geq min_i的非负整数解个数等于\\ &不定方程(x_1-min_1)+(x_2-min_2)+...+(x_n-min_n)=r- \sum_{i=1}^{n}min_i \\ &即x_1 +x_2 +...+x_n =r- \sum_{i=1}^{n}min_i \\ &的非负整数解的个数 \end{aligned}

        #

        不定方程x1+x2+...+xn=r的满足maxiximini的非负整数解个数:U是不定方程x1+x2+...+xn=ri=1nmini的非负整数解构成的集合N=U,性质Pi表示ximaxi,N(Pi)等于U中满足性质Pi的元素个数所求解的个数=NN(P1P2...Pn)=N(P1P2...Pn)利用容斥原理求解\begin{aligned} &不定方程x_1 +x_2 +...+x_n =r的满足max_i\geq x_i\geq min_i的非负整数解个数:\\ &令U是不定方程x_1 +x_2 +...+x_n =r- \sum_{i=1}^{n}min_i 的非负整数解构成的集合\\ &N=|U|,性质P_i表示x_i\geq max_i,N(P_i)等于U中满足性质P_i的元素个数\\ &所求解的个数= N-N(\overline{P_1\cup P_2\cup...\cup P_n}) =N(\overline{P_1}\cap\overline{P_2}\cap...\cap\overline{P_n})\\ &利用容斥原理求解 \end{aligned}

        \ No newline at end of file +离散数学基础:排列组合计数问题 - 离散数学基础 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 思维导图

        排列组合计数问题

        #

        P(n,r)=n!(nr)!P(n,r)=\frac{n!}{(n-r)!}

        n个不同物体不允许重复地选r个物体的排列数n个不同物体不允许重复地选r个物体的排列数

        n元素的字符集上长度为r且不含重复字符的字符串数n元素的字符集上长度为r且不含重复字符的字符串数

        #

        nrn^r

        n类不同物体不允许重复地选r个物体的排列数n类不同物体不允许重复地选r个物体的排列数

        n元素的字符集上长度为r且可含重复字符的字符串数n元素的字符集上长度为r且可含重复字符的字符串数

        #

        C(n,r)=n!r!(nr)!C(n,r)=\frac{n!}{r!(n-r)!}

        n个不同物体不允许重复地选r个物体的组合数n个不同物体不允许重复地选r个物体的组合数

        r个无区别的物体放到n个无区别的容器里将r个无区别的物体放到n个无区别的容器里

        长度为n且有r1(0)的二进制串数长度为n且有r个1(或0)的二进制串数

        n元素集合的r元素子集的个数n元素集合的r元素子集的个数

        #

        C(n1+r,r)=(n1+r)!r!(n1)!C(n-1+r,r)=\frac{(n-1+r)!}{r!(n-1)!}

        n类不同物体允许重复地选r个物体的组合数n类不同物体允许重复地选r个物体的组合数

        r个无区别的物体放到n个可区别的容器里将r个无区别的物体放到n个可区别的容器里

        不定方程x1+x2+...+xn=r的非负整数解的个数\begin{aligned} &不定方程x_1 +x_2 +...+x_n =r\\ &的非负整数解的个数 \end{aligned}

        #

        i=1n(rj=1i1mjmi)=(rm1)(rm1m2)...(rm1m2...mn1mn)=r!m1!m2!...m3!\begin{aligned} &\prod_{i=1}^{n} \begin{pmatrix}r-\sum_{j=1}^{i-1}m_j\\m_i\end{pmatrix}\\ =& \begin{pmatrix}r\\m_1\end{pmatrix} \begin{pmatrix}r-m_1\\m_2\end{pmatrix} ... \begin{pmatrix}r-m_1-m_2-...-m_{n-1}\\m_n\end{pmatrix} \\ =&\frac{r!}{m_1!m_2!...m_3!} \end{aligned}

        分别有m1,m2,...,mn个的n类不同物体的m1+m2+...+mn=r排列数\begin{aligned} &分别有m_1,m_2,...,m_n个的n类不同物体的\\ &m_1+m_2+...+m_n=r排列数 \end{aligned}

        r个无区别的物体放到n个可区别的容器里将r个无区别的物体放到n个可区别的容器里

        m1+m2+...+mn=r个可区别的物体放到n个可区别的容器使第i个容器恰有mi个物体的方法数\begin{aligned} &m_1+m_2+...+m_n=r个可区别的物体\\ &放到n个可区别的容器使第i个容器\\ &恰有m_i个物体的方法数 \end{aligned}

        #

        不定方程x1+x2+...+xn=r的满足ximini的非负整数解个数等于不定方程(x1min1)+(x2min2)+...+(xnminn)=ri=1nminix1+x2+...+xn=ri=1nmini的非负整数解的个数\begin{aligned} &不定方程x_1 +x_2 +...+x_n =r的满足x_i\geq min_i的非负整数解个数等于\\ &不定方程(x_1-min_1)+(x_2-min_2)+...+(x_n-min_n)=r- \sum_{i=1}^{n}min_i \\ &即x_1 +x_2 +...+x_n =r- \sum_{i=1}^{n}min_i \\ &的非负整数解的个数 \end{aligned}

        #

        不定方程x1+x2+...+xn=r的满足maxiximini的非负整数解个数:U是不定方程x1+x2+...+xn=ri=1nmini的非负整数解构成的集合N=U,性质Pi表示ximaxi,N(Pi)等于U中满足性质Pi的元素个数所求解的个数=NN(P1P2...Pn)=N(P1P2...Pn)利用容斥原理求解\begin{aligned} &不定方程x_1 +x_2 +...+x_n =r的满足max_i\geq x_i\geq min_i的非负整数解个数:\\ &令U是不定方程x_1 +x_2 +...+x_n =r- \sum_{i=1}^{n}min_i 的非负整数解构成的集合\\ &N=|U|,性质P_i表示x_i\geq max_i,N(P_i)等于U中满足性质P_i的元素个数\\ &所求解的个数= N-N(\overline{P_1\cup P_2\cup...\cup P_n}) =N(\overline{P_1}\cap\overline{P_2}\cap...\cap\overline{P_n})\\ &利用容斥原理求解 \end{aligned}

        \ No newline at end of file diff --git a/math/discrete-math/discrete-math-basics/dmbs/difference-equations/index.html b/math/discrete-math/discrete-math-basics/dmbs/difference-equations/index.html index 75c54aad..a6b4f103 100644 --- a/math/discrete-math/discrete-math-basics/dmbs/difference-equations/index.html +++ b/math/discrete-math/discrete-math-basics/dmbs/difference-equations/index.html @@ -1 +1 @@ -离散数学基础:递推关系式 - 离散数学基础 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 思维导图

        递推关系式

        # 递推关系式相关概念

        # 定义

        用序列中某些前面的项ai,0i<n表示第nan的等式\text{用序列中某些前面的项}a_i,0\leq i\lt n\text{表示第}n\text{项}a_n\text{的等式}

        #

        序列{an}满足一个递推关系式,则{an}即为该递推关系式的一个解\text{序列}\{a_n\}\text{满足一个递推关系式,则}\{a_n\}\text{即为该递推关系式的一个解}

        # 封闭公式解

        序列{an}的第n项可以用不含序列中任何项的通项公式表达,则该通项公式称为该递推关系式的封闭公式解\begin{aligned} &\text{序列}\{a_n\}\text{的第n项可以用不含序列中任何项的通项公式表达,}\\ &\text{则该通项公式称为该递推关系式的封闭公式解} \end{aligned}

        # 常系数线性递推关系式的一般形式

        an=c1an1+c2an2+...+ckank+F(n)\begin{aligned} &a_n = c_1a_{n-1}+c_2a_{n-2}+...+c_ka_{n-k}+F(n) \end{aligned}

        F(n)=0则为齐次\text{若}F(n)=0\text{则为齐次}

        F(n)0则为非齐次\text{若}F(n)\neq 0\text{则为非齐次}

        # 线性齐次递推关系式的解

        # 特征方程

        xn=c1xn1+c2xn2+...+ckxnk+F(n)\begin{aligned} &x^n = c_1x^{n-1}+c_2x^{n-2}+...+c_kx^{n-k}+F(n) \end{aligned}

        # 特征方程的根

        r1,r2,...rt,重数分别为m1,m2,...m3,这门课没管复根情况咱就不背了捏\begin{aligned} r_1,r_2,...r_t,\text{重数分别为}m_1,m_2,...m_3,\text{这门课没管复根情况咱就不背了捏} \end{aligned}

        # 解的形式

        an=(β1,0+β1,1n+...+β1,m11nm11)r1n+(β2,0+β2,1n+...+β2,m21nm21)r2n+...+(βt,0+βt,1n+...+βt,mt1nmt1)rtn代入至少t个初始条件解出待定系数β1,β2,...βt即可得到解\begin{aligned} a_n &= (\beta_{1,0}+\beta_{1,1}n+...+\beta_{1,m_1-1}n^{m_1-1})r_1^{n} \\& +(\beta_{2,0}+\beta_{2,1}n+...+\beta_{2,m_2-1}n^{m_2-1})r_2^{n} \\& +...\\& +(\beta_{t,0}+\beta_{t,1}n+...+\beta_{t,m_t-1}n^{m_t-1})r_t^{n}\\& \text{代入至少t个初始条件解出待定系数}\beta_1,\beta_2,...\beta_t\text{即可得到解} \end{aligned}

        # 线性非齐次递推关系式的解

        # 伴随齐次递推关系式

        F(n)=0得到:\text{令}F(n)=0\text{得到:}

        an=c1an1+c2an2+...+ckank\begin{aligned} &a_n = c_1a_{n-1}+c_2a_{n-2}+...+c_ka_{n-k} \end{aligned}

        • 按照线性齐次递推关系式求解得到:

          • 通解,即伴随齐次递推关系式的解

        # F (n) 形式

        F(n)=(btnt+bt1nt1+...+b1n+b0)sn\begin{aligned} F(n)=(b_tn^t+b_{t-1}n^{t-1}+...+b_1n+b_0)s^n \end{aligned}

        这门课没管的别的情况咱就不背了捏

        # 特解的形式

        • s 不是伴随线性齐次递推关系式的特征方程的根

        an(p)=(ptnt+pt1nt1+...+p1n+p0)sn\begin{aligned} a_n^{(p)} = (p_tn^t+p_{t-1}n^{t-1}+...+p_1n+p_0)s^n \end{aligned}

        • s 是伴随线性齐次递推关系式的特征方程的 m 重根

        an(p)=nm(ptnt+pt1nt1+...+p1n+p0)sn\begin{aligned} a_n^{(p)} = n^m(p_tn^t+p_{t-1}n^{t-1}+...+p_1n+p_0)s^n \end{aligned}

        将特解代入递推关系式,对得到的关于n的多项式的等式两边比较系数,可以确定待定系数p0,p1,...pt\text{将特解代入递推关系式,对得到的关于n的多项式的等式两边比较系数,可以确定待定系数}p_0,p_1,...p_t

        # 解的形式

        an=通解+特解=通解+an(p)\begin{aligned} a_n = \text{通解}+\text{特解}=\text{通解}+a_n^{(p)} \end{aligned}

        # 算法效率

        分析主定理
        (Master Therorem)

        #

        f(n)是实递增函数,且满足递推关系式f(n)=af(nb)+Cnd,则:f(n){O(nd)a<bdO(ndlogn)a=bdO(nlogba)a>bda,bZ,a1,b>1,C,dR,C>0,d0\begin{aligned} &f(n)\text{是实递增函数,且满足递推关系式}f(n)=af(\frac{n}{b})+Cn^d\text{,则:}\\& \begin{aligned} f(n)是\left\{\begin{aligned} &O(n^d) & a\lt b^d\\ &O(n^dlogn) & a= b^d\\ &O(n^{log_ba})& a\gt b^d \end{aligned}\right. &&&&&& \begin{aligned} &a,b\in\mathbb{Z},a\geq1,b\gt1,\\ &C,d\in\mathbb{R},C\gt0,d\geq0 \end{aligned} \end{aligned} \end{aligned}

        \ No newline at end of file +离散数学基础:递推关系式 - 离散数学基础 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 思维导图

        递推关系式

        # 递推关系式相关概念

        # 定义

        用序列中某些前面的项ai,0i<n表示第nan的等式\text{用序列中某些前面的项}a_i,0\leq i\lt n\text{表示第}n\text{项}a_n\text{的等式}

        #

        序列{an}满足一个递推关系式,则{an}即为该递推关系式的一个解\text{序列}\{a_n\}\text{满足一个递推关系式,则}\{a_n\}\text{即为该递推关系式的一个解}

        # 封闭公式解

        序列{an}的第n项可以用不含序列中任何项的通项公式表达,则该通项公式称为该递推关系式的封闭公式解\begin{aligned} &\text{序列}\{a_n\}\text{的第n项可以用不含序列中任何项的通项公式表达,}\\ &\text{则该通项公式称为该递推关系式的封闭公式解} \end{aligned}

        # 常系数线性递推关系式的一般形式

        an=c1an1+c2an2+...+ckank+F(n)\begin{aligned} &a_n = c_1a_{n-1}+c_2a_{n-2}+...+c_ka_{n-k}+F(n) \end{aligned}

        F(n)=0则为齐次\text{若}F(n)=0\text{则为齐次}

        F(n)0则为非齐次\text{若}F(n)\neq 0\text{则为非齐次}

        # 线性齐次递推关系式的解

        # 特征方程

        xn=c1xn1+c2xn2+...+ckxnk+F(n)\begin{aligned} &x^n = c_1x^{n-1}+c_2x^{n-2}+...+c_kx^{n-k}+F(n) \end{aligned}

        # 特征方程的根

        r1,r2,...rt,重数分别为m1,m2,...m3,这门课没管复根情况咱就不背了捏\begin{aligned} r_1,r_2,...r_t,\text{重数分别为}m_1,m_2,...m_3,\text{这门课没管复根情况咱就不背了捏} \end{aligned}

        # 解的形式

        an=(β1,0+β1,1n+...+β1,m11nm11)r1n+(β2,0+β2,1n+...+β2,m21nm21)r2n+...+(βt,0+βt,1n+...+βt,mt1nmt1)rtn代入至少t个初始条件解出待定系数β1,β2,...βt即可得到解\begin{aligned} a_n &= (\beta_{1,0}+\beta_{1,1}n+...+\beta_{1,m_1-1}n^{m_1-1})r_1^{n} \\& +(\beta_{2,0}+\beta_{2,1}n+...+\beta_{2,m_2-1}n^{m_2-1})r_2^{n} \\& +...\\& +(\beta_{t,0}+\beta_{t,1}n+...+\beta_{t,m_t-1}n^{m_t-1})r_t^{n}\\& \text{代入至少t个初始条件解出待定系数}\beta_1,\beta_2,...\beta_t\text{即可得到解} \end{aligned}

        # 线性非齐次递推关系式的解

        # 伴随齐次递推关系式

        F(n)=0得到:\text{令}F(n)=0\text{得到:}

        an=c1an1+c2an2+...+ckank\begin{aligned} &a_n = c_1a_{n-1}+c_2a_{n-2}+...+c_ka_{n-k} \end{aligned}

        • 按照线性齐次递推关系式求解得到:

          • 通解,即伴随齐次递推关系式的解

        # F (n) 形式

        F(n)=(btnt+bt1nt1+...+b1n+b0)sn\begin{aligned} F(n)=(b_tn^t+b_{t-1}n^{t-1}+...+b_1n+b_0)s^n \end{aligned}

        这门课没管的别的情况咱就不背了捏

        # 特解的形式

        • s 不是伴随线性齐次递推关系式的特征方程的根

        an(p)=(ptnt+pt1nt1+...+p1n+p0)sn\begin{aligned} a_n^{(p)} = (p_tn^t+p_{t-1}n^{t-1}+...+p_1n+p_0)s^n \end{aligned}

        • s 是伴随线性齐次递推关系式的特征方程的 m 重根

        an(p)=nm(ptnt+pt1nt1+...+p1n+p0)sn\begin{aligned} a_n^{(p)} = n^m(p_tn^t+p_{t-1}n^{t-1}+...+p_1n+p_0)s^n \end{aligned}

        将特解代入递推关系式,对得到的关于n的多项式的等式两边比较系数,可以确定待定系数p0,p1,...pt\text{将特解代入递推关系式,对得到的关于n的多项式的等式两边比较系数,可以确定待定系数}p_0,p_1,...p_t

        # 解的形式

        an=通解+特解=通解+an(p)\begin{aligned} a_n = \text{通解}+\text{特解}=\text{通解}+a_n^{(p)} \end{aligned}

        # 算法效率

        分析主定理
        (Master Therorem)

        #

        f(n)是实递增函数,且满足递推关系式f(n)=af(nb)+Cnd,则:f(n){O(nd)a<bdO(ndlogn)a=bdO(nlogba)a>bda,bZ,a1,b>1,C,dR,C>0,d0\begin{aligned} &f(n)\text{是实递增函数,且满足递推关系式}f(n)=af(\frac{n}{b})+Cn^d\text{,则:}\\& \begin{aligned} f(n)是\left\{\begin{aligned} &O(n^d) & a\lt b^d\\ &O(n^dlogn) & a= b^d\\ &O(n^{log_ba})& a\gt b^d \end{aligned}\right. &&&&&& \begin{aligned} &a,b\in\mathbb{Z},a\geq1,b\gt1,\\ &C,d\in\mathbb{R},C\gt0,d\geq0 \end{aligned} \end{aligned} \end{aligned}

        \ No newline at end of file diff --git a/math/discrete-math/discrete-math-basics/dmbs/functions/index.html b/math/discrete-math/discrete-math-basics/dmbs/functions/index.html index e18467a7..e381a1c5 100644 --- a/math/discrete-math/discrete-math-basics/dmbs/functions/index.html +++ b/math/discrete-math/discrete-math-basics/dmbs/functions/index.html @@ -1 +1 @@ -离散数学基础:函数 - 离散数学基础 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        函数

        # 函数

        (全函数)

        # 描述

        集合A到B的函数,是A×笛卡尔积的一类子集集合A到A的函数也称为A上的函数定义域的每个元素都存在陪域的唯一元素与之对应\begin{aligned} &\text{集合A到B的函数,是}A\times \text{笛卡尔积的一类子集}\\ &\text{集合A到A的函数也称为A上的函数}\\ &\text{定义域的每个元素都存在陪域的唯一元素与之对应} \end{aligned}

        # 记号

        f:AB\begin{aligned} f:A\to B \end{aligned}

        # 定义

        fP(A×B)orfA×B,aA,!bB,<a,b>f\begin{aligned} &f\in P(A\times B) \ or\ f \subseteq A\times B ,\\ &\forall a\in A,\ \exists!b\in B,\ <a,b>\in f \end{aligned}

        #

        b是a在函数f下的像\text{b是a在函数f下的像}

        # 原像

        a是b在函数f下的原像\text{a是b在函数f下的原像}

        # 定义域 or 域

        AA

        # 陪域

        BB

        # 像集

        # 描述

        A的子集S在f下的像集,是f的陪域B的子集\text{A的子集S在f下的像集,是f的陪域B的子集}

        # 记号

        f(S)\begin{aligned} f(S) \end{aligned}

        # 定义

        f(S)={f(x)BxS}={yBxS,y=f(x)}B\begin{aligned} f(S)&=\{f(x)\in B|x\in S\}\\ &=\{y\in B|\exists x\in S,y=f(x)\}\\ &\subseteq B \end{aligned}

        # 值域

        f(A)=ran(f)\begin{aligned} f(A)=\boldsymbol{ran}(f) \end{aligned}

        # 逆像集

        # 描述

        B的子集T在f下的逆像集,是f的域A的子集\text{B的子集T在f下的逆像集,是f的域A的子集}

        # 记号

        f1(T)\begin{aligned} f^{-1}(T) \end{aligned}

        # 定义

        f1(T)={xAf(x)T}A\begin{aligned} f^{-1}(T)=\{x\in A|f(x)\in T\}\subseteq A \end{aligned}

        # 恒等函数

        # 描述

        任意非空集A上的恒等关系ΔA是A的恒等函数来自恒等关系\begin{aligned}&\text{任意非空集A上的恒等关系}\Delta_A\text{是A的恒等函数}\\& \text{来自恒等关系} \end{aligned}

        # 记号

        idA=ΔA\begin{aligned} \boldsymbol{id}_A=\Delta_A \end{aligned}

        # 定义

        f1(T)={xAf(x)T}A\begin{aligned} f^{-1}(T)=\{x\in A|f(x)\in T\}\subseteq A \end{aligned}

        # 空函数

        # 描述

        以空集为定义域的函数都是空函数,当函数陪域为空集时定义域也只能是空集,从而函数也是空函数,否则不满足函数定义来自空关系\begin{aligned}& \text{以空集为定义域的函数都是空函数,}\\& \text{当函数陪域为空集时定义域也只能是空集,}\\& \text{从而函数也是空函数,否则不满足函数定义}\\& \text{来自空关系} \end{aligned}

        # 定义

        f:AB,A=f是空函数B=A=f是空函数\begin{aligned} &f:A\to B,\\ &A=\varnothing\to f\text{是空函数}\\ &B=\varnothing\to A=\varnothing\to f\text{是空函数} \end{aligned}

        # 特征函数

        # 描述

        子集S的特征函数是全集U的任意子集S到2={0,1}的函数\begin{aligned} &\text{子集S的特征函数是}\\ &\text{全集U的任意子集S到}\boldsymbol{2}=\{\boldsymbol{0},\boldsymbol{1}\}\text{的函数} \end{aligned}

        # 记号

        χs:U2\begin{aligned} \chi_s :U\to \boldsymbol{2} \end{aligned}

        # 定义

        χs(x)={10xSx∉S\begin{aligned} \chi_s (x)=\left\{\begin{aligned} \boldsymbol{1}\\ \boldsymbol{0} \end{aligned}\right. &&&& \begin{aligned} x\in S\\ x\not\in S \end{aligned} \end{aligned}

        # 自然映射

        # 描述

        R是非空集A上的等价关系,商集A/R的自然映射是A到A/R的函数,将A的任意元素a映射到它所在的等价类[a]R\begin{aligned} &R\text{是非空集A上的等价关系,}\\ &\text{商集A/R的自然映射是A到A/R的函数},\\ &\text{将A的任意元素a映射到它所在的等价类}[a]_R \end{aligned}

        # 记号

        ρ:AA/R\begin{aligned} \rho :A\to A/R \end{aligned}

        # 定义

        ρ(a)=[a]R\begin{aligned} \rho(a)=[a]_R \end{aligned}

        # 函数集

        (函数族)

        # 描述

        所有从集合A到集合B的函数的集合\text{所有从集合A到集合B的函数的集合}

        # 记号

        BA\begin{aligned} B^A \end{aligned}

        # 偏函数

        # 描述

        与全函数的区别在于:定义域的每个元素至多存在陪域的一个元素与之对应\begin{aligned} &与全函数的区别在于:\\ &定义域的每个元素至多存在陪域的一个元素与之对应 \end{aligned}

        # 记号

        f:AB\begin{aligned} f:A\rightharpoonup B \end{aligned}

        # 定义

        fP(A×B)orfA×B,aA,(!bB,<a,b>f)(!bB,<a,b>f)\begin{aligned} &f\in P(A\times B) \ or\ f \subseteq A\times B ,\ \ \forall a\in A,\\& (\exists!b\in B,\ <a,b>\in f)\lor(!\exists b\in B,\ <a,b>\in f) \end{aligned}

        # 备注

        本门课程里的函数默认是全函数\text{本门课程里的函数默认是全函数}

        # 单函数

        (一对一函数)

        # 描述

        陪域的每个元素至多存在定义域的一个元素与之对应\begin{aligned}\text{陪域的每个元素至多存在定义域的一个元素与之对应} \end{aligned}

        # 定义

        x,yA,xyf(x)f(y)\begin{aligned} &\forall x,y\in A,\ \ x\neq y\to f(x)\neq f(y) \end{aligned}

        x,yA,f(x)=f(y)x=y\begin{aligned} &\forall x,y\in A,\ \ f(x)=f(y)\to x=y \end{aligned}

        f存在左逆\begin{aligned} f\text{存在左逆} \end{aligned}

        # 满函数

        (映上函数)

        # 描述

        陪域的每个元素至少存在定义域的一个元素与之对应\begin{aligned}\text{陪域的每个元素至少存在定义域的一个元素与之对应} \end{aligned}

        # 定义

        ran(f)=B\begin{aligned} \boldsymbol{ran}(f)=B \end{aligned}

        yB,xA,f(x)=y\begin{aligned} &\forall y\in B,\ \ \exists x\in A,\ \ f(x)=y \end{aligned}

        f存在右逆\begin{aligned} f\text{存在右逆} \end{aligned}

        # 双函数

        (一一对应)

        # 描述

        陪域的每个元素都有定义域唯一一个元素与之对应\begin{aligned} \text{陪域的每个元素都有定义域唯一一个元素与之对应} \end{aligned}

        # 定义

        f既是单函数又是满函数\begin{aligned} f\text{既是单函数又是满函数} \end{aligned}

        f:AB是双函数g:BA,gf=idA,fg=idBf1,f1f=idA,ff1=idB\begin{aligned}& f:A\to B\text{是双函数}\\\Leftrightarrow& \exists g:B\to A\ ,\ g\circ f=\boldsymbol{id}_A\ ,\ f\circ g=\boldsymbol{id}_B\\\Leftrightarrow& \exists f^{-1},f^{-1}\circ f=\boldsymbol{id}_A\ ,\ f\circ f^{-1}=\boldsymbol{id}_B \end{aligned}

        # 函数的复合

        # 描述

        函数f:ABg:BC的复合与关系的复合是一致的,是二元运算\begin{aligned} &\text{函数}f:A\to B\text{和}g:B\to C\text{的复合}\\& \text{与关系的复合是一致的,是二元运算} \end{aligned}

        # 记号

        (gf)(x)=g(f(x))\begin{aligned} (g\circ f)(x)=g(f(x)) \end{aligned}

        # 定义

        (gf)(x)=g(f(x))\begin{aligned} (g\circ f)(x)=g(f(x)) \end{aligned}

        gf={<x,y>z(<x,z>f<z,y>g)}={<x,y>z(f(x)=zg(z)=y)}={<x,y>g(f(x))=y}\begin{aligned} g\circ f&=\{<x,y>|\exists z(<x,z>\in f\land<z,y>\in g)\}\\ &=\{<x,y>|\exists z(f(x)=z\land g(z)=y)\}\\ &=\{<x,y>|g(f(x))=y\} \end{aligned}

        # 性质

        关系复合不满足交换律\begin{aligned}& 关系复合不满足交换律 \end{aligned}

        关系复合满足结合律\begin{aligned}&\text{关系复合满足结合律} \end{aligned}

        fg是单函数gf是单函数f是单函数\begin{aligned}& f\text{和}g\text{是单函数}\to g\circ f\text{是单函数}\to f\text{是单函数} \end{aligned}

        fg是满函数gf是满函数g是满函数\begin{aligned}& f\text{和}g\text{是满函数}\to g\circ f\text{是满函数}\to g\text{是满函数} \end{aligned}

        fg是双函数gf是双函数f是单函数且g是满函数\begin{aligned}& f\text{和}g\text{是双函数}\to g\circ f\text{是双函数}\to f\text{是单函数且}g\text{是满函数} \end{aligned}

        # 逆函数

        # 描述

        双函数f作为关系的逆关系f1f的逆函数\begin{aligned}\text{双函数}f\text{作为关系的逆关系}f^{-1}\text{是}f\text{的逆函数} \end{aligned}

        # 定义

        f1={<y,x>B×A<x,y>f}={<y,x>f(x)=y}\begin{aligned} f^{-1}& =\{<y,x>\in B\times A|<x,y>\in f\}\\& =\{<y,x>|f(x)=y\} \end{aligned}

        f1:BA是双函数f的逆函数f1f=idA,ff1=idB\begin{aligned}& f^{-1}:B\to A\text{是双函数}f\text{的逆函数}\\\Leftrightarrow& f^{-1}\circ f=\boldsymbol{id}_A\ ,\ f\circ f^{-1}=\boldsymbol{id}_B \end{aligned}

        # 左逆

        # 描述

        g:BA是函数f的左逆,满足:\begin{aligned} g:B\to A\text{是函数}f\text{的左逆,满足:} \end{aligned}

        # 定义

        gf=idA\begin{aligned} &g\circ f=\boldsymbol{id}_A \end{aligned}

        # 左逆

        # 描述

        g:BA是函数f的右逆,满足:\begin{aligned} g:B\to A\text{是函数}f\text{的右逆,满足:} \end{aligned}

        # 定义

        fg=idB\begin{aligned} &f\circ g=\boldsymbol{id}_B \end{aligned}

        \ No newline at end of file +离散数学基础:函数 - 离散数学基础 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        函数

        # 函数

        (全函数)

        # 描述

        集合A到B的函数,是A×笛卡尔积的一类子集集合A到A的函数也称为A上的函数定义域的每个元素都存在陪域的唯一元素与之对应\begin{aligned} &\text{集合A到B的函数,是}A\times \text{笛卡尔积的一类子集}\\ &\text{集合A到A的函数也称为A上的函数}\\ &\text{定义域的每个元素都存在陪域的唯一元素与之对应} \end{aligned}

        # 记号

        f:AB\begin{aligned} f:A\to B \end{aligned}

        # 定义

        fP(A×B)orfA×B,aA,!bB,<a,b>f\begin{aligned} &f\in P(A\times B) \ or\ f \subseteq A\times B ,\\ &\forall a\in A,\ \exists!b\in B,\ <a,b>\in f \end{aligned}

        #

        b是a在函数f下的像\text{b是a在函数f下的像}

        # 原像

        a是b在函数f下的原像\text{a是b在函数f下的原像}

        # 定义域 or 域

        AA

        # 陪域

        BB

        # 像集

        # 描述

        A的子集S在f下的像集,是f的陪域B的子集\text{A的子集S在f下的像集,是f的陪域B的子集}

        # 记号

        f(S)\begin{aligned} f(S) \end{aligned}

        # 定义

        f(S)={f(x)BxS}={yBxS,y=f(x)}B\begin{aligned} f(S)&=\{f(x)\in B|x\in S\}\\ &=\{y\in B|\exists x\in S,y=f(x)\}\\ &\subseteq B \end{aligned}

        # 值域

        f(A)=ran(f)\begin{aligned} f(A)=\boldsymbol{ran}(f) \end{aligned}

        # 逆像集

        # 描述

        B的子集T在f下的逆像集,是f的域A的子集\text{B的子集T在f下的逆像集,是f的域A的子集}

        # 记号

        f1(T)\begin{aligned} f^{-1}(T) \end{aligned}

        # 定义

        f1(T)={xAf(x)T}A\begin{aligned} f^{-1}(T)=\{x\in A|f(x)\in T\}\subseteq A \end{aligned}

        # 恒等函数

        # 描述

        任意非空集A上的恒等关系ΔA是A的恒等函数来自恒等关系\begin{aligned}&\text{任意非空集A上的恒等关系}\Delta_A\text{是A的恒等函数}\\& \text{来自恒等关系} \end{aligned}

        # 记号

        idA=ΔA\begin{aligned} \boldsymbol{id}_A=\Delta_A \end{aligned}

        # 定义

        f1(T)={xAf(x)T}A\begin{aligned} f^{-1}(T)=\{x\in A|f(x)\in T\}\subseteq A \end{aligned}

        # 空函数

        # 描述

        以空集为定义域的函数都是空函数,当函数陪域为空集时定义域也只能是空集,从而函数也是空函数,否则不满足函数定义来自空关系\begin{aligned}& \text{以空集为定义域的函数都是空函数,}\\& \text{当函数陪域为空集时定义域也只能是空集,}\\& \text{从而函数也是空函数,否则不满足函数定义}\\& \text{来自空关系} \end{aligned}

        # 定义

        f:AB,A=f是空函数B=A=f是空函数\begin{aligned} &f:A\to B,\\ &A=\varnothing\to f\text{是空函数}\\ &B=\varnothing\to A=\varnothing\to f\text{是空函数} \end{aligned}

        # 特征函数

        # 描述

        子集S的特征函数是全集U的任意子集S到2={0,1}的函数\begin{aligned} &\text{子集S的特征函数是}\\ &\text{全集U的任意子集S到}\boldsymbol{2}=\{\boldsymbol{0},\boldsymbol{1}\}\text{的函数} \end{aligned}

        # 记号

        χs:U2\begin{aligned} \chi_s :U\to \boldsymbol{2} \end{aligned}

        # 定义

        χs(x)={10xSx∉S\begin{aligned} \chi_s (x)=\left\{\begin{aligned} \boldsymbol{1}\\ \boldsymbol{0} \end{aligned}\right. &&&& \begin{aligned} x\in S\\ x\not\in S \end{aligned} \end{aligned}

        # 自然映射

        # 描述

        R是非空集A上的等价关系,商集A/R的自然映射是A到A/R的函数,将A的任意元素a映射到它所在的等价类[a]R\begin{aligned} &R\text{是非空集A上的等价关系,}\\ &\text{商集A/R的自然映射是A到A/R的函数},\\ &\text{将A的任意元素a映射到它所在的等价类}[a]_R \end{aligned}

        # 记号

        ρ:AA/R\begin{aligned} \rho :A\to A/R \end{aligned}

        # 定义

        ρ(a)=[a]R\begin{aligned} \rho(a)=[a]_R \end{aligned}

        # 函数集

        (函数族)

        # 描述

        所有从集合A到集合B的函数的集合\text{所有从集合A到集合B的函数的集合}

        # 记号

        BA\begin{aligned} B^A \end{aligned}

        # 偏函数

        # 描述

        与全函数的区别在于:定义域的每个元素至多存在陪域的一个元素与之对应\begin{aligned} &与全函数的区别在于:\\ &定义域的每个元素至多存在陪域的一个元素与之对应 \end{aligned}

        # 记号

        f:AB\begin{aligned} f:A\rightharpoonup B \end{aligned}

        # 定义

        fP(A×B)orfA×B,aA,(!bB,<a,b>f)(!bB,<a,b>f)\begin{aligned} &f\in P(A\times B) \ or\ f \subseteq A\times B ,\ \ \forall a\in A,\\& (\exists!b\in B,\ <a,b>\in f)\lor(!\exists b\in B,\ <a,b>\in f) \end{aligned}

        # 备注

        本门课程里的函数默认是全函数\text{本门课程里的函数默认是全函数}

        # 单函数

        (一对一函数)

        # 描述

        陪域的每个元素至多存在定义域的一个元素与之对应\begin{aligned}\text{陪域的每个元素至多存在定义域的一个元素与之对应} \end{aligned}

        # 定义

        x,yA,xyf(x)f(y)\begin{aligned} &\forall x,y\in A,\ \ x\neq y\to f(x)\neq f(y) \end{aligned}

        x,yA,f(x)=f(y)x=y\begin{aligned} &\forall x,y\in A,\ \ f(x)=f(y)\to x=y \end{aligned}

        f存在左逆\begin{aligned} f\text{存在左逆} \end{aligned}

        # 满函数

        (映上函数)

        # 描述

        陪域的每个元素至少存在定义域的一个元素与之对应\begin{aligned}\text{陪域的每个元素至少存在定义域的一个元素与之对应} \end{aligned}

        # 定义

        ran(f)=B\begin{aligned} \boldsymbol{ran}(f)=B \end{aligned}

        yB,xA,f(x)=y\begin{aligned} &\forall y\in B,\ \ \exists x\in A,\ \ f(x)=y \end{aligned}

        f存在右逆\begin{aligned} f\text{存在右逆} \end{aligned}

        # 双函数

        (一一对应)

        # 描述

        陪域的每个元素都有定义域唯一一个元素与之对应\begin{aligned} \text{陪域的每个元素都有定义域唯一一个元素与之对应} \end{aligned}

        # 定义

        f既是单函数又是满函数\begin{aligned} f\text{既是单函数又是满函数} \end{aligned}

        f:AB是双函数g:BA,gf=idA,fg=idBf1,f1f=idA,ff1=idB\begin{aligned}& f:A\to B\text{是双函数}\\\Leftrightarrow& \exists g:B\to A\ ,\ g\circ f=\boldsymbol{id}_A\ ,\ f\circ g=\boldsymbol{id}_B\\\Leftrightarrow& \exists f^{-1},f^{-1}\circ f=\boldsymbol{id}_A\ ,\ f\circ f^{-1}=\boldsymbol{id}_B \end{aligned}

        # 函数的复合

        # 描述

        函数f:ABg:BC的复合与关系的复合是一致的,是二元运算\begin{aligned} &\text{函数}f:A\to B\text{和}g:B\to C\text{的复合}\\& \text{与关系的复合是一致的,是二元运算} \end{aligned}

        # 记号

        (gf)(x)=g(f(x))\begin{aligned} (g\circ f)(x)=g(f(x)) \end{aligned}

        # 定义

        (gf)(x)=g(f(x))\begin{aligned} (g\circ f)(x)=g(f(x)) \end{aligned}

        gf={<x,y>z(<x,z>f<z,y>g)}={<x,y>z(f(x)=zg(z)=y)}={<x,y>g(f(x))=y}\begin{aligned} g\circ f&=\{<x,y>|\exists z(<x,z>\in f\land<z,y>\in g)\}\\ &=\{<x,y>|\exists z(f(x)=z\land g(z)=y)\}\\ &=\{<x,y>|g(f(x))=y\} \end{aligned}

        # 性质

        关系复合不满足交换律\begin{aligned}& 关系复合不满足交换律 \end{aligned}

        关系复合满足结合律\begin{aligned}&\text{关系复合满足结合律} \end{aligned}

        fg是单函数gf是单函数f是单函数\begin{aligned}& f\text{和}g\text{是单函数}\to g\circ f\text{是单函数}\to f\text{是单函数} \end{aligned}

        fg是满函数gf是满函数g是满函数\begin{aligned}& f\text{和}g\text{是满函数}\to g\circ f\text{是满函数}\to g\text{是满函数} \end{aligned}

        fg是双函数gf是双函数f是单函数且g是满函数\begin{aligned}& f\text{和}g\text{是双函数}\to g\circ f\text{是双函数}\to f\text{是单函数且}g\text{是满函数} \end{aligned}

        # 逆函数

        # 描述

        双函数f作为关系的逆关系f1f的逆函数\begin{aligned}\text{双函数}f\text{作为关系的逆关系}f^{-1}\text{是}f\text{的逆函数} \end{aligned}

        # 定义

        f1={<y,x>B×A<x,y>f}={<y,x>f(x)=y}\begin{aligned} f^{-1}& =\{<y,x>\in B\times A|<x,y>\in f\}\\& =\{<y,x>|f(x)=y\} \end{aligned}

        f1:BA是双函数f的逆函数f1f=idA,ff1=idB\begin{aligned}& f^{-1}:B\to A\text{是双函数}f\text{的逆函数}\\\Leftrightarrow& f^{-1}\circ f=\boldsymbol{id}_A\ ,\ f\circ f^{-1}=\boldsymbol{id}_B \end{aligned}

        # 左逆

        # 描述

        g:BA是函数f的左逆,满足:\begin{aligned} g:B\to A\text{是函数}f\text{的左逆,满足:} \end{aligned}

        # 定义

        gf=idA\begin{aligned} &g\circ f=\boldsymbol{id}_A \end{aligned}

        # 左逆

        # 描述

        g:BA是函数f的右逆,满足:\begin{aligned} g:B\to A\text{是函数}f\text{的右逆,满足:} \end{aligned}

        # 定义

        fg=idB\begin{aligned} &f\circ g=\boldsymbol{id}_B \end{aligned}

        \ No newline at end of file diff --git a/math/discrete-math/discrete-math-basics/dmbs/operations-of-sets-and-relations/index.html b/math/discrete-math/discrete-math-basics/dmbs/operations-of-sets-and-relations/index.html index 778f1889..1b851de9 100644 --- a/math/discrete-math/discrete-math-basics/dmbs/operations-of-sets-and-relations/index.html +++ b/math/discrete-math/discrete-math-basics/dmbs/operations-of-sets-and-relations/index.html @@ -1 +1 @@ -离散数学基础:集合与关系的运算 - 离散数学基础 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 思维导图

        集合与关系的运算

        #

        # 定义

        RS={<a,b><a,b>R<a,b>S}\begin{aligned}& R\cup S =\{<a,b>|<a,b>\in R\ \ \lor <a,b>\in S\}\end{aligned}

        # 关系矩阵

        MRS=MRMS=[rijsij]\begin{aligned}& M_{R\cup S} = M_R \lor M_S = [r_{ij}\lor s_{ij}]\end{aligned}

        # 性质

        单位元: 空关系\begin{aligned}&\text{单位元: 空关系}\end{aligned}

        零元:全关系\begin{aligned}&\text{零元:全关系}\end{aligned}

        关系并满足交换律\begin{aligned}&\text{关系并满足交换律}\end{aligned}

        关系并满足结合律\begin{aligned}&\text{关系并满足结合律}\end{aligned}

        关系并的单调性:关系并保持子集关系AB,CDABCDAC,BCABC\begin{aligned}\text{关系并的单调性:关系并保持子集关系}\\ A\subseteq B,C\subseteq D \Rightarrow A\cup B\subseteq C\cup D\\ A\subseteq C,B\subseteq C \Leftrightarrow A\cup B\subseteq C\end{aligned}

        关系并对关系交有分配律\begin{aligned}&\text{关系并对关系交有分配律}\end{aligned}

        关系并对关系差不满足分配律ABACA(BC)\begin{aligned}\text{关系并对关系差不满足分配律}\\ A\cup B-A\cup C\subseteq A\cup(B-C)\end{aligned}

        #

        # 定义

        RS={<a,b><a,b>R<a,b>S}\begin{aligned}& R\cap S =\{<a,b>|<a,b>\in R\ \ \land <a,b>\in S\}\end{aligned}

        # 关系矩阵

        MRS=MRMS=[rijsij]\begin{aligned}& M_{R\cap S} = M_R \land M_S = [r_{ij}\land s_{ij}]\end{aligned}

        # 性质

        单位元:全关系\begin{aligned}& 单位元:全关系 \end{aligned}

        零元: 空关系\begin{aligned}&\text{零元: 空关系}\end{aligned}

        关系交满足交换律\begin{aligned}&\text{关系交满足交换律}\end{aligned}

        关系交满足结合律\begin{aligned}&\text{关系交满足结合律}\end{aligned}

        关系交的单调性:关系交保持子集关系AB,CDABCDCA,CBCAB\begin{aligned}\text{关系交的单调性:关系交保持子集关系}\\ A\subseteq B,C\subseteq D \Rightarrow A\cap B\subseteq C\cap D\\ C\subseteq A,C\subseteq B \Leftrightarrow C\subseteq A\cup B\end{aligned}

        关系交对关系并有分配律\begin{aligned}&\text{关系交对关系并有分配律}\end{aligned}

        关系交对关系差有分配律\begin{aligned}\text{关系交对关系差有分配律}\end{aligned}

        #

        # 定义

        RS={<a,b><a,b>R<a,b>S}\begin{aligned}& R- S =\{<a,b>|<a,b>\in R\ \ \land <a,b>\notin S\}\end{aligned}

        # 关系矩阵

        MRS=MRMS=[rijsij]\begin{aligned}& M_{R- S} = M_R \ominus M_S = [r_{ij}\ominus s_{ij}]\end{aligned}

        运算00=01=11=010=1\begin{aligned}&\text{运算}\ominus:0\ominus0=0\ominus1=1\ominus1=0,1\ominus0=1\end{aligned}

        # 性质

        左零单位: 全关系\begin{aligned}&\text{左零单位: 全关系}\end{aligned}

        右单位元: 空关系\begin{aligned}&\text{右单位元: 空关系}\end{aligned}

        左零元: 空关系\begin{aligned}&\text{左零元: 空关系}\end{aligned}

        右零元: 全关系\begin{aligned}&\text{右零元: 全关系}\end{aligned}

        关系差不满足交换律\begin{aligned}&\text{关系差不满足交换律}\end{aligned}

        关系差不满足结合律\begin{aligned}&\text{关系差不满足结合律}\end{aligned}

        关系差无单调性:关系差不保持子集关系\begin{aligned}&\text{关系差无单调性:关系差不保持子集关系}\end{aligned}

        关系差对关系并没有分配律:A(BC)(AB)(AC)\begin{aligned}\text{关系差对关系并没有分配律}:\\ A-(B\cup C) \subseteq (A-B)\cup(A-C)\end{aligned}

        关系差对关系交没有分配律:(AB)(AC)A(BC)\begin{aligned}\text{关系差对关系交没有分配律}:\\ (A-B)\cap(A-C) \subseteq A-(B\cap C)\end{aligned}

        关系差对关系交、关系并有反分配律:A(BC)=(AB)(AC)A(BC)=(AB)(AC)\begin{aligned}\text{关系差对关系交、关系并有反分配律}:\\ A-(B\cup C) = (A-B)\cap(A-C)\\ A-(B\cap C) = (A-B)\cup(A-C)\end{aligned}

        # 复合

        # 定义

        SR={<a,c>bB(<a,b>R<b,c>S)}\begin{aligned}& S\circ R=\{<a,c>|\exists b\in B(<a,b>\in R\land <b,c>\in S)\}\end{aligned}

        # 关系矩阵

        MRS=MRMS\begin{aligned}& M_{R\circ S} = M_R \odot M_S\end{aligned}

        逻辑积运算:与矩阵乘法本质相同,只是将其中的加法、乘法分别用逻辑或(逻辑加法) 、逻辑与(逻辑减法)替代\begin{aligned}&\text{逻辑积运算}\odot\text{:与矩阵乘法本质相同,}\\& \text{只是将其中的加法、乘法分别用}\\& \text{逻辑或}\lor\text{(逻辑加法) 、逻辑与}\land\text{(逻辑减法)替代}\end{aligned}

        # 备注

        本课程采用逆序复合\begin{aligned}&\text{本课程采用逆序复合}\end{aligned}

        # 性质

        单位元:恒等关系\begin{aligned}&\text{单位元:恒等关系}\end{aligned}

        零元: 空关系\begin{aligned}&\text{零元: 空关系}\end{aligned}

        关系复合不满足交换律\begin{aligned}&\text{关系复合不满足交换律}\end{aligned}

        关系复合满足结合律\begin{aligned}&\text{关系复合满足结合律}\end{aligned}

        关系复合的单调性:关系复合保持子集关系RS,UWURWS\begin{aligned}&\text{关系复合的单调性:关系复合保持子集关系}\\& R\subseteq S, U\subseteq W \to U\circ R\subseteq W\circ S\end{aligned}

        关系复合对关系并有分配律\begin{aligned}&\text{关系复合对关系并有分配律}\end{aligned}

        关系复合对关系交没有分配律:(RS)T(RT)(ST)\begin{aligned}\text{关系复合对关系交没有分配律}:\\ (R\cap S)\circ T \subseteq (R\circ T)\cap(S\circ T)\end{aligned}

        #

        # 定义

        R={<a,b>A×B<a,b>R}=A×BR\begin{aligned} \overline R &=\{<a,b>\in A\times B|<a,b>\notin R\}\\& =A\times B-R\end{aligned}

        # 关系矩阵

        MR=MR=[rij]\begin{aligned}& M_{\overline R}=\overline{M_R}=[\overline{r_{ij}}]\end{aligned}

        # 性质

        关系补的单调性:关系补运算保持相反的子集关系\begin{aligned}&\text{关系补的单调性:关系补运算保持相反的子集关系}\end{aligned}

        关系补对关系并没有分配律:ABAB\begin{aligned}&\text{关系补对关系并没有分配律}: \overline{A\cup B} \subseteq \overline{A}\cup\overline{B}\end{aligned}

        关系补对关系交有分配律\begin{aligned}&\text{关系补对关系交有分配律}\end{aligned}

        关系补对关系差没有分配律:ABAB\begin{aligned}&\text{关系补对关系差没有分配律}: \overline{A}-\overline{B}\subseteq\overline{A-B}\end{aligned}

        关系补对关系复合有分配律\begin{aligned}&\text{关系补对关系复合有分配律}\end{aligned}

        关系逆与关系补有交换律(两个一元运算)\begin{aligned}&\text{关系逆与关系补有交换律(两个一元运算)}\end{aligned}

        #

        # 定义

        R1={<b,a><a,b>R}\begin{aligned}& R^{-1}=\{<b,a>|<a,b>\in R\}\end{aligned}

        # 关系矩阵

        MR1=(MR)T\begin{aligned}& M_{R^{-1}}=(M_R)^T\end{aligned}

        # 性质

        关系逆的单调性:关系逆运算保持子集关系\begin{aligned}&\text{关系逆的单调性:关系逆运算保持子集关系}\end{aligned}

        关系逆对关系并有分配律\begin{aligned}&\text{关系逆对关系并有分配律}\end{aligned}

        关系逆对关系交有分配律\begin{aligned}&\text{关系逆对关系交有分配律}\end{aligned}

        关系逆对关系差有分配律\begin{aligned}&\text{关系逆对关系差有分配律}\end{aligned}

        关系逆对关系复合有分配律\begin{aligned}&\text{关系逆对关系复合有分配律}\end{aligned}

        关系逆与关系补有交换律(两个一元运算)\begin{aligned}&\text{关系逆与关系补有交换律(两个一元运算)}\end{aligned}

        \ No newline at end of file +离散数学基础:集合与关系的运算 - 离散数学基础 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 思维导图

        集合与关系的运算

        #

        # 定义

        RS={<a,b><a,b>R<a,b>S}\begin{aligned}& R\cup S =\{<a,b>|<a,b>\in R\ \ \lor <a,b>\in S\}\end{aligned}

        # 关系矩阵

        MRS=MRMS=[rijsij]\begin{aligned}& M_{R\cup S} = M_R \lor M_S = [r_{ij}\lor s_{ij}]\end{aligned}

        # 性质

        单位元: 空关系\begin{aligned}&\text{单位元: 空关系}\end{aligned}

        零元:全关系\begin{aligned}&\text{零元:全关系}\end{aligned}

        关系并满足交换律\begin{aligned}&\text{关系并满足交换律}\end{aligned}

        关系并满足结合律\begin{aligned}&\text{关系并满足结合律}\end{aligned}

        关系并的单调性:关系并保持子集关系AB,CDABCDAC,BCABC\begin{aligned}\text{关系并的单调性:关系并保持子集关系}\\ A\subseteq B,C\subseteq D \Rightarrow A\cup B\subseteq C\cup D\\ A\subseteq C,B\subseteq C \Leftrightarrow A\cup B\subseteq C\end{aligned}

        关系并对关系交有分配律\begin{aligned}&\text{关系并对关系交有分配律}\end{aligned}

        关系并对关系差不满足分配律ABACA(BC)\begin{aligned}\text{关系并对关系差不满足分配律}\\ A\cup B-A\cup C\subseteq A\cup(B-C)\end{aligned}

        #

        # 定义

        RS={<a,b><a,b>R<a,b>S}\begin{aligned}& R\cap S =\{<a,b>|<a,b>\in R\ \ \land <a,b>\in S\}\end{aligned}

        # 关系矩阵

        MRS=MRMS=[rijsij]\begin{aligned}& M_{R\cap S} = M_R \land M_S = [r_{ij}\land s_{ij}]\end{aligned}

        # 性质

        单位元:全关系\begin{aligned}& 单位元:全关系 \end{aligned}

        零元: 空关系\begin{aligned}&\text{零元: 空关系}\end{aligned}

        关系交满足交换律\begin{aligned}&\text{关系交满足交换律}\end{aligned}

        关系交满足结合律\begin{aligned}&\text{关系交满足结合律}\end{aligned}

        关系交的单调性:关系交保持子集关系AB,CDABCDCA,CBCAB\begin{aligned}\text{关系交的单调性:关系交保持子集关系}\\ A\subseteq B,C\subseteq D \Rightarrow A\cap B\subseteq C\cap D\\ C\subseteq A,C\subseteq B \Leftrightarrow C\subseteq A\cup B\end{aligned}

        关系交对关系并有分配律\begin{aligned}&\text{关系交对关系并有分配律}\end{aligned}

        关系交对关系差有分配律\begin{aligned}\text{关系交对关系差有分配律}\end{aligned}

        #

        # 定义

        RS={<a,b><a,b>R<a,b>S}\begin{aligned}& R- S =\{<a,b>|<a,b>\in R\ \ \land <a,b>\notin S\}\end{aligned}

        # 关系矩阵

        MRS=MRMS=[rijsij]\begin{aligned}& M_{R- S} = M_R \ominus M_S = [r_{ij}\ominus s_{ij}]\end{aligned}

        运算00=01=11=010=1\begin{aligned}&\text{运算}\ominus:0\ominus0=0\ominus1=1\ominus1=0,1\ominus0=1\end{aligned}

        # 性质

        左零单位: 全关系\begin{aligned}&\text{左零单位: 全关系}\end{aligned}

        右单位元: 空关系\begin{aligned}&\text{右单位元: 空关系}\end{aligned}

        左零元: 空关系\begin{aligned}&\text{左零元: 空关系}\end{aligned}

        右零元: 全关系\begin{aligned}&\text{右零元: 全关系}\end{aligned}

        关系差不满足交换律\begin{aligned}&\text{关系差不满足交换律}\end{aligned}

        关系差不满足结合律\begin{aligned}&\text{关系差不满足结合律}\end{aligned}

        关系差无单调性:关系差不保持子集关系\begin{aligned}&\text{关系差无单调性:关系差不保持子集关系}\end{aligned}

        关系差对关系并没有分配律:A(BC)(AB)(AC)\begin{aligned}\text{关系差对关系并没有分配律}:\\ A-(B\cup C) \subseteq (A-B)\cup(A-C)\end{aligned}

        关系差对关系交没有分配律:(AB)(AC)A(BC)\begin{aligned}\text{关系差对关系交没有分配律}:\\ (A-B)\cap(A-C) \subseteq A-(B\cap C)\end{aligned}

        关系差对关系交、关系并有反分配律:A(BC)=(AB)(AC)A(BC)=(AB)(AC)\begin{aligned}\text{关系差对关系交、关系并有反分配律}:\\ A-(B\cup C) = (A-B)\cap(A-C)\\ A-(B\cap C) = (A-B)\cup(A-C)\end{aligned}

        # 复合

        # 定义

        SR={<a,c>bB(<a,b>R<b,c>S)}\begin{aligned}& S\circ R=\{<a,c>|\exists b\in B(<a,b>\in R\land <b,c>\in S)\}\end{aligned}

        # 关系矩阵

        MRS=MRMS\begin{aligned}& M_{R\circ S} = M_R \odot M_S\end{aligned}

        逻辑积运算:与矩阵乘法本质相同,只是将其中的加法、乘法分别用逻辑或(逻辑加法) 、逻辑与(逻辑减法)替代\begin{aligned}&\text{逻辑积运算}\odot\text{:与矩阵乘法本质相同,}\\& \text{只是将其中的加法、乘法分别用}\\& \text{逻辑或}\lor\text{(逻辑加法) 、逻辑与}\land\text{(逻辑减法)替代}\end{aligned}

        # 备注

        本课程采用逆序复合\begin{aligned}&\text{本课程采用逆序复合}\end{aligned}

        # 性质

        单位元:恒等关系\begin{aligned}&\text{单位元:恒等关系}\end{aligned}

        零元: 空关系\begin{aligned}&\text{零元: 空关系}\end{aligned}

        关系复合不满足交换律\begin{aligned}&\text{关系复合不满足交换律}\end{aligned}

        关系复合满足结合律\begin{aligned}&\text{关系复合满足结合律}\end{aligned}

        关系复合的单调性:关系复合保持子集关系RS,UWURWS\begin{aligned}&\text{关系复合的单调性:关系复合保持子集关系}\\& R\subseteq S, U\subseteq W \to U\circ R\subseteq W\circ S\end{aligned}

        关系复合对关系并有分配律\begin{aligned}&\text{关系复合对关系并有分配律}\end{aligned}

        关系复合对关系交没有分配律:(RS)T(RT)(ST)\begin{aligned}\text{关系复合对关系交没有分配律}:\\ (R\cap S)\circ T \subseteq (R\circ T)\cap(S\circ T)\end{aligned}

        #

        # 定义

        R={<a,b>A×B<a,b>R}=A×BR\begin{aligned} \overline R &=\{<a,b>\in A\times B|<a,b>\notin R\}\\& =A\times B-R\end{aligned}

        # 关系矩阵

        MR=MR=[rij]\begin{aligned}& M_{\overline R}=\overline{M_R}=[\overline{r_{ij}}]\end{aligned}

        # 性质

        关系补的单调性:关系补运算保持相反的子集关系\begin{aligned}&\text{关系补的单调性:关系补运算保持相反的子集关系}\end{aligned}

        关系补对关系并没有分配律:ABAB\begin{aligned}&\text{关系补对关系并没有分配律}: \overline{A\cup B} \subseteq \overline{A}\cup\overline{B}\end{aligned}

        关系补对关系交有分配律\begin{aligned}&\text{关系补对关系交有分配律}\end{aligned}

        关系补对关系差没有分配律:ABAB\begin{aligned}&\text{关系补对关系差没有分配律}: \overline{A}-\overline{B}\subseteq\overline{A-B}\end{aligned}

        关系补对关系复合有分配律\begin{aligned}&\text{关系补对关系复合有分配律}\end{aligned}

        关系逆与关系补有交换律(两个一元运算)\begin{aligned}&\text{关系逆与关系补有交换律(两个一元运算)}\end{aligned}

        #

        # 定义

        R1={<b,a><a,b>R}\begin{aligned}& R^{-1}=\{<b,a>|<a,b>\in R\}\end{aligned}

        # 关系矩阵

        MR1=(MR)T\begin{aligned}& M_{R^{-1}}=(M_R)^T\end{aligned}

        # 性质

        关系逆的单调性:关系逆运算保持子集关系\begin{aligned}&\text{关系逆的单调性:关系逆运算保持子集关系}\end{aligned}

        关系逆对关系并有分配律\begin{aligned}&\text{关系逆对关系并有分配律}\end{aligned}

        关系逆对关系交有分配律\begin{aligned}&\text{关系逆对关系交有分配律}\end{aligned}

        关系逆对关系差有分配律\begin{aligned}&\text{关系逆对关系差有分配律}\end{aligned}

        关系逆对关系复合有分配律\begin{aligned}&\text{关系逆对关系复合有分配律}\end{aligned}

        关系逆与关系补有交换律(两个一元运算)\begin{aligned}&\text{关系逆与关系补有交换律(两个一元运算)}\end{aligned}

        \ No newline at end of file diff --git a/math/discrete-math/discrete-math-basics/dmbs/permutaion-and-combination/index.html b/math/discrete-math/discrete-math-basics/dmbs/permutaion-and-combination/index.html index 63a83cb2..2f696ec0 100644 --- a/math/discrete-math/discrete-math-basics/dmbs/permutaion-and-combination/index.html +++ b/math/discrete-math/discrete-math-basics/dmbs/permutaion-and-combination/index.html @@ -1 +1 @@ -离散数学基础:排列与组合 - 离散数学基础 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 思维导图

        排列组合计数问题

        # 排列

        # 记号

        P(n,r)=Pnr=AnrP(n,r)=P_n^r=A_n^r

        # 描述

        • n 个 (可区别的) 物体

          • n 个物体的 r - 排列

            • n 个物体的 n - 排列,或 n 个物体的全排列
        • 集合 S,|S|=n

          • S 的 r - 排列

            • S 的 n - 排列,或 S 的全排列

        # 公式

        P(n,r)=n!(nr)!P(n,r)=\frac{n!}{(n-r)!}

        P(n,n)=n!P(n,n)=n!

        P(n,r)={n!(nr)!,0r<nn!,r=n0,r>nP(n,r)=\left \{ \begin{aligned} & \frac{n!}{(n-r)!} &&,& 0\leq r\lt n \\ &n! &&,&r=n\\ & 0 &&,& r\gt n \end{aligned} \right.

        # 组合

        # 记号

        • 通常将第三个称为二项式系数

        C(n,r)=Cnr=(nr)C(n,r)=C_n^r=\begin{pmatrix}n\\r \end{pmatrix}

        # 描述

        • n 个 (可区别的) 物体

          • n 个物体的 r - 组合
        • 集合 S,|S|=n

          • S 的 r - 组合
        • 长度为 n 的二进制串

          • 长度为 n 且含 r 个 1 (或 0) 的二进制串数

        # 公式

        C(n,r)=C(r,n)=P(n,r)P(r,r)={n!r!(nr)!,0rn0,r>nC(n,r)=C(r,n)=\frac{P(n,r)}{P(r,r)}=\left \{ \begin{aligned} & \frac{n!}{r!(n-r)!} &,& 0\leq r\leq n \\ & 0 &,& r\gt n \end{aligned} \right.

        # 组合证明

        (不严谨)

        # 双计数证明

        • 论证等式两边是针对同一集合元素的不同计数方法

        # 双函数证明

        • 论证等式两边虽然是针对两个不同的集合的元素进行计数,但这两个集合之间存在双函数

        # 代数证明

        (严谨)

        # 利用数学归纳法、组合数、排列数等计算公式的证明

        # 二项式定理

        #

        (x+y)n=k=0n(nk)xkynk(x+y)^n=\sum_{k=0}^{n} \begin{pmatrix}n\\k\end{pmatrix}x^k y^{n-k}

        # 二项式定理的组合数推论

        #

        k=0n(nk)=2n\sum_{k=0}^{n}\begin{pmatrix}n\\k\end{pmatrix} =2^n

        # 帕斯卡等式

        #

        (nk)=(n1k)+(n1k1)\begin{pmatrix}n\\k \end{pmatrix} =\begin{pmatrix}n-1\\k \end{pmatrix} +\begin{pmatrix}n-1\\k-1 \end{pmatrix}

        # 递推式

        #

        r(nr)=(nr+1)(nr1)1r<nr\begin{pmatrix}n\\r\end{pmatrix} =(n-r+1)\begin{pmatrix}n\\r-1 \end{pmatrix} ,1\leq r\lt n

        #

        r(nr)=n(n1r1)1r<nr\begin{pmatrix}n\\r\end{pmatrix} =n\begin{pmatrix}n-1\\r-1 \end{pmatrix} ,1\leq r\lt n

        # 乘积化简式

        #

        (nm)(mk)=(nk)(nkmk)kmn\begin{pmatrix}n\\m\end{pmatrix} \begin{pmatrix}m\\k\end{pmatrix} =\begin{pmatrix}n\\k\end{pmatrix} \begin{pmatrix}n-k\\m-k\end{pmatrix} ,k\leq m\leq n

        # 上下标求和式

        朱世杰恒等式

        # Hockeystick 等式

        (m+r+1r)=k=0r(m+km)=(mm)+(m+1m)+...+(m+rm)=k=0r(m+kk)=(m0)+(m+11)+...+(m+rr)\begin{aligned} &\begin{pmatrix}m+r+1\\r\end{pmatrix}\\ =&\sum_{k=0}^{r} \begin{pmatrix}m+k\\m\end{pmatrix} =\begin{pmatrix}m\\m\end{pmatrix} +\begin{pmatrix}m+1\\m\end{pmatrix} +... +\begin{pmatrix}m+r\\m\end{pmatrix}\\ =&\sum_{k=0}^{r} \begin{pmatrix}m+k\\k\end{pmatrix} =\begin{pmatrix}m\\0\end{pmatrix} +\begin{pmatrix}m+1\\1\end{pmatrix} +... +\begin{pmatrix}m+r\\r\end{pmatrix} \end{aligned}

        # 令 n=r+1 可得到上面 Hockeystick 等式

        k=0n(km)=(0m)+(1m)+...+(nm)=(n+1m+1)\sum_{k=0}^{n} \begin{pmatrix}k\\m\end{pmatrix} =\begin{pmatrix}0\\m\end{pmatrix} +\begin{pmatrix}1\\m\end{pmatrix} +... +\begin{pmatrix}n\\m\end{pmatrix} =\begin{pmatrix}n+1\\m+1\end{pmatrix}

        # 朱世杰 - 范德蒙等式

        #

        k=0r(mrk)(nk)=(m+nr)=(mr)(n0)+(mr1)(n1)+...+(m1)(nr1)+(m0)(nr)\begin{aligned} & \sum_{k=0}^{r} \begin{pmatrix}m\\r-k\end{pmatrix} \begin{pmatrix}n\\k\end{pmatrix} =\begin{pmatrix}m+n\\r\end{pmatrix} \\& =\begin{pmatrix}m\\r\end{pmatrix} \begin{pmatrix}n\\0\end{pmatrix} +\begin{pmatrix}m\\r-1\end{pmatrix} \begin{pmatrix}n\\1\end{pmatrix} +... +\begin{pmatrix}m\\1\end{pmatrix} \begin{pmatrix}n\\r-1\end{pmatrix} +\begin{pmatrix}m\\0\end{pmatrix} \begin{pmatrix}n\\r\end{pmatrix} \end{aligned}

        \ No newline at end of file +离散数学基础:排列与组合 - 离散数学基础 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 思维导图

        排列组合计数问题

        # 排列

        # 记号

        P(n,r)=Pnr=AnrP(n,r)=P_n^r=A_n^r

        # 描述

        • n 个 (可区别的) 物体

          • n 个物体的 r - 排列

            • n 个物体的 n - 排列,或 n 个物体的全排列
        • 集合 S,|S|=n

          • S 的 r - 排列

            • S 的 n - 排列,或 S 的全排列

        # 公式

        P(n,r)=n!(nr)!P(n,r)=\frac{n!}{(n-r)!}

        P(n,n)=n!P(n,n)=n!

        P(n,r)={n!(nr)!,0r<nn!,r=n0,r>nP(n,r)=\left \{ \begin{aligned} & \frac{n!}{(n-r)!} &&,& 0\leq r\lt n \\ &n! &&,&r=n\\ & 0 &&,& r\gt n \end{aligned} \right.

        # 组合

        # 记号

        • 通常将第三个称为二项式系数

        C(n,r)=Cnr=(nr)C(n,r)=C_n^r=\begin{pmatrix}n\\r \end{pmatrix}

        # 描述

        • n 个 (可区别的) 物体

          • n 个物体的 r - 组合
        • 集合 S,|S|=n

          • S 的 r - 组合
        • 长度为 n 的二进制串

          • 长度为 n 且含 r 个 1 (或 0) 的二进制串数

        # 公式

        C(n,r)=C(r,n)=P(n,r)P(r,r)={n!r!(nr)!,0rn0,r>nC(n,r)=C(r,n)=\frac{P(n,r)}{P(r,r)}=\left \{ \begin{aligned} & \frac{n!}{r!(n-r)!} &,& 0\leq r\leq n \\ & 0 &,& r\gt n \end{aligned} \right.

        # 组合证明

        (不严谨)

        # 双计数证明

        • 论证等式两边是针对同一集合元素的不同计数方法

        # 双函数证明

        • 论证等式两边虽然是针对两个不同的集合的元素进行计数,但这两个集合之间存在双函数

        # 代数证明

        (严谨)

        # 利用数学归纳法、组合数、排列数等计算公式的证明

        # 二项式定理

        #

        (x+y)n=k=0n(nk)xkynk(x+y)^n=\sum_{k=0}^{n} \begin{pmatrix}n\\k\end{pmatrix}x^k y^{n-k}

        # 二项式定理的组合数推论

        #

        k=0n(nk)=2n\sum_{k=0}^{n}\begin{pmatrix}n\\k\end{pmatrix} =2^n

        # 帕斯卡等式

        #

        (nk)=(n1k)+(n1k1)\begin{pmatrix}n\\k \end{pmatrix} =\begin{pmatrix}n-1\\k \end{pmatrix} +\begin{pmatrix}n-1\\k-1 \end{pmatrix}

        # 递推式

        #

        r(nr)=(nr+1)(nr1)1r<nr\begin{pmatrix}n\\r\end{pmatrix} =(n-r+1)\begin{pmatrix}n\\r-1 \end{pmatrix} ,1\leq r\lt n

        #

        r(nr)=n(n1r1)1r<nr\begin{pmatrix}n\\r\end{pmatrix} =n\begin{pmatrix}n-1\\r-1 \end{pmatrix} ,1\leq r\lt n

        # 乘积化简式

        #

        (nm)(mk)=(nk)(nkmk)kmn\begin{pmatrix}n\\m\end{pmatrix} \begin{pmatrix}m\\k\end{pmatrix} =\begin{pmatrix}n\\k\end{pmatrix} \begin{pmatrix}n-k\\m-k\end{pmatrix} ,k\leq m\leq n

        # 上下标求和式

        朱世杰恒等式

        # Hockeystick 等式

        (m+r+1r)=k=0r(m+km)=(mm)+(m+1m)+...+(m+rm)=k=0r(m+kk)=(m0)+(m+11)+...+(m+rr)\begin{aligned} &\begin{pmatrix}m+r+1\\r\end{pmatrix}\\ =&\sum_{k=0}^{r} \begin{pmatrix}m+k\\m\end{pmatrix} =\begin{pmatrix}m\\m\end{pmatrix} +\begin{pmatrix}m+1\\m\end{pmatrix} +... +\begin{pmatrix}m+r\\m\end{pmatrix}\\ =&\sum_{k=0}^{r} \begin{pmatrix}m+k\\k\end{pmatrix} =\begin{pmatrix}m\\0\end{pmatrix} +\begin{pmatrix}m+1\\1\end{pmatrix} +... +\begin{pmatrix}m+r\\r\end{pmatrix} \end{aligned}

        # 令 n=r+1 可得到上面 Hockeystick 等式

        k=0n(km)=(0m)+(1m)+...+(nm)=(n+1m+1)\sum_{k=0}^{n} \begin{pmatrix}k\\m\end{pmatrix} =\begin{pmatrix}0\\m\end{pmatrix} +\begin{pmatrix}1\\m\end{pmatrix} +... +\begin{pmatrix}n\\m\end{pmatrix} =\begin{pmatrix}n+1\\m+1\end{pmatrix}

        # 朱世杰 - 范德蒙等式

        #

        k=0r(mrk)(nk)=(m+nr)=(mr)(n0)+(mr1)(n1)+...+(m1)(nr1)+(m0)(nr)\begin{aligned} & \sum_{k=0}^{r} \begin{pmatrix}m\\r-k\end{pmatrix} \begin{pmatrix}n\\k\end{pmatrix} =\begin{pmatrix}m+n\\r\end{pmatrix} \\& =\begin{pmatrix}m\\r\end{pmatrix} \begin{pmatrix}n\\0\end{pmatrix} +\begin{pmatrix}m\\r-1\end{pmatrix} \begin{pmatrix}n\\1\end{pmatrix} +... +\begin{pmatrix}m\\1\end{pmatrix} \begin{pmatrix}n\\r-1\end{pmatrix} +\begin{pmatrix}m\\0\end{pmatrix} \begin{pmatrix}n\\r\end{pmatrix} \end{aligned}

        \ No newline at end of file diff --git a/math/discrete-math/discrete-math-basics/dmbs/principles-of-counting/index.html b/math/discrete-math/discrete-math-basics/dmbs/principles-of-counting/index.html index ce62d600..43bdd17c 100644 --- a/math/discrete-math/discrete-math-basics/dmbs/principles-of-counting/index.html +++ b/math/discrete-math/discrete-math-basics/dmbs/principles-of-counting/index.html @@ -1 +1 @@ -离散数学基础:组合计数基本原理 - 离散数学基础 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 思维导图

        组合计数基本原理

        # 常用引理

        # 引理 1

        # 公式

        AB=AB=AAB| A\overline{B} |=| A-B | = |A| -| A B|

        AB=AB=AABA\overline{B} = A - B = A - A B

        # 证明

        # 容斥原理

        # 2 维

        • 公式

        AB=A+BAB| A \cup B | = |A|+|B|- |A \cap B|

        • 变形

        AB=UAB=UAB+AB| \overline{A} \cap \overline{B} | = |U|-|A\cup B| = |U|- |A| - |B| + |A \cap B|

        • 证明

          • 利用 加法原理 和 常用引理
        \begin{align} |A \cup B| &= |A-B|+|B-A|+|A \cup B|\\ &= |A|-|A \cap B|+|B|-|A \cap B|+|A \cap B|\\ &= |A|+|B|- |A \cap B|\end{align}

        # n 维

        • 公式

          • 奇加偶减
        • 证明

          • 数学归纳法

        # 鸽笼原理

        # 鸽笼原理 / 抽屉原理

        # 公式

        • k+1 或更多只鸽子放在 k 个鸽笼里,则至少有 1 个鸽笼里有两只或更多只鸽子

        # 证明

        • 反证法

        # 推论

        若集合AB都是有穷集合,fAB有:若集合A、B都是有穷集合,\forall f\in A^B有:

        A>B,则f不是单函数|A|\gt|B|,则f不是单函数

        A<B,则f不是满函数|A|\lt|B|\text{,则}f\text{不是满函数}

        A=B,则f是双函数|A|=|B|\text{,则}f\text{是双函数}

        # 广义鸽笼原理

        # 公式

        N个物体放到k个盒子里,至少有1个盒子至少有Nk个物体N\text{个物体放到}k\text{个盒子里,至少有}1\text{个盒子至少有}\left\lceil \frac{N}{k} \right\rceil\text{个物体}

        # 证明

        用反证法,设每个盒子的物体都少于Nk个,令N=kq+r0r<k,分情况考虑:1r>0,则Nk=q+1则每个盒子的物体都少于q+1个,即每个盒子的物体少于等于q个,从而Nkq<kq+1,这与N=kq+1矛盾2r=0,则Nk=q则每个盒子的物体都少于q个,即每个盒子的物体少于等于q1个,从而Nk(q1)=kqk<kq+1,这与N=kq+1矛盾\begin{aligned}&\text{用反证法,设每个盒子的物体都少于}\left\lceil \frac{N}{k} \right\rceil\text{个,令}N=kq+r,0 \le r\lt k\text{,分情况考虑:}\\& \begin{aligned} (1)&r\gt0\text{,则}\left\lceil \frac{N}{k} \right\rceil = q+1\\& \text{则每个盒子的物体都少于}q+1\text{个,即每个盒子的物体少于等于}q\text{个,} \text{从而}N\le kq\lt kq+1\text{,这与}N=kq+1\text{矛盾} \end{aligned}\\& \begin{aligned} (2)&r=0,则 \left\lceil \frac{N}{k} \right\rceil = q\\& \text{则每个盒子的物体都少于}q\text{个,即每个盒子的物体少于等于}q-1\text{个,从而}N\le k(q-1)=kq-k\lt kq+1\text{,这与}N=kq+1\text{矛盾} \end{aligned} \end{aligned}

        # 推论

        # N 个物体放到 k 个盒子,至少有一个盒子有至少几个物体

        最小容量=物体总数盒子数\text{最小容量}=\left\lceil \frac{\text{物体总数}}{\text{盒子数}} \right\rceil

        n=Nkn=\left\lceil \frac{N}{k} \right\rceil

        # 至少需要几个物体放到 k 个盒子里能保证至少有一个盒子有至少 n 个物体

        物体总数(最小容量1)×盒子数+1\text{物体总数}\ge (\text{最小容量}-1)\times\text{盒子数}+1

        N(n1)×k+1N\ge (n-1)\times k+1

        # N 个物体放到至多几个盒子能保证至少有一个盒子有至少 n 个物体

        盒子数物体总数1最小容量1盒子数\le \left\lfloor \frac{\text{物体总数}-1}{\text{最小容量}-1} \right\rfloor

        kN1n1k\le\left\lfloor \frac{N-1}{n-1} \right\rfloor

        # 题型

        证明有连续若干个容器恰好技巧:构造sk=i=1kai作为容器\text{证明有连续若干个容器恰好…… 技巧:构造}s_k=\sum^k_{i=1}{a_i}\text{作为容器}

        # 拉姆齐 (Ramsey) 定理

        # 最简单、最经典形式

        R(3,3)

        # 公式

        # 解释

        任意6个人中,必定有\text{任意}6\text{个人中,必定有}

        要么3个人互相认识\text{要么}3\text{个人互相认识}

        要么3个人互相不认识\text{要么}3\text{个人互相不认识}

        # 图论解释

        对一个6阶完全图,用红、黑两种颜色任意对它的边上色,必定存在\text{对一个}6\text{阶完全图,用红、黑两种颜色任意对它的边上色,必定存在}

        要么一个红色三角形(3阶完全图)\text{要么一个红色三角形(}3\text{阶完全图)}

        要么一个黑色三角形(3阶完全图)\text{要么一个黑色三角形(}3\text{阶完全图)}

        # 证明

        在6个人中任选1人,设为a,将剩下5人分为两个集合:a不认识的人、a认识的人,并分别记为E和F根据鸽笼原理,E和F至少有一个集合至少有52=3分情况考虑:(1)若F中有至少3人(i)若F里的人互相都不认识,则可从F中选3个人得到3个人互相不认识,命题成立(ii)否则,F中至少有2个人是相互认识的,则这这两个人与a构成了3人互相认识,命题成立(2)若E中有至少3人(i)若E里的人互相都认识,则可从E中选3个人得到3个人互相认识,命题成立(ii)否则,F中至少有2个人是相互不认识的,则这这两个人与a构成了3人互相不认识,命题成立综上总是有命题成立\begin{aligned}& \text{在6个人中任选1人,设为a,将剩下5人分为两个集合:a不认识的人、a认识的人,并分别记为E和F} \text{根据鸽笼原理,E和F至少有一个集合至少有}\lceil \frac{5}{2}\rceil=3 \text{人}\\& \text{分情况考虑:}\\& \begin{aligned} \text{(1)}&\text{若F中有至少3人}\\& \begin{aligned} \text{(i)}&\text{若F里的人互相都不认识,则可从F中选3个人得到3个人互相不认识,命题成立} \end{aligned}\\& \begin{aligned} \text{(ii)}&\text{否则,F中至少有2个人是相互认识的,则这这两个人与a构成了3人互相认识,命题成立} \end{aligned} \end{aligned}\\& \begin{aligned} \text{(2)}&\text{若E中有至少3人}\\& \begin{aligned} \text{(i)}&\text{若E里的人互相都认识,则可从E中选3个人得到3个人互相认识,命题成立} \end{aligned}\\& \begin{aligned} \text{(ii)}&\text{否则,F中至少有2个人是相互不认识的,则这这两个人与a构成了3人互相不认识,命题成立} \end{aligned} \end{aligned}\\& \text{综上总是有命题成立} \end{aligned}

        # 一般形式

        R(m,n)

        # 公式

        m,nZ+,m>2,n>2,R(m,n)Z+,使\forall m,n\in\mathbb{Z^+},m\gt 2,n\gt 2,\exists R(m,n)\in \mathbb{Z^+},使

        # 解释

        R(m,n)个人中,必定有R(m,n)\text{个人中,必定有}

        要么m个人互相认识或n个人互相不认识\text{要么m个人互相认识或n个人互相不认识}

        要么n个人互相认识或m个人互相不认识\text{要么n个人互相认识或m个人互相不认识}

        # 图论解释

        对一个R(m,n)阶完全图,用两种颜色任意对它的边上色,必定存在\text{对一个}R(m,n)\text{阶完全图,用两种颜色任意对它的边上色,必定存在}

        要么一个同色m阶完全子图\text{要么一个同色m阶完全子图}

        要么一个同色n阶完全子图\text{要么一个同色n阶完全子图}

        \ No newline at end of file +离散数学基础:组合计数基本原理 - 离散数学基础 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 思维导图

        组合计数基本原理

        # 常用引理

        # 引理 1

        # 公式

        AB=AB=AAB| A\overline{B} |=| A-B | = |A| -| A B|

        AB=AB=AABA\overline{B} = A - B = A - A B

        # 证明

        # 容斥原理

        # 2 维

        • 公式

        AB=A+BAB| A \cup B | = |A|+|B|- |A \cap B|

        • 变形

        AB=UAB=UAB+AB| \overline{A} \cap \overline{B} | = |U|-|A\cup B| = |U|- |A| - |B| + |A \cap B|

        • 证明

          • 利用 加法原理 和 常用引理
        \begin{align} |A \cup B| &= |A-B|+|B-A|+|A \cup B|\\ &= |A|-|A \cap B|+|B|-|A \cap B|+|A \cap B|\\ &= |A|+|B|- |A \cap B|\end{align}

        # n 维

        • 公式

          • 奇加偶减
        • 证明

          • 数学归纳法

        # 鸽笼原理

        # 鸽笼原理 / 抽屉原理

        # 公式

        • k+1 或更多只鸽子放在 k 个鸽笼里,则至少有 1 个鸽笼里有两只或更多只鸽子

        # 证明

        • 反证法

        # 推论

        若集合AB都是有穷集合,fAB有:若集合A、B都是有穷集合,\forall f\in A^B有:

        A>B,则f不是单函数|A|\gt|B|,则f不是单函数

        A<B,则f不是满函数|A|\lt|B|\text{,则}f\text{不是满函数}

        A=B,则f是双函数|A|=|B|\text{,则}f\text{是双函数}

        # 广义鸽笼原理

        # 公式

        N个物体放到k个盒子里,至少有1个盒子至少有Nk个物体N\text{个物体放到}k\text{个盒子里,至少有}1\text{个盒子至少有}\left\lceil \frac{N}{k} \right\rceil\text{个物体}

        # 证明

        用反证法,设每个盒子的物体都少于Nk个,令N=kq+r0r<k,分情况考虑:1r>0,则Nk=q+1则每个盒子的物体都少于q+1个,即每个盒子的物体少于等于q个,从而Nkq<kq+1,这与N=kq+1矛盾2r=0,则Nk=q则每个盒子的物体都少于q个,即每个盒子的物体少于等于q1个,从而Nk(q1)=kqk<kq+1,这与N=kq+1矛盾\begin{aligned}&\text{用反证法,设每个盒子的物体都少于}\left\lceil \frac{N}{k} \right\rceil\text{个,令}N=kq+r,0 \le r\lt k\text{,分情况考虑:}\\& \begin{aligned} (1)&r\gt0\text{,则}\left\lceil \frac{N}{k} \right\rceil = q+1\\& \text{则每个盒子的物体都少于}q+1\text{个,即每个盒子的物体少于等于}q\text{个,} \text{从而}N\le kq\lt kq+1\text{,这与}N=kq+1\text{矛盾} \end{aligned}\\& \begin{aligned} (2)&r=0,则 \left\lceil \frac{N}{k} \right\rceil = q\\& \text{则每个盒子的物体都少于}q\text{个,即每个盒子的物体少于等于}q-1\text{个,从而}N\le k(q-1)=kq-k\lt kq+1\text{,这与}N=kq+1\text{矛盾} \end{aligned} \end{aligned}

        # 推论

        # N 个物体放到 k 个盒子,至少有一个盒子有至少几个物体

        最小容量=物体总数盒子数\text{最小容量}=\left\lceil \frac{\text{物体总数}}{\text{盒子数}} \right\rceil

        n=Nkn=\left\lceil \frac{N}{k} \right\rceil

        # 至少需要几个物体放到 k 个盒子里能保证至少有一个盒子有至少 n 个物体

        物体总数(最小容量1)×盒子数+1\text{物体总数}\ge (\text{最小容量}-1)\times\text{盒子数}+1

        N(n1)×k+1N\ge (n-1)\times k+1

        # N 个物体放到至多几个盒子能保证至少有一个盒子有至少 n 个物体

        盒子数物体总数1最小容量1盒子数\le \left\lfloor \frac{\text{物体总数}-1}{\text{最小容量}-1} \right\rfloor

        kN1n1k\le\left\lfloor \frac{N-1}{n-1} \right\rfloor

        # 题型

        证明有连续若干个容器恰好技巧:构造sk=i=1kai作为容器\text{证明有连续若干个容器恰好…… 技巧:构造}s_k=\sum^k_{i=1}{a_i}\text{作为容器}

        # 拉姆齐 (Ramsey) 定理

        # 最简单、最经典形式

        R(3,3)

        # 公式

        # 解释

        任意6个人中,必定有\text{任意}6\text{个人中,必定有}

        要么3个人互相认识\text{要么}3\text{个人互相认识}

        要么3个人互相不认识\text{要么}3\text{个人互相不认识}

        # 图论解释

        对一个6阶完全图,用红、黑两种颜色任意对它的边上色,必定存在\text{对一个}6\text{阶完全图,用红、黑两种颜色任意对它的边上色,必定存在}

        要么一个红色三角形(3阶完全图)\text{要么一个红色三角形(}3\text{阶完全图)}

        要么一个黑色三角形(3阶完全图)\text{要么一个黑色三角形(}3\text{阶完全图)}

        # 证明

        在6个人中任选1人,设为a,将剩下5人分为两个集合:a不认识的人、a认识的人,并分别记为E和F根据鸽笼原理,E和F至少有一个集合至少有52=3分情况考虑:(1)若F中有至少3人(i)若F里的人互相都不认识,则可从F中选3个人得到3个人互相不认识,命题成立(ii)否则,F中至少有2个人是相互认识的,则这这两个人与a构成了3人互相认识,命题成立(2)若E中有至少3人(i)若E里的人互相都认识,则可从E中选3个人得到3个人互相认识,命题成立(ii)否则,F中至少有2个人是相互不认识的,则这这两个人与a构成了3人互相不认识,命题成立综上总是有命题成立\begin{aligned}& \text{在6个人中任选1人,设为a,将剩下5人分为两个集合:a不认识的人、a认识的人,并分别记为E和F} \text{根据鸽笼原理,E和F至少有一个集合至少有}\lceil \frac{5}{2}\rceil=3 \text{人}\\& \text{分情况考虑:}\\& \begin{aligned} \text{(1)}&\text{若F中有至少3人}\\& \begin{aligned} \text{(i)}&\text{若F里的人互相都不认识,则可从F中选3个人得到3个人互相不认识,命题成立} \end{aligned}\\& \begin{aligned} \text{(ii)}&\text{否则,F中至少有2个人是相互认识的,则这这两个人与a构成了3人互相认识,命题成立} \end{aligned} \end{aligned}\\& \begin{aligned} \text{(2)}&\text{若E中有至少3人}\\& \begin{aligned} \text{(i)}&\text{若E里的人互相都认识,则可从E中选3个人得到3个人互相认识,命题成立} \end{aligned}\\& \begin{aligned} \text{(ii)}&\text{否则,F中至少有2个人是相互不认识的,则这这两个人与a构成了3人互相不认识,命题成立} \end{aligned} \end{aligned}\\& \text{综上总是有命题成立} \end{aligned}

        # 一般形式

        R(m,n)

        # 公式

        m,nZ+,m>2,n>2,R(m,n)Z+,使\forall m,n\in\mathbb{Z^+},m\gt 2,n\gt 2,\exists R(m,n)\in \mathbb{Z^+},使

        # 解释

        R(m,n)个人中,必定有R(m,n)\text{个人中,必定有}

        要么m个人互相认识或n个人互相不认识\text{要么m个人互相认识或n个人互相不认识}

        要么n个人互相认识或m个人互相不认识\text{要么n个人互相认识或m个人互相不认识}

        # 图论解释

        对一个R(m,n)阶完全图,用两种颜色任意对它的边上色,必定存在\text{对一个}R(m,n)\text{阶完全图,用两种颜色任意对它的边上色,必定存在}

        要么一个同色m阶完全子图\text{要么一个同色m阶完全子图}

        要么一个同色n阶完全子图\text{要么一个同色n阶完全子图}

        \ No newline at end of file diff --git a/math/discrete-math/discrete-math-basics/dmbs/relations/index.html b/math/discrete-math/discrete-math-basics/dmbs/relations/index.html index ddc3cfbc..2ffd31a5 100644 --- a/math/discrete-math/discrete-math-basics/dmbs/relations/index.html +++ b/math/discrete-math/discrete-math-basics/dmbs/relations/index.html @@ -1 +1 @@ -离散数学基础:关系 - 离散数学基础 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 思维导图

        关系

        # 笛卡尔积

        # 定义

        A×B={a,baAbB}\begin{aligned}& A\times B=\{\langle a,b\rangle|a\in A\land b\in B\} \end{aligned}

        # 性质

        笛卡尔积运算不满足交换律\begin{aligned}& \text{笛卡尔积运算不满足交换律} \end{aligned}

        笛卡尔积运算不满足结合律\begin{aligned}& \text{笛卡尔积运算不满足结合律} \end{aligned}

        笛卡尔积运算对集合交有分配律\begin{aligned}& \text{笛卡尔积运算对集合交有分配律} \end{aligned}

        笛卡尔积运算对集合并有分配律\begin{aligned}& \text{笛卡尔积运算对集合并有分配律} \end{aligned}

        # 二元关系

        # 描述

        集合AB的二元关系是笛卡尔积A×B的子集\begin{aligned}& \text{集合}A\text{到}B\text{的二元关系是}\\& \text{笛卡尔积}A\times B\text{的子集} \end{aligned}

        集合A上的二元关系是集合A到自己的二元关系\begin{aligned}& \text{集合}A\text{上的二元关系是}\\& \text{集合A到自己的二元关系} \end{aligned}

        # 定义

        RA×B\begin{aligned}& R\subseteq A\times B \end{aligned}

        RA×A\begin{aligned}& R\subseteq A\times A \end{aligned}

        # 描述

        ab有关系R\begin{aligned}& a\text{和}b\text{有关系}R \end{aligned}

        aRb\begin{aligned}& a\ R\ b \end{aligned}

        a,bR\begin{aligned}& \langle a,b\rangle\in R \end{aligned}

        ab没有关系R\begin{aligned}& a\text{和}b\text{没有关系}R \end{aligned}

        ab\begin{aligned}& a\ \not R\ b \end{aligned}

        a,b∉R\begin{aligned}& \langle a,b\rangle\not\in R \end{aligned}

        备注:本课程的关系只考察二元关系

        # 空关系

        # 定义

        \begin{aligned}& \varnothing \end{aligned}

        # 全关系

        # 定义

        A×B\begin{aligned}& A\times B \end{aligned}

        # 恒等关系

        or
        对角关系

        # 定义

        ΔA={<a,a>aA},A\begin{aligned}& \Delta_A=\{<a,a>|a\in A\},\ \ A\neq \varnothing \end{aligned}

        # 关系矩阵

        单位矩阵\begin{aligned}& 单位矩阵 \end{aligned}

        # 自反关系

        # 定义

        aA,<a,a>R\begin{aligned}& \forall a\in A,\ <a,a>\in R \end{aligned}

        # 关系矩阵

        ΔAR\begin{aligned}& \Delta_A\subseteq R \end{aligned}

        # 反自反关系

        # 定义

        aA,<a,a>∉R\begin{aligned}& \forall a\in A,\ <a,a>\not\in R \end{aligned}

        # 关系矩阵

        ΔAR=\begin{aligned}& \Delta_A \cap R=\varnothing \end{aligned}

        # 对称关系

        # 定义

        a,bA,a,bR<b,a>R\begin{aligned}& \forall a,b\in A,\ \langle a,b\rangle\in R\to<b,a>\in R \end{aligned}

        # 关系矩阵

        R=R1\begin{aligned}& R=R^{-1} \end{aligned}

        # 反对称关系

        # 定义

        a,bA,(a,bR)(<b,a>R)a=b\begin{aligned}& \forall a,b\in A,\ (\langle a,b\rangle\in R)\land(<b,a>\in R)\to a=b \end{aligned}

        # 关系矩阵

        RR1ΔA\begin{aligned}& R\cap R^{-1}\subseteq\Delta_A \end{aligned}

        # 备注

        反对称关系一定是对称关系\begin{aligned}& 反对称关系一定是对称关系 \end{aligned}

        # 传递关系

        # 定义

        a,b,cA,(a,bR)(<b,c>R)<a,c>R\begin{aligned}& \forall a,b,c\in A,\ (\langle a,b\rangle\in R)\land(<b,c>\in R)\to <a,c>\in R \end{aligned}

        # 关系矩阵

        RR1R\begin{aligned}& R\circ R^{-1}\subseteq R \end{aligned}

        # 等价关系

        # 定义

        R是自反的、对称的、传递的\begin{aligned}& R\text{是自反的、对称的、传递的} \end{aligned}

        # 等价类

        • 描述

        a所在的等价关系R的等价类,简称a的等价类\begin{aligned}& a\text{所在的等价关系}R\text{的等价类,简称}a\text{的等价类} \end{aligned}

        • 定义

        [a]R={xA<a,x>R}\begin{aligned}& [a]_R=\{x\in A|<a,x>\in R\} \end{aligned}

        # 代表

        b[a]R,则称b为等价类[a]R的一个代表\begin{aligned}& b\in [a]_R\text{,则称}b\text{为等价类}[a]_R\text{的一个代表} \end{aligned}

        # 商集

        • 描述

        集合A关于等价关系R的商集是集合A的所有等价类构成的集合\begin{aligned}& \text{集合}A\text{关于等价关系R的商集是}\\& \text{集合A的所有等价类构成的集合} \end{aligned}

        • 定义

        A/R={[a]RaA}\begin{aligned}& A/R=\{[a]_R|a\in A\} \end{aligned}

        • 基本性质

        商集是一个集合族且是A的划分其中每一个元素都是等价类每个等价类都是这个划分的一个划分块\begin{aligned}& \text{商集是一个集合族且是}A\text{的划分}\\& \text{其中每一个元素都是等价类}\\& \text{每个等价类都是这个划分的一个划分块} \end{aligned}

        # 偏序关系

        # 定义

        R是自反的、反对称的、传递的\begin{aligned}& R\text{是自反的、反对称的、传递的} \end{aligned}

        # 符号

        \begin{aligned}& \preceq \end{aligned}

        # 经典例子

        • 数集上的 小于或等于关系、大于或等于关系

          • 集合的子集关系

            • 整数集上的整除关系

        # 可比

        (ab)(ba)\begin{aligned}& (a\preceq b) \lor (b\preceq a) \end{aligned}

        # 不可比

        (a⪯̸b)(b⪯̸a)\begin{aligned}& (a\not\preceq b) \land (b\not\preceq a) \end{aligned}

        # 覆盖

        b覆盖a(ab)!c(accb)\begin{aligned}& b\text{覆盖}a\Leftrightarrow(a\preceq b)\land!\exists c(a\preceq c\land c\preceq b) \end{aligned}

        # 极大元

        aA的极大元bA(abb=a)\begin{aligned}& a\text{是}A\text{的极大元}\Leftrightarrow \forall b\in A(a\preceq b\to b=a) \end{aligned}

        # 极小元

        aA的极小元bA(bab=a)\begin{aligned}& a\text{是}A\text{的极小元}\Leftrightarrow \forall b\in A(b\preceq a\to b=a) \end{aligned}

        # 最大元

        aA的最大元bA(ba)\begin{aligned}& a\text{是}A\text{的最大元}\Leftrightarrow \forall b\in A(b\preceq a) \end{aligned}

        # 最小元

        aA的最小元bA(ab)\begin{aligned}& a\text{是}A\text{的最小元}\Leftrightarrow \forall b\in A(a\preceq b) \end{aligned}

        # 偏序集

        # 定义

        (A,)orabbreviatedasA\begin{aligned}& (A,\preceq)\ \ or\ \ abbreviated\ as\ A \end{aligned}

        # 全序

        or
        线序

        # 定义

        偏序集A的任意两个元素都可比,则这个偏序集是全序或线序\begin{aligned}&\text{偏序集A的任意两个元素都可比,则这个偏序集是全序或线序} \end{aligned}

        # 关系举例 - 幂集上的:

        # 真包含关系

        • 反自反、反对称、传递

        # 恒等关系

        • 自反、对称、反对称、传递

        # 全关系

        • 自反、对称、传递

        # 空关系

        • 反自反、对称、反对称、传递
        \ No newline at end of file +离散数学基础:关系 - 离散数学基础 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 思维导图

        关系

        # 笛卡尔积

        # 定义

        A×B={a,baAbB}\begin{aligned}& A\times B=\{\langle a,b\rangle|a\in A\land b\in B\} \end{aligned}

        # 性质

        笛卡尔积运算不满足交换律\begin{aligned}& \text{笛卡尔积运算不满足交换律} \end{aligned}

        笛卡尔积运算不满足结合律\begin{aligned}& \text{笛卡尔积运算不满足结合律} \end{aligned}

        笛卡尔积运算对集合交有分配律\begin{aligned}& \text{笛卡尔积运算对集合交有分配律} \end{aligned}

        笛卡尔积运算对集合并有分配律\begin{aligned}& \text{笛卡尔积运算对集合并有分配律} \end{aligned}

        # 二元关系

        # 描述

        集合AB的二元关系是笛卡尔积A×B的子集\begin{aligned}& \text{集合}A\text{到}B\text{的二元关系是}\\& \text{笛卡尔积}A\times B\text{的子集} \end{aligned}

        集合A上的二元关系是集合A到自己的二元关系\begin{aligned}& \text{集合}A\text{上的二元关系是}\\& \text{集合A到自己的二元关系} \end{aligned}

        # 定义

        RA×B\begin{aligned}& R\subseteq A\times B \end{aligned}

        RA×A\begin{aligned}& R\subseteq A\times A \end{aligned}

        # 描述

        ab有关系R\begin{aligned}& a\text{和}b\text{有关系}R \end{aligned}

        aRb\begin{aligned}& a\ R\ b \end{aligned}

        a,bR\begin{aligned}& \langle a,b\rangle\in R \end{aligned}

        ab没有关系R\begin{aligned}& a\text{和}b\text{没有关系}R \end{aligned}

        ab\begin{aligned}& a\ \not R\ b \end{aligned}

        a,b∉R\begin{aligned}& \langle a,b\rangle\not\in R \end{aligned}

        备注:本课程的关系只考察二元关系

        # 空关系

        # 定义

        \begin{aligned}& \varnothing \end{aligned}

        # 全关系

        # 定义

        A×B\begin{aligned}& A\times B \end{aligned}

        # 恒等关系

        or
        对角关系

        # 定义

        ΔA={<a,a>aA},A\begin{aligned}& \Delta_A=\{<a,a>|a\in A\},\ \ A\neq \varnothing \end{aligned}

        # 关系矩阵

        单位矩阵\begin{aligned}& 单位矩阵 \end{aligned}

        # 自反关系

        # 定义

        aA,<a,a>R\begin{aligned}& \forall a\in A,\ <a,a>\in R \end{aligned}

        # 关系矩阵

        ΔAR\begin{aligned}& \Delta_A\subseteq R \end{aligned}

        # 反自反关系

        # 定义

        aA,<a,a>∉R\begin{aligned}& \forall a\in A,\ <a,a>\not\in R \end{aligned}

        # 关系矩阵

        ΔAR=\begin{aligned}& \Delta_A \cap R=\varnothing \end{aligned}

        # 对称关系

        # 定义

        a,bA,a,bR<b,a>R\begin{aligned}& \forall a,b\in A,\ \langle a,b\rangle\in R\to<b,a>\in R \end{aligned}

        # 关系矩阵

        R=R1\begin{aligned}& R=R^{-1} \end{aligned}

        # 反对称关系

        # 定义

        a,bA,(a,bR)(<b,a>R)a=b\begin{aligned}& \forall a,b\in A,\ (\langle a,b\rangle\in R)\land(<b,a>\in R)\to a=b \end{aligned}

        # 关系矩阵

        RR1ΔA\begin{aligned}& R\cap R^{-1}\subseteq\Delta_A \end{aligned}

        # 备注

        反对称关系一定是对称关系\begin{aligned}& 反对称关系一定是对称关系 \end{aligned}

        # 传递关系

        # 定义

        a,b,cA,(a,bR)(<b,c>R)<a,c>R\begin{aligned}& \forall a,b,c\in A,\ (\langle a,b\rangle\in R)\land(<b,c>\in R)\to <a,c>\in R \end{aligned}

        # 关系矩阵

        RR1R\begin{aligned}& R\circ R^{-1}\subseteq R \end{aligned}

        # 等价关系

        # 定义

        R是自反的、对称的、传递的\begin{aligned}& R\text{是自反的、对称的、传递的} \end{aligned}

        # 等价类

        • 描述

        a所在的等价关系R的等价类,简称a的等价类\begin{aligned}& a\text{所在的等价关系}R\text{的等价类,简称}a\text{的等价类} \end{aligned}

        • 定义

        [a]R={xA<a,x>R}\begin{aligned}& [a]_R=\{x\in A|<a,x>\in R\} \end{aligned}

        # 代表

        b[a]R,则称b为等价类[a]R的一个代表\begin{aligned}& b\in [a]_R\text{,则称}b\text{为等价类}[a]_R\text{的一个代表} \end{aligned}

        # 商集

        • 描述

        集合A关于等价关系R的商集是集合A的所有等价类构成的集合\begin{aligned}& \text{集合}A\text{关于等价关系R的商集是}\\& \text{集合A的所有等价类构成的集合} \end{aligned}

        • 定义

        A/R={[a]RaA}\begin{aligned}& A/R=\{[a]_R|a\in A\} \end{aligned}

        • 基本性质

        商集是一个集合族且是A的划分其中每一个元素都是等价类每个等价类都是这个划分的一个划分块\begin{aligned}& \text{商集是一个集合族且是}A\text{的划分}\\& \text{其中每一个元素都是等价类}\\& \text{每个等价类都是这个划分的一个划分块} \end{aligned}

        # 偏序关系

        # 定义

        R是自反的、反对称的、传递的\begin{aligned}& R\text{是自反的、反对称的、传递的} \end{aligned}

        # 符号

        \begin{aligned}& \preceq \end{aligned}

        # 经典例子

        • 数集上的 小于或等于关系、大于或等于关系

          • 集合的子集关系

            • 整数集上的整除关系

        # 可比

        (ab)(ba)\begin{aligned}& (a\preceq b) \lor (b\preceq a) \end{aligned}

        # 不可比

        (a⪯̸b)(b⪯̸a)\begin{aligned}& (a\not\preceq b) \land (b\not\preceq a) \end{aligned}

        # 覆盖

        b覆盖a(ab)!c(accb)\begin{aligned}& b\text{覆盖}a\Leftrightarrow(a\preceq b)\land!\exists c(a\preceq c\land c\preceq b) \end{aligned}

        # 极大元

        aA的极大元bA(abb=a)\begin{aligned}& a\text{是}A\text{的极大元}\Leftrightarrow \forall b\in A(a\preceq b\to b=a) \end{aligned}

        # 极小元

        aA的极小元bA(bab=a)\begin{aligned}& a\text{是}A\text{的极小元}\Leftrightarrow \forall b\in A(b\preceq a\to b=a) \end{aligned}

        # 最大元

        aA的最大元bA(ba)\begin{aligned}& a\text{是}A\text{的最大元}\Leftrightarrow \forall b\in A(b\preceq a) \end{aligned}

        # 最小元

        aA的最小元bA(ab)\begin{aligned}& a\text{是}A\text{的最小元}\Leftrightarrow \forall b\in A(a\preceq b) \end{aligned}

        # 偏序集

        # 定义

        (A,)orabbreviatedasA\begin{aligned}& (A,\preceq)\ \ or\ \ abbreviated\ as\ A \end{aligned}

        # 全序

        or
        线序

        # 定义

        偏序集A的任意两个元素都可比,则这个偏序集是全序或线序\begin{aligned}&\text{偏序集A的任意两个元素都可比,则这个偏序集是全序或线序} \end{aligned}

        # 关系举例 - 幂集上的:

        # 真包含关系

        • 反自反、反对称、传递

        # 恒等关系

        • 自反、对称、反对称、传递

        # 全关系

        • 自反、对称、传递

        # 空关系

        • 反自反、对称、反对称、传递
        \ No newline at end of file diff --git a/math/discrete-math/discrete-math-basics/problem1/index.html b/math/discrete-math/discrete-math-basics/problem1/index.html index 3147b5e9..e37f807f 100644 --- a/math/discrete-math/discrete-math-basics/problem1/index.html +++ b/math/discrete-math/discrete-math-basics/problem1/index.html @@ -1 +1 @@ -等价关系经典计数问题 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 题面

        f:ABf:A\to B 是函数,定义 A 上的关系 R,a,bA\forall a,b \in A,a R b 当且仅当 f (a) = f (b) 。

        证明 R 是等价关系,并给出它的等价类和商集,求这样的等价关系有多少个?

        # 证明 R 是等价关系

        显然 R 是等价关系,因为对∀a, b, c\in A,f (a)=f (a), f (a)=f (b) 蕴涵 f (b) = f (a) , f (a) = f (b) 且 f (b) = f (c) 蕴涵 f (a) = f (c) ,即 R 是自反、对称、传递的。

        # 求等价类

        对任意 a\in A,[a]R={xf(x)=f(a)},A/R={[a]RaA}[a]_R= \{x∣f(x)=f(a) \},A/R=\{[a]_R∣a\in A\}

        实际上,对任意 a\in A,若 f (a)=y\in B,则[a]R=f1(y)[a]_R=f^{-1} (y)

        因此若 f 是满函数,则 A/R={f1({b})bB}A/R=\{f^{-1} (\{b\})∣b\in B\},即商集 A/R 中的等价类与 B 的元素一一对应。

        对集合 A 上的任意等价关系 R ,自然映射ρ:AA/R,ρ(a)=[a]Rρ:A→A/R,ρ(a)=[a]_R 是满函数,因此 A 上的等价关系与以 A 为定义域的满函数对应。

        设 |A|=n ,则 A 上有 m 个等价类的等价关系个数等于 A 到 Zm={0,1,,m1}Z_m=\{0, 1, ⋯, m-1\} 的满函数个数除以 m! (在 Zm 的元素作为原像的所有可能排列中选一个即可)。

        # 求等价关系个数

        设 |A|=n ,则 A 上不同的等价关系有多少个?

        等价于:设 | A|=n,则 A 上含 m 个等价类的等价关系有多少个?

        当 | A|=n, |B|=m,n≥m,A 到 B 的满函数个数是:

        mnC(m,1)(m1)n++(1)kC(m,k)(mk)n++(1)m1C(m,m1)1nm^n- C(m, 1)(m-1)^n+ ⋯+ (-1)^k C(m, k) (m-k)^n+ ⋯+ (-1)^{m-1} C(m,m-1)\cdot 1^n

        因此 n 元素集合 A 上有 m 个等价类的等价关系有:

        B(n,m)=k=0m1(1)kC(m,k)(mk)nm!B(n, m)=\frac{\sum_{k=0}^{m-1}(-1)^k C(m,k) (m-k)^n }{m!}

        从而 n 元素集合 A 上的不同等价关系个数有:

        B(n)=m=1nk=0m1(1)kC(m,k)(mk)nm!B(n)=\sum_{m=1}^n\frac{\sum_{k=0}^{m-1}(-1)^k C(m,k) (m-k)^n}{m!}

        用 B (n) 表示 n 元素集合上不同等价关系的个数,教材给出了如下递推关系式:

        B(n+1)=k=0nC(n,k)B(k)B(n+1)=\sum_{k=0}^n C(n, k)B(k)

        这个递推关系式的理解是:对于有 n+1 元素集合 (不妨假定为 {0,1, ⋯, n}) 的划分,按照最后一个元素 n 所在的划分块进行分类:

        • 若 n 不与 {0, ⋯, n-1} 的任意元素在一个划分块,即 n 单独在一个划分块,这种划分的个数就等于 {0, ⋯, n-1} 的划分个数,即等于 B (n)
        • 若 n 与 {0, ⋯, n-1} 的某 j=n-k (k=0, ⋯, n) 个元素在一个划分块,则这 j 个元素有 C (n, j) = C (n,n-k) = C (n,k) 种选择,而每种选择的划分个数等于剩下的 k 个元素构成集合的划分个数,即等于 B (k),因此有 C (n,k) B (k) 个划分

        # 根据公式计算结果

        当 | A|=3,则 A 上等价关系个数:

        1+23C(2,1)2!+33C(3,1)23+C(3,2)3!=1+3+1=51+\frac{2^3-C(2,1)}{2!}+\frac{3^3-C(3,1) 2^3+ C(3,2)}{3!}=1+3+1=5

        当 | A|=4,则 A 上等价关系个数:

        1+24C(2,1)2!+34C(3,1)24+C(3,2)3!+44C(4,1)34+C(4,2)24C(4,3)4!=151+\frac{2^4-C(2,1)}{2!}+\frac{3^4-C(3,1) 2^4+C(3,2)}{3!}+\frac{4^4-C(4,1) 3^4+C(4,2)2^4-C(4,3)}{4!}=15

        \ No newline at end of file +等价关系经典计数问题 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 题面

        f:ABf:A\to B 是函数,定义 A 上的关系 R,a,bA\forall a,b \in A,a R b 当且仅当 f (a) = f (b) 。

        证明 R 是等价关系,并给出它的等价类和商集,求这样的等价关系有多少个?

        # 证明 R 是等价关系

        显然 R 是等价关系,因为对∀a, b, c\in A,f (a)=f (a), f (a)=f (b) 蕴涵 f (b) = f (a) , f (a) = f (b) 且 f (b) = f (c) 蕴涵 f (a) = f (c) ,即 R 是自反、对称、传递的。

        # 求等价类

        对任意 a\in A,[a]R={xf(x)=f(a)},A/R={[a]RaA}[a]_R= \{x∣f(x)=f(a) \},A/R=\{[a]_R∣a\in A\}

        实际上,对任意 a\in A,若 f (a)=y\in B,则[a]R=f1(y)[a]_R=f^{-1} (y)

        因此若 f 是满函数,则 A/R={f1({b})bB}A/R=\{f^{-1} (\{b\})∣b\in B\},即商集 A/R 中的等价类与 B 的元素一一对应。

        对集合 A 上的任意等价关系 R ,自然映射ρ:AA/R,ρ(a)=[a]Rρ:A→A/R,ρ(a)=[a]_R 是满函数,因此 A 上的等价关系与以 A 为定义域的满函数对应。

        设 |A|=n ,则 A 上有 m 个等价类的等价关系个数等于 A 到 Zm={0,1,,m1}Z_m=\{0, 1, ⋯, m-1\} 的满函数个数除以 m! (在 Zm 的元素作为原像的所有可能排列中选一个即可)。

        # 求等价关系个数

        设 |A|=n ,则 A 上不同的等价关系有多少个?

        等价于:设 | A|=n,则 A 上含 m 个等价类的等价关系有多少个?

        当 | A|=n, |B|=m,n≥m,A 到 B 的满函数个数是:

        mnC(m,1)(m1)n++(1)kC(m,k)(mk)n++(1)m1C(m,m1)1nm^n- C(m, 1)(m-1)^n+ ⋯+ (-1)^k C(m, k) (m-k)^n+ ⋯+ (-1)^{m-1} C(m,m-1)\cdot 1^n

        因此 n 元素集合 A 上有 m 个等价类的等价关系有:

        B(n,m)=k=0m1(1)kC(m,k)(mk)nm!B(n, m)=\frac{\sum_{k=0}^{m-1}(-1)^k C(m,k) (m-k)^n }{m!}

        从而 n 元素集合 A 上的不同等价关系个数有:

        B(n)=m=1nk=0m1(1)kC(m,k)(mk)nm!B(n)=\sum_{m=1}^n\frac{\sum_{k=0}^{m-1}(-1)^k C(m,k) (m-k)^n}{m!}

        用 B (n) 表示 n 元素集合上不同等价关系的个数,教材给出了如下递推关系式:

        B(n+1)=k=0nC(n,k)B(k)B(n+1)=\sum_{k=0}^n C(n, k)B(k)

        这个递推关系式的理解是:对于有 n+1 元素集合 (不妨假定为 {0,1, ⋯, n}) 的划分,按照最后一个元素 n 所在的划分块进行分类:

        • 若 n 不与 {0, ⋯, n-1} 的任意元素在一个划分块,即 n 单独在一个划分块,这种划分的个数就等于 {0, ⋯, n-1} 的划分个数,即等于 B (n)
        • 若 n 与 {0, ⋯, n-1} 的某 j=n-k (k=0, ⋯, n) 个元素在一个划分块,则这 j 个元素有 C (n, j) = C (n,n-k) = C (n,k) 种选择,而每种选择的划分个数等于剩下的 k 个元素构成集合的划分个数,即等于 B (k),因此有 C (n,k) B (k) 个划分

        # 根据公式计算结果

        当 | A|=3,则 A 上等价关系个数:

        1+23C(2,1)2!+33C(3,1)23+C(3,2)3!=1+3+1=51+\frac{2^3-C(2,1)}{2!}+\frac{3^3-C(3,1) 2^3+ C(3,2)}{3!}=1+3+1=5

        当 | A|=4,则 A 上等价关系个数:

        1+24C(2,1)2!+34C(3,1)24+C(3,2)3!+44C(4,1)34+C(4,2)24C(4,3)4!=151+\frac{2^4-C(2,1)}{2!}+\frac{3^4-C(3,1) 2^4+C(3,2)}{3!}+\frac{4^4-C(4,1) 3^4+C(4,2)2^4-C(4,3)}{4!}=15

        \ No newline at end of file diff --git a/math/discrete-math/discrete-math-basics/set/index.html b/math/discrete-math/discrete-math-basics/set/index.html index 236d6419..b29154fb 100644 --- a/math/discrete-math/discrete-math-basics/set/index.html +++ b/math/discrete-math/discrete-math-basics/set/index.html @@ -1 +1 @@ -集合 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 基本概念

        # 不严格定义的概念

        # 集合

        作为整体研究的一堆东西,用大写字母 A, B, C, ⋯表示

        # 元素

        集合这一堆东西中的每一个,用小写字母 a, b, c, ⋯表示

        # 属于

        元素与集合间的关系

        • 元素 a 属于集合 A ,记为 aAa\in A
        • a 不属于 A,记为 aAa\notin A

        元素与集合间的属于关系也称为成员关系,元素是集合的成员

        # 全集

        研究范围内的所有东西,记为 U

        # 用逻辑语言严格定义的概念

        # 子集关系

        ABx(xAxB)A \subseteq B \lrArr \forall x(x\in A \to x\in B)

        # 集合相等

        A=Bx(xAxB)A = B \lrArr \forall x(x\in A \lrarr x\in B)

        或者

        A=BABBAA=B \lrArr A\subseteq B \land B \subseteq A

        # 空集

        x(x)\forall x(x\notin \varnothing)

        # 朴素集合论

        # 外延原则

        两个集合只要有完全相同的元素则是相等的集合,不考虑集合名字本身的内涵

        # 概念 (名字) 的外延

        概念 (名字) 的外延是它所指称的对象,内涵是它有区别于其他概念的属性全体

        # 集合 (名字) 的外延

        对于集合 (名字),外延是它包含的所有元素,内涵则视具体的应用而定

        # 定义集合的方法

        # 元素枚举法

        将集合的所有元素一一罗列出来

        • 适合元素比较少,或可按明显规律罗列元素时定义集合
        • 元素罗列规律明显时可使用省略号

        # 性质概括法

        用谓词概括一个集合的所有元素满足的共同性质

        # 基本形式

        A={xP(x)}A = \{x∣P(x)\}

        含义是:

        x(xAP(x))\forall x(x\in A \lrarr P(x))

        允许 P 是任意性质时有可能产生悖论:罗素悖论

        公理集合论运用子集分离原则避免悖论:

        A={xBP(x)}A=\{x\in B∣P(x)\}

        B 是已知的大集合

        # 扩展形式

        A={f(y)P(y)}A=\{f(y)∣P(y)\}

        含义是

        x(xAy(x=f(y)P(y)))\forall x(x\in A↔∃y(x=f(y)∧P(y)))

        这里 f 是一个函数,或说 f (x) 是含有自由变量 x 的表达式

        # 归纳定义法

        给出基本元素和从已有元素构造其他元素的规则

        从某种意义上说,集合的归纳定义给出了构造集合元素的算法

        # 集合的划分 (partition)

        # 定义

        设 A 是非空集合,F 是集合族,其中每个集合都是 A 的子集。说集合族 F 是 A 的划分,如果:

        • 非空:对任意的 SFS\in \mathcal{F} , S\neq \mathcal
        • 两两不交:对任意两个集合 S1,S2FS_1, S_2\in F , S1S2=S_1\cap S_2=\varnothing
        • 覆盖集合 A:F=A\cup \mathcal{F}= A

        非空集合 A 的划分 F\mathcal{F} 中的每个集合称为这个划分的一个 划分块 (block)

        # 非空集合 A 上的等价关系与它的划分有一一对应关系

        A 关于一个等价关系的商集是 A 的划分,而 A 的一个划分导出的 “在同一划分块” 关系是等价关系

        进一步,A 关于 “在同一划分块” 这个等价关系的商集就是这个划分,而 A 关于等价关系的商集作为 A 的划分所导出的 “在同一划分块” 关系就是这个等价关系本身

        # 常用集合

        # 模 m 剩余类

        是一种常用等价类

        对任意整数 aZa\in \mathbb{Z} , a 在 R 下的等价类 [a]R[a]_R 称为整数集 Z\mathbb Z 的一个(与 a 同余的)模 m 剩余类,并记为:

        a={xZxa(modm)}={xZmxa}={a+mzzZ}\overline{a}=\{x\in \mathbb Z ∣ x ≡ a(\mod m) \}=\{x\in \mathbb Z∣m∣x-a\}=\{a+mz∣z\in \mathbb Z \}

        # 模 m 剩余类的商集

        Zm={0,1,...,m1}={0,1,...,m1}\mathbb{Z}_m = \{ \overline 0,\overline 1,...,\overline{m-1} \} = \{ 0,1,...,m-1 \}

        # 有序对

        集合论定义的有序对

        从二元有序对开始定义,归纳定义 n 元有序对

        # 二元有序对

        <a,b>={{a},{a,b}}<b,a>={{b},{a,b}}<a,b>=\{\{a\},\{a,b\}\}\\ <b,a>=\{\{b\},\{a,b\}\}

        # 三元有序对

        <a,b,c>=<<a,b>,c><a,b,c>=<<a,b>,c>

        # n 元有序对

        <a1,a2,...an>=<<a1,a2,...,an1>,an><a_1,a_2,...a_n>=<<a_1,a_2,...,a_{n-1}>,a_n>

        # 有序 n 元组性质定理

        <a1,a2,...an>=<b1,b2,...bn>ai=bi,i=1,2,3,...,n<a_1,a_2,...a_n>=<b_1,b_2,...b_n> \lrArr a_i=b_i \ ,\ i=1,2,3,...,n

        \ No newline at end of file +集合 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 基本概念

        # 不严格定义的概念

        # 集合

        作为整体研究的一堆东西,用大写字母 A, B, C, ⋯表示

        # 元素

        集合这一堆东西中的每一个,用小写字母 a, b, c, ⋯表示

        # 属于

        元素与集合间的关系

        • 元素 a 属于集合 A ,记为 aAa\in A
        • a 不属于 A,记为 aAa\notin A

        元素与集合间的属于关系也称为成员关系,元素是集合的成员

        # 全集

        研究范围内的所有东西,记为 U

        # 用逻辑语言严格定义的概念

        # 子集关系

        ABx(xAxB)A \subseteq B \lrArr \forall x(x\in A \to x\in B)

        # 集合相等

        A=Bx(xAxB)A = B \lrArr \forall x(x\in A \lrarr x\in B)

        或者

        A=BABBAA=B \lrArr A\subseteq B \land B \subseteq A

        # 空集

        x(x)\forall x(x\notin \varnothing)

        # 朴素集合论

        # 外延原则

        两个集合只要有完全相同的元素则是相等的集合,不考虑集合名字本身的内涵

        # 概念 (名字) 的外延

        概念 (名字) 的外延是它所指称的对象,内涵是它有区别于其他概念的属性全体

        # 集合 (名字) 的外延

        对于集合 (名字),外延是它包含的所有元素,内涵则视具体的应用而定

        # 定义集合的方法

        # 元素枚举法

        将集合的所有元素一一罗列出来

        • 适合元素比较少,或可按明显规律罗列元素时定义集合
        • 元素罗列规律明显时可使用省略号

        # 性质概括法

        用谓词概括一个集合的所有元素满足的共同性质

        # 基本形式

        A={xP(x)}A = \{x∣P(x)\}

        含义是:

        x(xAP(x))\forall x(x\in A \lrarr P(x))

        允许 P 是任意性质时有可能产生悖论:罗素悖论

        公理集合论运用子集分离原则避免悖论:

        A={xBP(x)}A=\{x\in B∣P(x)\}

        B 是已知的大集合

        # 扩展形式

        A={f(y)P(y)}A=\{f(y)∣P(y)\}

        含义是

        x(xAy(x=f(y)P(y)))\forall x(x\in A↔∃y(x=f(y)∧P(y)))

        这里 f 是一个函数,或说 f (x) 是含有自由变量 x 的表达式

        # 归纳定义法

        给出基本元素和从已有元素构造其他元素的规则

        从某种意义上说,集合的归纳定义给出了构造集合元素的算法

        # 集合的划分 (partition)

        # 定义

        设 A 是非空集合,F 是集合族,其中每个集合都是 A 的子集。说集合族 F 是 A 的划分,如果:

        • 非空:对任意的 SFS\in \mathcal{F} , S\neq \mathcal
        • 两两不交:对任意两个集合 S1,S2FS_1, S_2\in F , S1S2=S_1\cap S_2=\varnothing
        • 覆盖集合 A:F=A\cup \mathcal{F}= A

        非空集合 A 的划分 F\mathcal{F} 中的每个集合称为这个划分的一个 划分块 (block)

        # 非空集合 A 上的等价关系与它的划分有一一对应关系

        A 关于一个等价关系的商集是 A 的划分,而 A 的一个划分导出的 “在同一划分块” 关系是等价关系

        进一步,A 关于 “在同一划分块” 这个等价关系的商集就是这个划分,而 A 关于等价关系的商集作为 A 的划分所导出的 “在同一划分块” 关系就是这个等价关系本身

        # 常用集合

        # 模 m 剩余类

        是一种常用等价类

        对任意整数 aZa\in \mathbb{Z} , a 在 R 下的等价类 [a]R[a]_R 称为整数集 Z\mathbb Z 的一个(与 a 同余的)模 m 剩余类,并记为:

        a={xZxa(modm)}={xZmxa}={a+mzzZ}\overline{a}=\{x\in \mathbb Z ∣ x ≡ a(\mod m) \}=\{x\in \mathbb Z∣m∣x-a\}=\{a+mz∣z\in \mathbb Z \}

        # 模 m 剩余类的商集

        Zm={0,1,...,m1}={0,1,...,m1}\mathbb{Z}_m = \{ \overline 0,\overline 1,...,\overline{m-1} \} = \{ 0,1,...,m-1 \}

        # 有序对

        集合论定义的有序对

        从二元有序对开始定义,归纳定义 n 元有序对

        # 二元有序对

        <a,b>={{a},{a,b}}<b,a>={{b},{a,b}}<a,b>=\{\{a\},\{a,b\}\}\\ <b,a>=\{\{b\},\{a,b\}\}

        # 三元有序对

        <a,b,c>=<<a,b>,c><a,b,c>=<<a,b>,c>

        # n 元有序对

        <a1,a2,...an>=<<a1,a2,...,an1>,an><a_1,a_2,...a_n>=<<a_1,a_2,...,a_{n-1}>,a_n>

        # 有序 n 元组性质定理

        <a1,a2,...an>=<b1,b2,...bn>ai=bi,i=1,2,3,...,n<a_1,a_2,...a_n>=<b_1,b_2,...b_n> \lrArr a_i=b_i \ ,\ i=1,2,3,...,n

        \ No newline at end of file diff --git a/math/discrete-math/graph/algebra-graph/index.html b/math/discrete-math/graph/algebra-graph/index.html index dee32ed8..6690c81f 100644 --- a/math/discrete-math/graph/algebra-graph/index.html +++ b/math/discrete-math/graph/algebra-graph/index.html @@ -1 +1 @@ -图的代数表示 - 图论 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        用邻接矩阵或关联矩阵表示图,称为图的代数表示。用矩阵表示图,主要有两个优点:

        • (1) 能够把图输入到计算机中
        • (2) 可以用代数方法研究图论

        Spectral Graph Theory 是研究图和代数的一个领域

        # 邻接矩阵

        Adjaceny matrix is n×nn\times n symmetric matrix

        # 定义

        GGnn 阶图,V={v1,z2,,vn}V=\{v_1,z_2,\cdots,v_n\} ,邻接矩阵 A(G)=(aij)A(G)= (a_{ij}) ,其中:

        aij={l,vivj间边数0,vivj不邻接ai上的环数,i=ja_{ij}=\begin{cases} l,\ \ \ v_i\text{与}v_j\text{间边数}\\ 0,\ \ \ v_i\text{与}v_j\text{不邻接}\\ a_i\text{上的环数},\ \ \ i=j \end{cases}

        # 代数性质

        非负性 —— 由邻接矩阵定义知 aija_{ij} 是非负整数,即邻接矩阵是非负整数矩阵

        (无向图) 对称性 —— 显然 aij=ajia_{ij}=a_{ji} ,所以邻接矩阵是对称矩阵。

        同一图的不同形式的邻接矩阵是相似矩阵。这是因为,同图的两种不同形式矩阵可以通过交换行和相应的列变成致。

        如果 G 为简单图,则 A(G)A(G)布尔矩阵;行和 (列和) 等于对应顶点的度数;矩阵元素总和为图的总度数,也就是 GG 的边数的 2 倍

        # 连通的充要条件

        GG 连通的充要条件是:A(G)A(G) 不能与如下矩阵相似:

        (A1100A22)\begin{pmatrix} A_{11} & 0\\ 0 & A_{22} \end{pmatrix} \quad

        # 性质 (定理)

        Ak(G)=(aij(k))A^{k}(G)=(a_{ij}^{(k)}) ,则 aij(k)a_{ij}^{(k)} 表示顶点 viv_ivjv_j 的途径长度为 kk 的途径数目;


        证明:对 kk 作数学归纳法证明

        k=1k=1 时,由邻接矩阵定义;

        设结论对 k1k-1 时成立,考虑 kk 的情况:

        Ak=Ak1A=(ai1(k1)aj1+ai2(k1)aj2++ain(k1)ajn)n×n=(aij(k))A^k=A^{k-1}A=(a_{i1}^{(k-1)}a_{j1}+a_{i2}^{(k-1)}a_{j2}+\cdots+a_{in}^{(k-1)}a_{jn})_{n\times n}=(a_{ij}^{(k)})

        同时,考虑顶点 viv_ivjv_j 的途径长度为 kk 的途径,设 vmv_mviv_ivjv_j 的途径中点,且该点与 vjv_j 邻接。则 viv_ivjv_j 的经过 vmv_m 且途径长度为 kk 的途径数目为:

        aim(k1)amja_{im}^{(k-1)}a_{mj}

        所以 viv_ivjv_j 的途径长度为 kk 的途径数目为:

        ai1(k1)aj1+ai2(k1)aj2++ain(k1)ajn=aij(k)a_{i1}^{(k-1)}a_{j1}+a_{i2}^{(k-1)}a_{j2}+\cdots+a_{in}^{(k-1)}a_{jn}=a_{ij}^{(k)}


        推论:

        GG 是连通的,则对于 iji\neq jviv_ivjv_j 的距离是使 AnA^{n}aij(k)0a_{ij}^{(k)}\neq0 的最小整数

        这是显然的

        # 关联矩阵

        Incidence matrix is n×mn\times m matrix of a (n,m)(n,m) graph

        # 定义

        GG(n,m)(n,m) 图。定义 GG 的关联矩阵: M(G)=(aij)n×mM(G)=(a_{ij})_{n\times m} ,其中:aija_{ij}viv_ieje_j 关联的次数 (取值为 0,1, 或 2 (环)).

        • 关联矩阵的元素为 0,1 或 2 ;
        • 关联矩阵的每列和为 2 ; 每行的和为对应顶点度数;

        # 连通的充要条件

        无环图 GG 连通的充分必要条件𝑅(𝑀)=n1𝑅(𝑀)=n-1


        证明充分性:

        GG 不连通,假设 GG 有两个连通分支 G1G_1G2G_2 , 又设 G1G_1G2G_2 的关联矩阵分别为 𝑀1𝑀_1𝑀2𝑀_2 , 则 GG 的关联矩阵可以写为:

        M(G)=(M1M2)M(G)= \begin{pmatrix} M_1 & \\ & M_2 \end{pmatrix}

        于是 𝑅(𝑀)=V(G1)1+V(G2)1=n2𝑅(𝑀)=V(G_1)-1+V(G_2)-1=n-2 ,矛盾!所以 GG 一定连通。


        证明必要性:

        令:

        M=(m1m2mn)M= \begin{pmatrix} m_1\\ m_2\\ \vdots\\ m_n \end{pmatrix}

        由于 𝑀𝑀 中每列恰有两个 “1” 元素,所有行向量的和为 0 (模 2 加法运算,即为偶数),所以有 R(M)n1R(M)\leq n-1

        另一方面,在 MM 中任意去掉一行 mkm_k ,由于 GG连通的,因此,mkm_k 中存在元素 “1” ,这样,MM 中去掉行 mkm_k ,后的行按模 2 相加不等于零,即它们是线性无关的。所以有 R(M)n1R(M)\geq n-1

        因此,R(M)=n1R(M)=n-1

        # 基本关联矩阵

        GG 的关联矩阵中删掉任意一行后得到的矩阵可以完全决定 GG ,该矩阵称为 GG基本关联矩阵。删掉的行对应的顶点称为该基本关联矩阵的参考点

        图的关联矩阵及其性质是网络图论的基础,在电路分析中有重要应用

        图的关联矩阵比邻接矩阵大得多,不便于计算机存储。但二者都有各自的应用特点

        # 定义 2

        Incidence matrix CC is m×nm\times n matrix of a (n,m)(n,m) graph

        用来定义拉普拉斯矩阵 L=CTCL=C^{T}C

        • 每行代表一条边,每列代表一个顶点
        • 每行和为 0 ,有一个 1 和 -1 ,分别表示边的起点和终点
        • 其余元素全部为 0

        C=(e1Te2TemT)C= \begin{pmatrix} e_1^{T}\\ e_2^{T}\\ \vdots\\ e_m^{T} \end{pmatrix}

        # 度矩阵

        Degree matrix is n×nn\times n diagonal matrix

        # 定义

        度矩阵是对角阵,对角上的元素为各个顶点的度。

        ⚠️有向图中,可能会有不同的定义。一般度数为各个顶点入度与出度之和,例如拉普拉斯矩阵引入的度矩阵。

        # 拉普拉斯矩阵

        Laplacian matrix (LL) is n×nn\times n symmetric matrix

        拉普拉斯矩阵是对称矩阵,保留了图的 “无向” 的属性,丢失了 “有向” 的属性

        GG 是顶点集合为 V(G)={v1,v2,,vn}V(G)=\{v_1,v_2,\cdots,v_n\} 的图, DD 为图的度矩阵A=aijA=a_{ij}GG邻接矩阵CC 为图的关联矩阵

        定义:

        L=DA=CTCL=D-A=C^{T}C

        拉普拉斯矩阵 L=(lij)L=(l_{ij})nn 阶方阵,其中:

        lij={d(vi),i=jaij,ijl_{ij}=\begin{cases} d(v_i), & i=j\\ -a_{ij}, & i\neq j \end{cases}

        • 所有特征值为非负实数: 0λ0λ1λn10\leq\lambda_0\leq\lambda_1\leq\cdots\leq\lambda_{n-1}
        • 特征向量为实 (正交) 向量
        • 每一行、列之和都等于 0
        • xTLx0x^{T}Lx\geq0 是半正定二次型,或者说拉普拉斯矩阵 LL 是半正定矩阵

        [Fiedler, 1973] GGkk 个连通分支当且仅当:

        λ0=λ1==λk1=0\lambda_0=\lambda_1=\cdots=\lambda_{k-1}=0

        GG 显然至少有一个连通分支,所以 λ0=0\lambda_0=0

        GG 的割边 (cut edges) 数量等于:

        14xTLx=14(i,j)E(xixj)2\frac{1}{4}x^{T}Lx=\frac{1}{4}\sum_{(i,j)\in E}(x_i-x_j)^2

        (等式右侧推导在下面)

        对任意对称矩阵 MM 定义 λ2\lambda_2

        λ2=minxxTMxxTx\lambda_2=\min_{x}\frac{x^{T}Mx}{x^{T}x}

        考虑拉普拉斯矩阵 LL 对应的二次型 xTLxx^{T}Lx

        xTLx=i,j=1nLijxixj=i,j=1n(DijAij)xixj=i=1nDiixi2(i,j)E2xixj=(i,j)E(xi2+xj22xixj)=(i,j)E(xixj)2\begin{aligned} x^{T}Lx &=\sum_{i,j=1}^{n}L_{ij}x_ix_j\\ &=\sum_{i,j=1}^{n}(D_{ij}-A_{ij})x_ix_j\\ &=\sum_{i=1}^{n}D_{ii}x_i^2-\sum_{(i,j)\in E}2x_ix_j\\ &=\sum_{(i,j)\in E}(x_i^2+x_j^2-2x_ix_j)\\ &=\sum_{(i,j)\in E}(x_i-x_j)^2 \end{aligned}

        此时:

        λ2=minAll labelings of so thatxi=0(i,j)E(xixj)2ixi2\lambda_2=\min_{\text{All labelings of so that }\sum x_i=0}\frac{\sum_{(i,j)\in E}(x_i-x_j)^2}{\sum_ix_i^2}

        # 子图空间

        # 回路空间

        https://lfool.github.io/LFool-Notes/other/ 图论.html# 回路系统简介

        # 断集空间

        \ No newline at end of file +图的代数表示 - 图论 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        用邻接矩阵或关联矩阵表示图,称为图的代数表示。用矩阵表示图,主要有两个优点:

        • (1) 能够把图输入到计算机中
        • (2) 可以用代数方法研究图论

        Spectral Graph Theory 是研究图和代数的一个领域

        # 邻接矩阵

        Adjaceny matrix is n×nn\times n symmetric matrix

        # 定义

        GGnn 阶图,V={v1,z2,,vn}V=\{v_1,z_2,\cdots,v_n\} ,邻接矩阵 A(G)=(aij)A(G)= (a_{ij}) ,其中:

        aij={l,vivj间边数0,vivj不邻接ai上的环数,i=ja_{ij}=\begin{cases} l,\ \ \ v_i\text{与}v_j\text{间边数}\\ 0,\ \ \ v_i\text{与}v_j\text{不邻接}\\ a_i\text{上的环数},\ \ \ i=j \end{cases}

        # 代数性质

        非负性 —— 由邻接矩阵定义知 aija_{ij} 是非负整数,即邻接矩阵是非负整数矩阵

        (无向图) 对称性 —— 显然 aij=ajia_{ij}=a_{ji} ,所以邻接矩阵是对称矩阵。

        同一图的不同形式的邻接矩阵是相似矩阵。这是因为,同图的两种不同形式矩阵可以通过交换行和相应的列变成致。

        如果 G 为简单图,则 A(G)A(G)布尔矩阵;行和 (列和) 等于对应顶点的度数;矩阵元素总和为图的总度数,也就是 GG 的边数的 2 倍

        # 连通的充要条件

        GG 连通的充要条件是:A(G)A(G) 不能与如下矩阵相似:

        (A1100A22)\begin{pmatrix} A_{11} & 0\\ 0 & A_{22} \end{pmatrix} \quad

        # 性质 (定理)

        Ak(G)=(aij(k))A^{k}(G)=(a_{ij}^{(k)}) ,则 aij(k)a_{ij}^{(k)} 表示顶点 viv_ivjv_j 的途径长度为 kk 的途径数目;


        证明:对 kk 作数学归纳法证明

        k=1k=1 时,由邻接矩阵定义;

        设结论对 k1k-1 时成立,考虑 kk 的情况:

        Ak=Ak1A=(ai1(k1)aj1+ai2(k1)aj2++ain(k1)ajn)n×n=(aij(k))A^k=A^{k-1}A=(a_{i1}^{(k-1)}a_{j1}+a_{i2}^{(k-1)}a_{j2}+\cdots+a_{in}^{(k-1)}a_{jn})_{n\times n}=(a_{ij}^{(k)})

        同时,考虑顶点 viv_ivjv_j 的途径长度为 kk 的途径,设 vmv_mviv_ivjv_j 的途径中点,且该点与 vjv_j 邻接。则 viv_ivjv_j 的经过 vmv_m 且途径长度为 kk 的途径数目为:

        aim(k1)amja_{im}^{(k-1)}a_{mj}

        所以 viv_ivjv_j 的途径长度为 kk 的途径数目为:

        ai1(k1)aj1+ai2(k1)aj2++ain(k1)ajn=aij(k)a_{i1}^{(k-1)}a_{j1}+a_{i2}^{(k-1)}a_{j2}+\cdots+a_{in}^{(k-1)}a_{jn}=a_{ij}^{(k)}


        推论:

        GG 是连通的,则对于 iji\neq jviv_ivjv_j 的距离是使 AnA^{n}aij(k)0a_{ij}^{(k)}\neq0 的最小整数

        这是显然的

        # 关联矩阵

        Incidence matrix is n×mn\times m matrix of a (n,m)(n,m) graph

        # 定义

        GG(n,m)(n,m) 图。定义 GG 的关联矩阵: M(G)=(aij)n×mM(G)=(a_{ij})_{n\times m} ,其中:aija_{ij}viv_ieje_j 关联的次数 (取值为 0,1, 或 2 (环)).

        • 关联矩阵的元素为 0,1 或 2 ;
        • 关联矩阵的每列和为 2 ; 每行的和为对应顶点度数;

        # 连通的充要条件

        无环图 GG 连通的充分必要条件𝑅(𝑀)=n1𝑅(𝑀)=n-1


        证明充分性:

        GG 不连通,假设 GG 有两个连通分支 G1G_1G2G_2 , 又设 G1G_1G2G_2 的关联矩阵分别为 𝑀1𝑀_1𝑀2𝑀_2 , 则 GG 的关联矩阵可以写为:

        M(G)=(M1M2)M(G)= \begin{pmatrix} M_1 & \\ & M_2 \end{pmatrix}

        于是 𝑅(𝑀)=V(G1)1+V(G2)1=n2𝑅(𝑀)=V(G_1)-1+V(G_2)-1=n-2 ,矛盾!所以 GG 一定连通。


        证明必要性:

        令:

        M=(m1m2mn)M= \begin{pmatrix} m_1\\ m_2\\ \vdots\\ m_n \end{pmatrix}

        由于 𝑀𝑀 中每列恰有两个 “1” 元素,所有行向量的和为 0 (模 2 加法运算,即为偶数),所以有 R(M)n1R(M)\leq n-1

        另一方面,在 MM 中任意去掉一行 mkm_k ,由于 GG连通的,因此,mkm_k 中存在元素 “1” ,这样,MM 中去掉行 mkm_k ,后的行按模 2 相加不等于零,即它们是线性无关的。所以有 R(M)n1R(M)\geq n-1

        因此,R(M)=n1R(M)=n-1

        # 基本关联矩阵

        GG 的关联矩阵中删掉任意一行后得到的矩阵可以完全决定 GG ,该矩阵称为 GG基本关联矩阵。删掉的行对应的顶点称为该基本关联矩阵的参考点

        图的关联矩阵及其性质是网络图论的基础,在电路分析中有重要应用

        图的关联矩阵比邻接矩阵大得多,不便于计算机存储。但二者都有各自的应用特点

        # 定义 2

        Incidence matrix CC is m×nm\times n matrix of a (n,m)(n,m) graph

        用来定义拉普拉斯矩阵 L=CTCL=C^{T}C

        • 每行代表一条边,每列代表一个顶点
        • 每行和为 0 ,有一个 1 和 -1 ,分别表示边的起点和终点
        • 其余元素全部为 0

        C=(e1Te2TemT)C= \begin{pmatrix} e_1^{T}\\ e_2^{T}\\ \vdots\\ e_m^{T} \end{pmatrix}

        # 度矩阵

        Degree matrix is n×nn\times n diagonal matrix

        # 定义

        度矩阵是对角阵,对角上的元素为各个顶点的度。

        ⚠️有向图中,可能会有不同的定义。一般度数为各个顶点入度与出度之和,例如拉普拉斯矩阵引入的度矩阵。

        # 拉普拉斯矩阵

        Laplacian matrix (LL) is n×nn\times n symmetric matrix

        拉普拉斯矩阵是对称矩阵,保留了图的 “无向” 的属性,丢失了 “有向” 的属性

        GG 是顶点集合为 V(G)={v1,v2,,vn}V(G)=\{v_1,v_2,\cdots,v_n\} 的图, DD 为图的度矩阵A=aijA=a_{ij}GG邻接矩阵CC 为图的关联矩阵

        定义:

        L=DA=CTCL=D-A=C^{T}C

        拉普拉斯矩阵 L=(lij)L=(l_{ij})nn 阶方阵,其中:

        lij={d(vi),i=jaij,ijl_{ij}=\begin{cases} d(v_i), & i=j\\ -a_{ij}, & i\neq j \end{cases}

        • 所有特征值为非负实数: 0λ0λ1λn10\leq\lambda_0\leq\lambda_1\leq\cdots\leq\lambda_{n-1}
        • 特征向量为实 (正交) 向量
        • 每一行、列之和都等于 0
        • xTLx0x^{T}Lx\geq0 是半正定二次型,或者说拉普拉斯矩阵 LL 是半正定矩阵

        [Fiedler, 1973] GGkk 个连通分支当且仅当:

        λ0=λ1==λk1=0\lambda_0=\lambda_1=\cdots=\lambda_{k-1}=0

        GG 显然至少有一个连通分支,所以 λ0=0\lambda_0=0

        GG 的割边 (cut edges) 数量等于:

        14xTLx=14(i,j)E(xixj)2\frac{1}{4}x^{T}Lx=\frac{1}{4}\sum_{(i,j)\in E}(x_i-x_j)^2

        (等式右侧推导在下面)

        对任意对称矩阵 MM 定义 λ2\lambda_2

        λ2=minxxTMxxTx\lambda_2=\min_{x}\frac{x^{T}Mx}{x^{T}x}

        考虑拉普拉斯矩阵 LL 对应的二次型 xTLxx^{T}Lx

        xTLx=i,j=1nLijxixj=i,j=1n(DijAij)xixj=i=1nDiixi2(i,j)E2xixj=(i,j)E(xi2+xj22xixj)=(i,j)E(xixj)2\begin{aligned} x^{T}Lx &=\sum_{i,j=1}^{n}L_{ij}x_ix_j\\ &=\sum_{i,j=1}^{n}(D_{ij}-A_{ij})x_ix_j\\ &=\sum_{i=1}^{n}D_{ii}x_i^2-\sum_{(i,j)\in E}2x_ix_j\\ &=\sum_{(i,j)\in E}(x_i^2+x_j^2-2x_ix_j)\\ &=\sum_{(i,j)\in E}(x_i-x_j)^2 \end{aligned}

        此时:

        λ2=minAll labelings of so thatxi=0(i,j)E(xixj)2ixi2\lambda_2=\min_{\text{All labelings of so that }\sum x_i=0}\frac{\sum_{(i,j)\in E}(x_i-x_j)^2}{\sum_ix_i^2}

        # 子图空间

        # 回路空间

        https://lfool.github.io/LFool-Notes/other/ 图论.html# 回路系统简介

        # 断集空间

        \ No newline at end of file diff --git a/math/discrete-math/graph/coloring/index.html b/math/discrete-math/graph/coloring/index.html index 4c502cae..0bfbfada 100644 --- a/math/discrete-math/graph/coloring/index.html +++ b/math/discrete-math/graph/coloring/index.html @@ -1 +1 @@ -着色问题 - 图论 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 边着色问题

        图的边着色,本质上是对应实际问题中的 “划分” 问题或 “分类” 问题。

        GG 的边进行着色,若相邻边着不同颜色,则称对 GG 进行正常边着色

        在对 GG 正常边着色时,着相同颜色的边集称为该正常边着色的一个色组

        如果能用 kk 种颜色对图 GG 进行正常边着色,称 GGkk 边可着色的

        GG 进行正常边着色需要的最少颜色数称为 GG边色数,记为:

        χ(G)\chi'(G)

        着色需要满足相邻边着不同色,所以很自然地有:(对于无环图)

        χ(G)Δ\chi'(G)\geq\Delta

        也就是边色数以最大顶点度为下限

        # 二部图

        χ(Km,n)=Δ\chi'(K_{m,n})=\Delta


        证明

        假设:

        • n>mn\gt m
          • 所以 Δ=n\Delta=n
        • X={x0,x1,,xm1}X=\{x_0,x_1,\cdots,x_{m-1}\}
        • Y={y0,y1,,yn1}Y=\{y_0,y_1,\cdots,y_{n-1}\}
        • nn 个颜色集合为:{0,1,2,,n1}\{0,1,2,\cdots,n-1\}

        π\piKm,nK_{m,n} 的一个 nn 着色方案,满足:

        xiyjE(Km,n),π(xiyj)=(i+j)(modn)\forall x_i y_j\in E(K_{m,n}),\ \ \pi(x_iy_j)=(i+j)(\bmod n)

        Km,nK_{m,n} 中任意两条邻接边 xiyjx_iy_jxiykx_iy_k ,若二者在 π\pi 中着色相同,即 π(xiyj)=π(xiyk)\pi(x_iy_j)=\pi(x_iy_k) ,则有:

        (i+j)(modn)=(i+k)(modn)j=k(i+j)(\bmod n)=(i+k)(\bmod n)\\ \Downarrow\\ j=k

        于是 xiyj=xiykx_iy_j=x_iy_k ,所以 π\pi 是一个正常的 nn 着色(满足相邻边着不同色)。也即 (X,Y)(X,Y)nn 可着色的。

        同时 χ(Km,n)Δ=n\chi'(K_{m,n})\geq\Delta=n ,所以 χ(Km,n)=n\chi'(K_{m,n})=n

        # 偶图 (哥尼定理)

        偶图 GG 的边色数:

        χ(G)=Δ\chi'(G)=\Delta


        证明可利用数学归纳法

        m=1m=1 时,Δ=1\Delta=1,有 χ(G)=Δ=1\chi'(G)=\Delta=1 成立。

        设对于小于 mm 条边的偶图来说命题成立。

        GG 是具有 mm 条边的偶图。

        uvGuv\in G , 考虑 G1=GuvG_1=G-uv, 由归纳假设有:

        χ(G1)=Δ(G1)Δ(G)\chi'(G_1)=\Delta(G_1)\leq\Delta(G)

        这说明, GG 存在一种 Δ(G)\Delta(G) 边着色方案 π\pi . 对于该着色方案,因为 uvuv 未着色,所以点 uuvv 均至少缺少一种色。


        情形 1:如果 uuvv 均缺同一种色 ii

        则在 G1+uvG_1+uv 中给 uvuv 着色 ii , 而 GG 其它边按 π\pi 方案着色。这样得到 GGΔ\Delta 着色方案,所以: χ(G)=Δ\chi'(G)=\Delta


        情形 2:如果 uu 缺色 ii , 而 vv 缺色 jj , 但不缺色 ii

        H(i,j)H(i,j) 表示 GG 中由 ii 色边与 jj 色边导出的子图。显然,该图每个分支是 ii 色边和 jj 色边交替出现的路或圈。
        对于 H(i,j)H(i,j) 中含点 vv 的分支来说,因 vv 缺色 jj , 但不缺色 ii , 所以,在 H(i,j)H(i,j) 中,点 vv 的度数为 1。这说明,H(i,j)H(i,j) 中含 vv 的分支是一条路 PP . 进一步地,我们可以说明,上面的路 PP 不含点 uu . 因为,如果 PP 含有点 uu , 那么 PP 必然是一条长度为偶数的路,这样,P+uvP+uvGG 中的奇圈,这与 GG 是偶图矛盾!

        既然 PP 不含点 uu , 所以我们可以交换 PP 中着色,而不破坏 G1G_1 的正常边着色。但交换着色后,vv 就变成缺色 jj , 但不缺色 iiuuvv 均缺色 ii , 于是由情形 1, 可以得到 GGΔ\Delta 正常边着色,即证明 χ(G)=Δ\chi'(G)=\Delta

        # 一般简单图

        # 引理

        GG 是简单图,xxyyGG 中两个不相邻的顶点,π\piGG 的一个正常 kk 边着色。若对该着色,x,yx,y 以及与 xx 相邻的点均至少缺少一种颜色,则 G+xyG+xy 也是 kk 边可着色的

        # 维津定理 (Vizing)

        若图 GG 是简单图,则 χ(G)=Δ\chi'(G)=\Deltaχ(G)=Δ+1\chi'(G)=\Delta+1


        只需要证明 χ(G)Δ+1\chi'(G)≤\Delta+1 即可。 对 GG 的边数 mm 作数学归纳证明。

        m=1m=1 时,Δ=1\Delta=1, χ(G)=1<Δ+1\chi'(G)=1<Δ+1

        设当边数少于 mm 时,结论成立。下面考虑边数为 m2m≥2 的单图 GG

        xyE(G)xy\in E(G) , 令 G1=GxyG_1=G-xy. 由归纳假设有:

        χ(G1)Δ(G1)+1Δ(G)+1\chi'(G_1)≤\Delta(G_1)+1≤\Delta(G)+1

        于是,存在 G1G_1Δ(G)+1\Delta(G)+1 正常边作色 π\pi。显然 GG 的每个顶点都至少缺少一种颜色。根据引理知 G+xyG+xyΔ(G)+1\Delta(G) +1 可着色的。即证明: χ(G)Δ(G)+1\chi'(G) ≤ \Delta(G)+1

        # 充分条件 1

        给出了 Vizing 定理结论中 χ(G)=Δ\chi'(G)=\Delta 情况的充分条件

        GG 是单图且 Δ(G)>0\Delta(G)>0。若 GG 中只有一个最大度点或恰有两个相邻的最大度点,则:

        χ(G)=Δ(G)\chi'(G)=\Delta(G)

        # 充分条件 2

        给出了 Vizing 定理结论中 χ(G)=Δ+1\chi'(G)=\Delta+1 情况的充分条件

        GG 是单图。若点数 n=2k+1n=2k+1 且边数 m>kΔm>k\Delta,则:

        χ(G)=Δ(G)+1\chi'(G)=\Delta(G)+1


        证明可利用反证法,若不然,由维津定理,χ(G)=Δ(G)\chi'(G)=\Delta(G)

        π\piGGΔ(G)\Delta(G) 正常边着色方案,对于 GG 的每个色组来说,包含的边数至多 n12=k\frac{n-1}{2}=k。这样:m(G)Δkm(G)≤\Delta k,与条件矛盾。

        # 充分条件 3

        给出了 Vizing 定理结论中 χ(G)=Δ+1\chi'(G)=\Delta+1 情况的充分条件

        GG 是奇数阶 Δ\Delta 正则单图,若 Δ>0\Delta>0,则:

        χ(G)=Δ(G)+1\chi'(G)=\Delta(G)+1


        证明:

        n=2k+1n=2k+1,因 GGΔ\Delta 正则单图,且 Δ>0\Delta>0,所以:

        m(G)=nΔ2=(2k+1)Δ2>kΔm(G)=\frac{n\Delta}{2}=\frac{(2k+1)\Delta}{2}\gt k\Delta

        由定理 χ(G)=Δ(G)+1\chi'(G)=\Delta(G)+1

        # 无环有重边图

        维津定理 (Vizing)

        设无环图 GG 中边的最大重数为 μ\mu​,则:

        χ(G)Δ(G)+μ\chi'(G)\leq\Delta(G)+\mu

        其中 \leq 边界是紧的,也就是等号是可以相等的。

        # 边着色的应用

        # 排课表问题

        X,YX,Y 分别表示教师、教学班的集合:

        X={x1,x2,,xm}Y={y1,y2,,yn}X=\{x_1,x_2,\cdots,x_m\}\\ Y=\{y_1,y_2,\cdots,y_n\}

        xix_iyjy_j 间连 pijp_{ij} 条边,得二部图 G=(X,Y)G=(X,Y)

        求最少排多少节课的问题,每节课可以有多个教学班同时上课,转化为如何在 GG 中将边集 EE 划分为互不相交的 pp 个匹配,且使得 pp 最小

        如果每个匹配中的边用同一种颜色着色,不同匹配中的边不同颜色,则问题转化为在 GG 中给每条边着色,相邻边着不同色,至少需要的颜色数

        # 比赛安排问题

        玩家为顶点,2 个玩家间进行一次比赛为一条边,求最少安排多少场比赛,每场比赛可以有多对玩家同时比赛。

        基于圈的边着色问题和顶点着色问题是完全等价的,因为边和点可互换。

        # 顶点着色

        对图 GG 的每个顶点着色,使得相邻顶点着不同颜色,称为对 GG正常顶点着色

        如果用 kk 种颜色可以对 GG 进行正常点着色,称 GGkk 可着色的

        着同一种颜色的顶点集合称为一个色组,它们彼此互不相邻,所有又称为点独立集

        GG 正常顶点着色需要的最少颜色数,称为图 GG点色数,简称色数,记为:

        χ(G)\chi(G)

        色数为 kk 的图称为 kk 色数图

        # 色数上界

        对任意的无环图 GG,均有:

        χ(G)Δ+1\chi(G)\leq\Delta+1

        任意一个顶点度数至多为 Δ\Delta, 因此,正常着色过程中,其邻点最多用去 Δ\Delta 种颜色,所以,至少还有一种色可供该点正常着色使用。


        证明 :我们对顶点数作数学归纳证明。

        n=1n=1 时,结论显然成立。

        设对顶点数少于 nn 的图来说,定理结论成立。考虑一般的 nn 阶图 GG,任取 vV(G)v\in V(G),令 G1=GvG_1=G-v, 由归纳假设:

        χ(G1)Δ(G1)+1Δ(G)+1\chi(G_1)\leq\Delta(G_1)+1\leq\Delta(G)+1

        π\piG1G_1 的一种 Δ(G1)+1\Delta(G_1)+1 正常点着色方案,因为 vv 的邻点在 π\pi 下至多用去 Δ(G)\Delta(G) 种色,所以给 vv 染上其邻点没有用过的色,就把 π\pi 扩充成了 GGΔ(G1)+1\Delta(G_1)+1 着色方案。

        # 最大色数的正常着色算法

        G=(V,E)G=(V,E)V={v1,v2,,vn}V=\{v_1,v_2,\cdots,v_n\},色集合 C={1,2,,Δ+1}C=\{1,2,\cdots,\Delta+1\},着色方案为 π\pi.

        • (1) 令 π(v1)=1\pi(v_1)=1
        • (2) for i in 1..n 循环 n1n-1 次:(若 i=ni=n,则停止)
          • 令:C(vi+1)={π(vj)ji,vjadjvi+1}C(v_{i+1})=\{\pi(v_j)|j≤i,\ v_j\ \text{adj}\ v_{i+1}\}
          • kkCC(vi+1)C-C(v_{i+1}) 中的最小整数;
          • π(vi+1)=k\pi(v_{i+1})=k

        Welsh-Powell 稍微对上面算法做了一个修改,着色时按所谓最大度优先策略,即使用上面算法时,V={v1,v2,,vn}V=\{v_1,v_2,\cdots,v_n\} 按顶点度数由大到小的次序着色。这样的着色方案起到了对上面算法的一个改进作用,有机会得到色数更小的结果。

        # 布鲁克斯定理

        给出了一些图的更小的上界

        GG 是连通的单图,并且它既不是奇圈,又不是完全图,则:

        χ(G)Δ(G)\chi(G)\leq\Delta(G)

        # 次大度

        GG 是至少有一条边的简单图,定义:

        Δ2(G)=maxuV(G)maxvN(u)d(v)d(u)d(v)\Delta_2(G)=\max_{u\in V(G)}\max_{\begin{aligned} v\in N(u)\\ d(v)\leq d(u) \end{aligned}} d(v)

        其中 N(u)N(u)GG 中点 uu 的邻域。称 Δ2(G)\Delta_2(G)GG 的次大度

        布鲁斯克定理的改进

        χ(G)Δ2(G)+1\chi(G)\leq\Delta_2(G)+1

        GG 是非空简单图,若 GG 中最大度点互不邻接,则有:

        χ(G)Δ(G)\chi(G)\leq\Delta(G)

        # 四色和五色定理

        # 四色定理

        # 五色定理

        希伍德。

        对任意简单图,都有:

        χ5\chi\leq5

        该定理说明每个平面图是 5 可着色的。根据平面图和其对偶图的关系,该定理等价于每个平面图是 5 可顶点正常着色的。

        \ No newline at end of file +着色问题 - 图论 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 边着色问题

        图的边着色,本质上是对应实际问题中的 “划分” 问题或 “分类” 问题。

        GG 的边进行着色,若相邻边着不同颜色,则称对 GG 进行正常边着色

        在对 GG 正常边着色时,着相同颜色的边集称为该正常边着色的一个色组

        如果能用 kk 种颜色对图 GG 进行正常边着色,称 GGkk 边可着色的

        GG 进行正常边着色需要的最少颜色数称为 GG边色数,记为:

        χ(G)\chi'(G)

        着色需要满足相邻边着不同色,所以很自然地有:(对于无环图)

        χ(G)Δ\chi'(G)\geq\Delta

        也就是边色数以最大顶点度为下限

        # 二部图

        χ(Km,n)=Δ\chi'(K_{m,n})=\Delta


        证明

        假设:

        • n>mn\gt m
          • 所以 Δ=n\Delta=n
        • X={x0,x1,,xm1}X=\{x_0,x_1,\cdots,x_{m-1}\}
        • Y={y0,y1,,yn1}Y=\{y_0,y_1,\cdots,y_{n-1}\}
        • nn 个颜色集合为:{0,1,2,,n1}\{0,1,2,\cdots,n-1\}

        π\piKm,nK_{m,n} 的一个 nn 着色方案,满足:

        xiyjE(Km,n),π(xiyj)=(i+j)(modn)\forall x_i y_j\in E(K_{m,n}),\ \ \pi(x_iy_j)=(i+j)(\bmod n)

        Km,nK_{m,n} 中任意两条邻接边 xiyjx_iy_jxiykx_iy_k ,若二者在 π\pi 中着色相同,即 π(xiyj)=π(xiyk)\pi(x_iy_j)=\pi(x_iy_k) ,则有:

        (i+j)(modn)=(i+k)(modn)j=k(i+j)(\bmod n)=(i+k)(\bmod n)\\ \Downarrow\\ j=k

        于是 xiyj=xiykx_iy_j=x_iy_k ,所以 π\pi 是一个正常的 nn 着色(满足相邻边着不同色)。也即 (X,Y)(X,Y)nn 可着色的。

        同时 χ(Km,n)Δ=n\chi'(K_{m,n})\geq\Delta=n ,所以 χ(Km,n)=n\chi'(K_{m,n})=n

        # 偶图 (哥尼定理)

        偶图 GG 的边色数:

        χ(G)=Δ\chi'(G)=\Delta


        证明可利用数学归纳法

        m=1m=1 时,Δ=1\Delta=1,有 χ(G)=Δ=1\chi'(G)=\Delta=1 成立。

        设对于小于 mm 条边的偶图来说命题成立。

        GG 是具有 mm 条边的偶图。

        uvGuv\in G , 考虑 G1=GuvG_1=G-uv, 由归纳假设有:

        χ(G1)=Δ(G1)Δ(G)\chi'(G_1)=\Delta(G_1)\leq\Delta(G)

        这说明, GG 存在一种 Δ(G)\Delta(G) 边着色方案 π\pi . 对于该着色方案,因为 uvuv 未着色,所以点 uuvv 均至少缺少一种色。


        情形 1:如果 uuvv 均缺同一种色 ii

        则在 G1+uvG_1+uv 中给 uvuv 着色 ii , 而 GG 其它边按 π\pi 方案着色。这样得到 GGΔ\Delta 着色方案,所以: χ(G)=Δ\chi'(G)=\Delta


        情形 2:如果 uu 缺色 ii , 而 vv 缺色 jj , 但不缺色 ii

        H(i,j)H(i,j) 表示 GG 中由 ii 色边与 jj 色边导出的子图。显然,该图每个分支是 ii 色边和 jj 色边交替出现的路或圈。
        对于 H(i,j)H(i,j) 中含点 vv 的分支来说,因 vv 缺色 jj , 但不缺色 ii , 所以,在 H(i,j)H(i,j) 中,点 vv 的度数为 1。这说明,H(i,j)H(i,j) 中含 vv 的分支是一条路 PP . 进一步地,我们可以说明,上面的路 PP 不含点 uu . 因为,如果 PP 含有点 uu , 那么 PP 必然是一条长度为偶数的路,这样,P+uvP+uvGG 中的奇圈,这与 GG 是偶图矛盾!

        既然 PP 不含点 uu , 所以我们可以交换 PP 中着色,而不破坏 G1G_1 的正常边着色。但交换着色后,vv 就变成缺色 jj , 但不缺色 iiuuvv 均缺色 ii , 于是由情形 1, 可以得到 GGΔ\Delta 正常边着色,即证明 χ(G)=Δ\chi'(G)=\Delta

        # 一般简单图

        # 引理

        GG 是简单图,xxyyGG 中两个不相邻的顶点,π\piGG 的一个正常 kk 边着色。若对该着色,x,yx,y 以及与 xx 相邻的点均至少缺少一种颜色,则 G+xyG+xy 也是 kk 边可着色的

        # 维津定理 (Vizing)

        若图 GG 是简单图,则 χ(G)=Δ\chi'(G)=\Deltaχ(G)=Δ+1\chi'(G)=\Delta+1


        只需要证明 χ(G)Δ+1\chi'(G)≤\Delta+1 即可。 对 GG 的边数 mm 作数学归纳证明。

        m=1m=1 时,Δ=1\Delta=1, χ(G)=1<Δ+1\chi'(G)=1<Δ+1

        设当边数少于 mm 时,结论成立。下面考虑边数为 m2m≥2 的单图 GG

        xyE(G)xy\in E(G) , 令 G1=GxyG_1=G-xy. 由归纳假设有:

        χ(G1)Δ(G1)+1Δ(G)+1\chi'(G_1)≤\Delta(G_1)+1≤\Delta(G)+1

        于是,存在 G1G_1Δ(G)+1\Delta(G)+1 正常边作色 π\pi。显然 GG 的每个顶点都至少缺少一种颜色。根据引理知 G+xyG+xyΔ(G)+1\Delta(G) +1 可着色的。即证明: χ(G)Δ(G)+1\chi'(G) ≤ \Delta(G)+1

        # 充分条件 1

        给出了 Vizing 定理结论中 χ(G)=Δ\chi'(G)=\Delta 情况的充分条件

        GG 是单图且 Δ(G)>0\Delta(G)>0。若 GG 中只有一个最大度点或恰有两个相邻的最大度点,则:

        χ(G)=Δ(G)\chi'(G)=\Delta(G)

        # 充分条件 2

        给出了 Vizing 定理结论中 χ(G)=Δ+1\chi'(G)=\Delta+1 情况的充分条件

        GG 是单图。若点数 n=2k+1n=2k+1 且边数 m>kΔm>k\Delta,则:

        χ(G)=Δ(G)+1\chi'(G)=\Delta(G)+1


        证明可利用反证法,若不然,由维津定理,χ(G)=Δ(G)\chi'(G)=\Delta(G)

        π\piGGΔ(G)\Delta(G) 正常边着色方案,对于 GG 的每个色组来说,包含的边数至多 n12=k\frac{n-1}{2}=k。这样:m(G)Δkm(G)≤\Delta k,与条件矛盾。

        # 充分条件 3

        给出了 Vizing 定理结论中 χ(G)=Δ+1\chi'(G)=\Delta+1 情况的充分条件

        GG 是奇数阶 Δ\Delta 正则单图,若 Δ>0\Delta>0,则:

        χ(G)=Δ(G)+1\chi'(G)=\Delta(G)+1


        证明:

        n=2k+1n=2k+1,因 GGΔ\Delta 正则单图,且 Δ>0\Delta>0,所以:

        m(G)=nΔ2=(2k+1)Δ2>kΔm(G)=\frac{n\Delta}{2}=\frac{(2k+1)\Delta}{2}\gt k\Delta

        由定理 χ(G)=Δ(G)+1\chi'(G)=\Delta(G)+1

        # 无环有重边图

        维津定理 (Vizing)

        设无环图 GG 中边的最大重数为 μ\mu​,则:

        χ(G)Δ(G)+μ\chi'(G)\leq\Delta(G)+\mu

        其中 \leq 边界是紧的,也就是等号是可以相等的。

        # 边着色的应用

        # 排课表问题

        X,YX,Y 分别表示教师、教学班的集合:

        X={x1,x2,,xm}Y={y1,y2,,yn}X=\{x_1,x_2,\cdots,x_m\}\\ Y=\{y_1,y_2,\cdots,y_n\}

        xix_iyjy_j 间连 pijp_{ij} 条边,得二部图 G=(X,Y)G=(X,Y)

        求最少排多少节课的问题,每节课可以有多个教学班同时上课,转化为如何在 GG 中将边集 EE 划分为互不相交的 pp 个匹配,且使得 pp 最小

        如果每个匹配中的边用同一种颜色着色,不同匹配中的边不同颜色,则问题转化为在 GG 中给每条边着色,相邻边着不同色,至少需要的颜色数

        # 比赛安排问题

        玩家为顶点,2 个玩家间进行一次比赛为一条边,求最少安排多少场比赛,每场比赛可以有多对玩家同时比赛。

        基于圈的边着色问题和顶点着色问题是完全等价的,因为边和点可互换。

        # 顶点着色

        对图 GG 的每个顶点着色,使得相邻顶点着不同颜色,称为对 GG正常顶点着色

        如果用 kk 种颜色可以对 GG 进行正常点着色,称 GGkk 可着色的

        着同一种颜色的顶点集合称为一个色组,它们彼此互不相邻,所有又称为点独立集

        GG 正常顶点着色需要的最少颜色数,称为图 GG点色数,简称色数,记为:

        χ(G)\chi(G)

        色数为 kk 的图称为 kk 色数图

        # 色数上界

        对任意的无环图 GG,均有:

        χ(G)Δ+1\chi(G)\leq\Delta+1

        任意一个顶点度数至多为 Δ\Delta, 因此,正常着色过程中,其邻点最多用去 Δ\Delta 种颜色,所以,至少还有一种色可供该点正常着色使用。


        证明 :我们对顶点数作数学归纳证明。

        n=1n=1 时,结论显然成立。

        设对顶点数少于 nn 的图来说,定理结论成立。考虑一般的 nn 阶图 GG,任取 vV(G)v\in V(G),令 G1=GvG_1=G-v, 由归纳假设:

        χ(G1)Δ(G1)+1Δ(G)+1\chi(G_1)\leq\Delta(G_1)+1\leq\Delta(G)+1

        π\piG1G_1 的一种 Δ(G1)+1\Delta(G_1)+1 正常点着色方案,因为 vv 的邻点在 π\pi 下至多用去 Δ(G)\Delta(G) 种色,所以给 vv 染上其邻点没有用过的色,就把 π\pi 扩充成了 GGΔ(G1)+1\Delta(G_1)+1 着色方案。

        # 最大色数的正常着色算法

        G=(V,E)G=(V,E)V={v1,v2,,vn}V=\{v_1,v_2,\cdots,v_n\},色集合 C={1,2,,Δ+1}C=\{1,2,\cdots,\Delta+1\},着色方案为 π\pi.

        • (1) 令 π(v1)=1\pi(v_1)=1
        • (2) for i in 1..n 循环 n1n-1 次:(若 i=ni=n,则停止)
          • 令:C(vi+1)={π(vj)ji,vjadjvi+1}C(v_{i+1})=\{\pi(v_j)|j≤i,\ v_j\ \text{adj}\ v_{i+1}\}
          • kkCC(vi+1)C-C(v_{i+1}) 中的最小整数;
          • π(vi+1)=k\pi(v_{i+1})=k

        Welsh-Powell 稍微对上面算法做了一个修改,着色时按所谓最大度优先策略,即使用上面算法时,V={v1,v2,,vn}V=\{v_1,v_2,\cdots,v_n\} 按顶点度数由大到小的次序着色。这样的着色方案起到了对上面算法的一个改进作用,有机会得到色数更小的结果。

        # 布鲁克斯定理

        给出了一些图的更小的上界

        GG 是连通的单图,并且它既不是奇圈,又不是完全图,则:

        χ(G)Δ(G)\chi(G)\leq\Delta(G)

        # 次大度

        GG 是至少有一条边的简单图,定义:

        Δ2(G)=maxuV(G)maxvN(u)d(v)d(u)d(v)\Delta_2(G)=\max_{u\in V(G)}\max_{\begin{aligned} v\in N(u)\\ d(v)\leq d(u) \end{aligned}} d(v)

        其中 N(u)N(u)GG 中点 uu 的邻域。称 Δ2(G)\Delta_2(G)GG 的次大度

        布鲁斯克定理的改进

        χ(G)Δ2(G)+1\chi(G)\leq\Delta_2(G)+1

        GG 是非空简单图,若 GG 中最大度点互不邻接,则有:

        χ(G)Δ(G)\chi(G)\leq\Delta(G)

        # 四色和五色定理

        # 四色定理

        # 五色定理

        希伍德。

        对任意简单图,都有:

        χ5\chi\leq5

        该定理说明每个平面图是 5 可着色的。根据平面图和其对偶图的关系,该定理等价于每个平面图是 5 可顶点正常着色的。

        \ No newline at end of file diff --git a/math/discrete-math/graph/connectivity/index.html b/math/discrete-math/graph/connectivity/index.html index 7679cc99..14b7ea4f 100644 --- a/math/discrete-math/graph/connectivity/index.html +++ b/math/discrete-math/graph/connectivity/index.html @@ -1 +1 @@ -图的连通性 - 图论 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 割边

        # 严格定义

        如果 ω(Ge)>ω(G)\omega(G-e)\gt\omega(G) ,则 eeGG 的一条割边

        # 充要条件

        eeGG 的割边当且仅当 ee 不在 GG 的任何圈中。


        必要性:

        假设 eeGG 的割边且 eeGG 的某圈中


        充分性:

        假设 ee 不在 GG 的任何圈中且 ee 不是 GG 的割边,则 GeG-e 连通。

        于是 GeG-e 中存在一条 (u,v)(u,v)ll ,显然 lel\cup e 就是 GG 的一个圈,与 ee 不在 GG 的任何圈中矛盾!

        # 推论 1

        eeGG 的一条边,若 ee 包含于 GG 的某圈中,则 GeG-e 连通。


        证明:若 GeG-e 不连通,则 ee 就是割边,这就与割边的充要条件矛盾。

        # 推论 2

        GG 的每个顶点的度数均为偶数,则 GG 没有割边;


        若不然,设 e=ue=uGG 的割边。则 GeG-e 的含有顶点 uu (或vv) 的那个分支中点 uu (或vv) 的度数为奇数,而其余点的度数为偶数,与握手定理推论相矛盾!

        # 推论 3

        GGkk 正则二部图 (k2)(k≥2) ,则 GG 无割边。


        若不然,设 e=ue=uGG 的割边。取 GeG-e 的其中一个分支 G1G_1 ,显然 G1G_1 中只有一个顶点的度数是 k1k-1 ,其余点的度数为 kk 。并且 G1G_1 仍然为偶图
        假若 GG 的两个顶点子集包含的顶点数分别为 mmnn ,并且包含 mm 个顶点的顶点子集包含度为 k1k-1 的那个点,那么有 km1=knkm-1= kn 。但是因 k2k≥2 ,所以等式不能成立!矛盾!

        # 割边集

        边割集跟回路、树等概念一样,是图论中重要概念。在应用上,它是电路网络图论的基本概念之一。

        破坏连通性的极小边子集

        #

        一个具有 nn 个顶点的连通图 GG ,定义 n1n-1 为该连通图的秩;

        具有 pp 个分支的图的秩定义为 npn-p ,记为 R(G)R(G)

        # 边割 (集)

        SS 是连通图 GG 的一个边子集,如果:

        • R(GS)=n2R(G-S)=n-2
        • SS 的任一真子集 SS ,有 R(GS)=n1R(G-S)=n-1

        SSGG 的一个边割集,简称 GG 的一个边割

        # 关联集

        GG 中,与顶点 vv 关联的边的集合,称为 vv关联集,记为:

        S(v)S(v)

        # 断集

        GG 中,如果 V=V1V2,V1V2=V=V_1∪V_2,V_1\cap V_2=\varnothing , E1E_1GG 中端点分属于 V1V_1V2V_2GG 的边子集,称 E1E_1GG 的一个断集。

        # 断集与关联集的关系

        任意一个断集均是若干关联集的环和 (对称差)

        E1E2=(E1E2)(E2E1)E_1\oplus E_2=(E_1-E_2)\cup(E_2-E_1)

        # 断集空间

        连通图 GG 的断集的集合作成子图空间的一个子空间,其维数为 n1n-1 ,该空间称为图的断集空间。(其基为 n1n-1 个线性无关的关联集)

        K4K_4 为例:

        K4

        其断集空间的基为 3 个线性无关的关联集:

        K4的3个线性无关的关联集

        形成的断集空间为:

        {,S(1),S(2),S(3),{a,c,d,e},{a,b,e,f},{b,c,d,f},{b,d,e}}\{\varnothing,S(1), S(2), S(3),\{a, c, d, e\},\{a, b, e, f\},\{b, c, d, f\},\{b, d, e\}\}

        # 基本 (边) 割集

        GG 是连通图,TTGG 的一棵生成树。如果 GG 的一个割集 SS 恰好包含 TT 的一条树枝,称 SSGG 的对于 TT 的一个基本 (边) 割集

        连通图 GG 的断集均可表为 GG 的对应于某生成树 TT 的基本割集的环和 (对称差)

        连通图 GG 对应于某生成树 TT 的基本割集的个数为 n1n-1 ,它们作成断集空间的一组基。

        # 割点

        # 严格定义

        GG 中,如果 E(G)E(G) 可以划分为两个非空子集 E1E_1E2E_2,使 G[E1]G[E_1]G[E2]G[E_2] 以点 vv 为公共顶点,称 vvGG 的一个割点

        # 充要条件

        # 无环非平凡图

        无环且非平凡情形下,割点与割边的定义一致

        GG 无环且非平凡,则 vvGG 的割点,当且仅当 ω(Gv)>ω(G)\omega(G-v)\gt\omega(G)


        必要性

        vvGG 的割点。则 E(G)E(G) 可划分为两个非空边子集 E1E_1E2E_2 ,使 G(E1),G(E2)G(E_1) , G(E_2) 恰好以 vv 为公共点。由于 GG 没有环,所以 G(E1),G(E2)G(E_1) , G(E_2) 分别至少包含异于 vvGG 的点,这样,GvG-v 的分支数比 GG 的分支数至少多 1,所以:ω(Gv)>ω(G)\omega(G-v)>\omega(G)


        充分性

        #

        vv 是树 TT 的顶点,则 vv 是割点,当且仅当 vv 是树的分支点(即不是叶子)

        # 无环连通图

        vv 是无环连通图 GG 的一个顶点,则 vvGG 的割点,当且仅当 V(Gv)V(G-v) 可以划分为两个非空子集 V1V_1V2V_2 ,使得对任意 xV1,vV2x\in V_1, v\in V_2 , 点 vv 在每一条 xyxy 路上

        无环非平凡连通图至少有两个非割点。

        证明:由于 GG 是无环非平凡连通图,所以存在非平凡生成树,而非平凡生成树至少两片树叶,它不能为割点,所以,它也不能为 GG 的割点。

        恰有两个非割点的连通简单图是一条路。

        证明:设 TTGG 的一棵生成树。由于 GGn2n-2 个割点,所以,TTn2n-2 个割点,即 T 只有两片树叶,所以 TT 是一条路。这说明,GG 的任意生成树为路。

        一个单图的任意生成树为路,则该图为圈或路,若为圈,则 GG 没有割点,矛盾,所以,GG 为路。

        v 是单图 G 的割点,则它不是 G 的补图的割点。

        证明:vv 是单图 GG 的割点,则 GvG-v 至少两个连通分支。现任取 x,yV(Gv)x,y\in V(\overline{G}-v) , 如果 x,yx,yGvG-v 的同一分支中,令 uu 是与 x,yx,y 处于不同分支的点,那么,通过 uu ,可说明,xxyyGvG-v 的补图中连通。若 x,yx,yGvG-v 的不同分支中,则它们在 GvG-v 的补图中邻接。所以,若 vvGG 的割点,则 vv 不是其补图的割点。

        #

        # 块 (图) 定义

        没有割点的连通图称为是一个块图,简称块GG 的一个子图 𝐵𝐵 如果:

        • 它本身是块
        • 若没有真包含 𝐵𝐵GG 的块存在(极大性 )

        BBGG 的一个块

        V(G)3|V(G)|\geq3GG 是块,当且仅当 GG 无环且任意两顶点位于同一圈上。

        vv 是图 GG 的割点当且仅当 vv 至少属于 GG 的两个不同的块

        # 块割点树

        GG 是非平凡连通图。𝐵1,𝐵2,,𝐵k𝐵_1,𝐵_2,\cdots,𝐵_kGG 的全部块,而 v1,v2,,vtv_1,v_2,\cdots,v_tGG 的全部割点。构作 GG 的块割点树 bc(G)bc(G) : 它的顶点是 GG 的块和割点,连线只在块割点之间进行,一个块和一个割点连线,当且仅当该割点是该块的一个顶点。

        # 连通度

        # 点连通度

        GG 中:

        • GG 连通:

          • 若存在顶点割,称 GG 的最小顶点割的顶点数称为 GG 的点连通度;
          • 否则称 n-1 为其点连通度. G 的点连通度记为 κ(G)\kappa(G),简记为 κ\kappa
        • GG 不连通,κ(G)=0\kappa(G)=0

        # 边连通度

        GG 中,最小边割集所含边数称为 GG 的边连通度。边连通度记为 λ(G)\lambda(G)

        GG 不连通或 GG 是平凡图,则定义 λ(G)=0\lambda(G)=0

        # k - 连通

        • κ(G)k\kappa(G)\geq kGGk - 连通的。
        • λ(G)k\lambda(G)\geq kGGk - 边连通的。

        # 惠特尼连通度定理

        κ(G)λ(G)(G)\kappa(G)\leq\lambda(G)\leq(G)

        abc,G:κ(G)=a,λ(G)=b,(G)=c\forall a\leq b\leq c\ \ ,\ \ \exist G:\\ \kappa(G)=a,\ \ \lambda(G)=b,\ \ (G)=c

        κ(G)2mn\kappa(G)\leq\lfloor\frac{2m}{n}\rfloor

        GG(n,m)(n,m) 单图,若 δ(G)n2\delta(G)≥\lfloor\frac{n}{2}\rfloor ,则 GG 连通

        # k - 连通充分条件

        GGnn 阶简单图,若对任意正整数 knk<n ,有:

        δ(G)n+k22\delta(G)\geq\frac{n+k-2}{2}

        GGkk 连通的。

        # 哈拉里图

        # 坚韧性

        C(G)C(G) 表示图 GG 的全体点割集构成的集合,非平凡非完全图的坚韧度,记作 τ(G)\tau(G) ,定义为:

        τ(G)=min{Sω(GS)SC(G)}\tau(G)=\min\{ \frac{|S|}{\omega(G-S)}\big|S\in C(G) \}

        GG 是一个非完全 n(n3)n(n≥3) 阶连通图,SC(G)S\in C(G) ,若 SS^* 满足:

        τ(G)=Sω(GS)\tau(G)=\frac{|S^*|}{\omega(G-S^*)}

        SS^*GG坚韧集

        # 敏格尔定理

        # 分离集

        uuvv 是图 GG 的两个不同顶点,SS 表示 GG 的一个顶点子集或边子集,如果 uuvv 不在 GSG-S 的同一分支上,称 SS 分离 uuvvSSuuvv分离 (点 / 边) 集

        # Menger

        n - 弧定理

        xxyy 是图 GG 中的两个不相邻点,则GG 中分离点 xxyy 的:

        • 最少点数等于 GG 中独立(内点不交)的 (x,y)(x,y) 路的最大数目;
        • 最少边数等于 GG 中边不重的 (x,y)(x,y) 路的最大数目;

        # 惠特尼定理

        一个非平凡的图 GGkk 连通 k2k\geq2 的,当且仅当 GG 的任意两个顶点间至少存在 kk 条:(以下两种条件之一)

        • 独立(内点不交)的 (u,v)(u,v) 路;
        • 边不重的 (u,v)(u,v) 路;

        GGkk 连通图,SS 是由 GG 中任意 kk 个顶点构成的集合。若图 HH 是由 GG 通过添加一个新点 ww 以及连接 wwSS 中所有顶点得到的新图,求证:HHkk - 连通的。


        证明:

        首先, GGkk 连通图,所以分离 GG 中两个不相邻顶点至少要 kk 个点;

        其次,分离 wwGG 中不在 SS 中顶点需要 kk 个顶点;

        因此 HHkk 连通的。

        GGkk - 连通图,u,v1,v2,,vku,v_1,v_2,\cdots,v_kGGk+1k+1 个不同顶点。求证:GG 中有 kk 条内点不交路 (u,vi)(1ik)(u,v_i)\ (1≤i≤k)

        对于一个阶至少为 3 的无环图 GG ,下面三个命题等价。

        • (1) GG 是 2 - 连通的;
        • (2) GG 中任意两点位于同一个圈上;
        • (3) GG 无孤立点,且任意两条边在同一个圈上;

        # 图的宽直径

        所有距离(distance)中最大的就是直径(diameter),这里都记为 d(G)d(G)

        GG 是强连通有向图,若其阶 n3n\geq3 且最大度 Δ2\Delta\geq2 ,则:

        d(G){n2,Δ=2log(Δ1)n(Δ2)+2Δ,Δ3d(G)\geq\begin{cases} \lfloor\frac{n}{2}\rfloor, & \Delta=2\\ \lceil\log_{(\Delta-1)}\frac{n(\Delta-2)+2}{\Delta}\rceil, & \Delta\geq3\\ \end{cases}

        # 路族 / 容器

        x,yV(G)x,y\in V(G) , Cw(x,y)C_w(x,y) 表示 GGww 条内点不交路的路族,ww 称为路族的宽度,Cw(x,y)C_w(x,y) 中最长路的路长成为该路族的长度,记为:l(Cw(x,y))l (C_w(x,y))

        # w 宽距离和最优路族

        x,yV(G)x,y\in V(G) , 定义 xxyy 间所有宽度为 ww 的路族长度的最小值 dw(x,y)d_w(x,y)xxyyww 宽距离,xxyy 间长度等于 ww 宽距离的路族称为 xxyy 间最优路族。

        所以,求 ww 宽距离,就是要找到最优路族。

        # 宽直径

        GGww 连通的,GG 的所有点对间的 ww 宽距离的最大值,称为 GGww 宽直径,记为 dw(G)d_w(G )

        例如:⭐️

        • nn 点圈 CnC_n:(连通度 = 2)
          • d1(Cn)=n2d_1(C_n)=\lfloor\frac{n}{2}\rfloor
          • d2(Cn)=n1d_2(C_n)=n-1
        • kk 阶完全图 KnK_n:(连通度 = n1n-1
          • d1(Kn)=1d_1(K_n)=1
          • dw(Kn)=2d_w(K_n)=2;(2wn12\leq w\leq n-1

        容错直径(了解)

        \ No newline at end of file +图的连通性 - 图论 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 割边

        # 严格定义

        如果 ω(Ge)>ω(G)\omega(G-e)\gt\omega(G) ,则 eeGG 的一条割边

        # 充要条件

        eeGG 的割边当且仅当 ee 不在 GG 的任何圈中。


        必要性:

        假设 eeGG 的割边且 eeGG 的某圈中


        充分性:

        假设 ee 不在 GG 的任何圈中且 ee 不是 GG 的割边,则 GeG-e 连通。

        于是 GeG-e 中存在一条 (u,v)(u,v)ll ,显然 lel\cup e 就是 GG 的一个圈,与 ee 不在 GG 的任何圈中矛盾!

        # 推论 1

        eeGG 的一条边,若 ee 包含于 GG 的某圈中,则 GeG-e 连通。


        证明:若 GeG-e 不连通,则 ee 就是割边,这就与割边的充要条件矛盾。

        # 推论 2

        GG 的每个顶点的度数均为偶数,则 GG 没有割边;


        若不然,设 e=ue=uGG 的割边。则 GeG-e 的含有顶点 uu (或vv) 的那个分支中点 uu (或vv) 的度数为奇数,而其余点的度数为偶数,与握手定理推论相矛盾!

        # 推论 3

        GGkk 正则二部图 (k2)(k≥2) ,则 GG 无割边。


        若不然,设 e=ue=uGG 的割边。取 GeG-e 的其中一个分支 G1G_1 ,显然 G1G_1 中只有一个顶点的度数是 k1k-1 ,其余点的度数为 kk 。并且 G1G_1 仍然为偶图
        假若 GG 的两个顶点子集包含的顶点数分别为 mmnn ,并且包含 mm 个顶点的顶点子集包含度为 k1k-1 的那个点,那么有 km1=knkm-1= kn 。但是因 k2k≥2 ,所以等式不能成立!矛盾!

        # 割边集

        边割集跟回路、树等概念一样,是图论中重要概念。在应用上,它是电路网络图论的基本概念之一。

        破坏连通性的极小边子集

        #

        一个具有 nn 个顶点的连通图 GG ,定义 n1n-1 为该连通图的秩;

        具有 pp 个分支的图的秩定义为 npn-p ,记为 R(G)R(G)

        # 边割 (集)

        SS 是连通图 GG 的一个边子集,如果:

        • R(GS)=n2R(G-S)=n-2
        • SS 的任一真子集 SS ,有 R(GS)=n1R(G-S)=n-1

        SSGG 的一个边割集,简称 GG 的一个边割

        # 关联集

        GG 中,与顶点 vv 关联的边的集合,称为 vv关联集,记为:

        S(v)S(v)

        # 断集

        GG 中,如果 V=V1V2,V1V2=V=V_1∪V_2,V_1\cap V_2=\varnothing , E1E_1GG 中端点分属于 V1V_1V2V_2GG 的边子集,称 E1E_1GG 的一个断集。

        # 断集与关联集的关系

        任意一个断集均是若干关联集的环和 (对称差)

        E1E2=(E1E2)(E2E1)E_1\oplus E_2=(E_1-E_2)\cup(E_2-E_1)

        # 断集空间

        连通图 GG 的断集的集合作成子图空间的一个子空间,其维数为 n1n-1 ,该空间称为图的断集空间。(其基为 n1n-1 个线性无关的关联集)

        K4K_4 为例:

        K4

        其断集空间的基为 3 个线性无关的关联集:

        K4的3个线性无关的关联集

        形成的断集空间为:

        {,S(1),S(2),S(3),{a,c,d,e},{a,b,e,f},{b,c,d,f},{b,d,e}}\{\varnothing,S(1), S(2), S(3),\{a, c, d, e\},\{a, b, e, f\},\{b, c, d, f\},\{b, d, e\}\}

        # 基本 (边) 割集

        GG 是连通图,TTGG 的一棵生成树。如果 GG 的一个割集 SS 恰好包含 TT 的一条树枝,称 SSGG 的对于 TT 的一个基本 (边) 割集

        连通图 GG 的断集均可表为 GG 的对应于某生成树 TT 的基本割集的环和 (对称差)

        连通图 GG 对应于某生成树 TT 的基本割集的个数为 n1n-1 ,它们作成断集空间的一组基。

        # 割点

        # 严格定义

        GG 中,如果 E(G)E(G) 可以划分为两个非空子集 E1E_1E2E_2,使 G[E1]G[E_1]G[E2]G[E_2] 以点 vv 为公共顶点,称 vvGG 的一个割点

        # 充要条件

        # 无环非平凡图

        无环且非平凡情形下,割点与割边的定义一致

        GG 无环且非平凡,则 vvGG 的割点,当且仅当 ω(Gv)>ω(G)\omega(G-v)\gt\omega(G)


        必要性

        vvGG 的割点。则 E(G)E(G) 可划分为两个非空边子集 E1E_1E2E_2 ,使 G(E1),G(E2)G(E_1) , G(E_2) 恰好以 vv 为公共点。由于 GG 没有环,所以 G(E1),G(E2)G(E_1) , G(E_2) 分别至少包含异于 vvGG 的点,这样,GvG-v 的分支数比 GG 的分支数至少多 1,所以:ω(Gv)>ω(G)\omega(G-v)>\omega(G)


        充分性

        #

        vv 是树 TT 的顶点,则 vv 是割点,当且仅当 vv 是树的分支点(即不是叶子)

        # 无环连通图

        vv 是无环连通图 GG 的一个顶点,则 vvGG 的割点,当且仅当 V(Gv)V(G-v) 可以划分为两个非空子集 V1V_1V2V_2 ,使得对任意 xV1,vV2x\in V_1, v\in V_2 , 点 vv 在每一条 xyxy 路上

        无环非平凡连通图至少有两个非割点。

        证明:由于 GG 是无环非平凡连通图,所以存在非平凡生成树,而非平凡生成树至少两片树叶,它不能为割点,所以,它也不能为 GG 的割点。

        恰有两个非割点的连通简单图是一条路。

        证明:设 TTGG 的一棵生成树。由于 GGn2n-2 个割点,所以,TTn2n-2 个割点,即 T 只有两片树叶,所以 TT 是一条路。这说明,GG 的任意生成树为路。

        一个单图的任意生成树为路,则该图为圈或路,若为圈,则 GG 没有割点,矛盾,所以,GG 为路。

        v 是单图 G 的割点,则它不是 G 的补图的割点。

        证明:vv 是单图 GG 的割点,则 GvG-v 至少两个连通分支。现任取 x,yV(Gv)x,y\in V(\overline{G}-v) , 如果 x,yx,yGvG-v 的同一分支中,令 uu 是与 x,yx,y 处于不同分支的点,那么,通过 uu ,可说明,xxyyGvG-v 的补图中连通。若 x,yx,yGvG-v 的不同分支中,则它们在 GvG-v 的补图中邻接。所以,若 vvGG 的割点,则 vv 不是其补图的割点。

        #

        # 块 (图) 定义

        没有割点的连通图称为是一个块图,简称块GG 的一个子图 𝐵𝐵 如果:

        • 它本身是块
        • 若没有真包含 𝐵𝐵GG 的块存在(极大性 )

        BBGG 的一个块

        V(G)3|V(G)|\geq3GG 是块,当且仅当 GG 无环且任意两顶点位于同一圈上。

        vv 是图 GG 的割点当且仅当 vv 至少属于 GG 的两个不同的块

        # 块割点树

        GG 是非平凡连通图。𝐵1,𝐵2,,𝐵k𝐵_1,𝐵_2,\cdots,𝐵_kGG 的全部块,而 v1,v2,,vtv_1,v_2,\cdots,v_tGG 的全部割点。构作 GG 的块割点树 bc(G)bc(G) : 它的顶点是 GG 的块和割点,连线只在块割点之间进行,一个块和一个割点连线,当且仅当该割点是该块的一个顶点。

        # 连通度

        # 点连通度

        GG 中:

        • GG 连通:

          • 若存在顶点割,称 GG 的最小顶点割的顶点数称为 GG 的点连通度;
          • 否则称 n-1 为其点连通度. G 的点连通度记为 κ(G)\kappa(G),简记为 κ\kappa
        • GG 不连通,κ(G)=0\kappa(G)=0

        # 边连通度

        GG 中,最小边割集所含边数称为 GG 的边连通度。边连通度记为 λ(G)\lambda(G)

        GG 不连通或 GG 是平凡图,则定义 λ(G)=0\lambda(G)=0

        # k - 连通

        • κ(G)k\kappa(G)\geq kGGk - 连通的。
        • λ(G)k\lambda(G)\geq kGGk - 边连通的。

        # 惠特尼连通度定理

        κ(G)λ(G)(G)\kappa(G)\leq\lambda(G)\leq(G)

        abc,G:κ(G)=a,λ(G)=b,(G)=c\forall a\leq b\leq c\ \ ,\ \ \exist G:\\ \kappa(G)=a,\ \ \lambda(G)=b,\ \ (G)=c

        κ(G)2mn\kappa(G)\leq\lfloor\frac{2m}{n}\rfloor

        GG(n,m)(n,m) 单图,若 δ(G)n2\delta(G)≥\lfloor\frac{n}{2}\rfloor ,则 GG 连通

        # k - 连通充分条件

        GGnn 阶简单图,若对任意正整数 knk<n ,有:

        δ(G)n+k22\delta(G)\geq\frac{n+k-2}{2}

        GGkk 连通的。

        # 哈拉里图

        # 坚韧性

        C(G)C(G) 表示图 GG 的全体点割集构成的集合,非平凡非完全图的坚韧度,记作 τ(G)\tau(G) ,定义为:

        τ(G)=min{Sω(GS)SC(G)}\tau(G)=\min\{ \frac{|S|}{\omega(G-S)}\big|S\in C(G) \}

        GG 是一个非完全 n(n3)n(n≥3) 阶连通图,SC(G)S\in C(G) ,若 SS^* 满足:

        τ(G)=Sω(GS)\tau(G)=\frac{|S^*|}{\omega(G-S^*)}

        SS^*GG坚韧集

        # 敏格尔定理

        # 分离集

        uuvv 是图 GG 的两个不同顶点,SS 表示 GG 的一个顶点子集或边子集,如果 uuvv 不在 GSG-S 的同一分支上,称 SS 分离 uuvvSSuuvv分离 (点 / 边) 集

        # Menger

        n - 弧定理

        xxyy 是图 GG 中的两个不相邻点,则GG 中分离点 xxyy 的:

        • 最少点数等于 GG 中独立(内点不交)的 (x,y)(x,y) 路的最大数目;
        • 最少边数等于 GG 中边不重的 (x,y)(x,y) 路的最大数目;

        # 惠特尼定理

        一个非平凡的图 GGkk 连通 k2k\geq2 的,当且仅当 GG 的任意两个顶点间至少存在 kk 条:(以下两种条件之一)

        • 独立(内点不交)的 (u,v)(u,v) 路;
        • 边不重的 (u,v)(u,v) 路;

        GGkk 连通图,SS 是由 GG 中任意 kk 个顶点构成的集合。若图 HH 是由 GG 通过添加一个新点 ww 以及连接 wwSS 中所有顶点得到的新图,求证:HHkk - 连通的。


        证明:

        首先, GGkk 连通图,所以分离 GG 中两个不相邻顶点至少要 kk 个点;

        其次,分离 wwGG 中不在 SS 中顶点需要 kk 个顶点;

        因此 HHkk 连通的。

        GGkk - 连通图,u,v1,v2,,vku,v_1,v_2,\cdots,v_kGGk+1k+1 个不同顶点。求证:GG 中有 kk 条内点不交路 (u,vi)(1ik)(u,v_i)\ (1≤i≤k)

        对于一个阶至少为 3 的无环图 GG ,下面三个命题等价。

        • (1) GG 是 2 - 连通的;
        • (2) GG 中任意两点位于同一个圈上;
        • (3) GG 无孤立点,且任意两条边在同一个圈上;

        # 图的宽直径

        所有距离(distance)中最大的就是直径(diameter),这里都记为 d(G)d(G)

        GG 是强连通有向图,若其阶 n3n\geq3 且最大度 Δ2\Delta\geq2 ,则:

        d(G){n2,Δ=2log(Δ1)n(Δ2)+2Δ,Δ3d(G)\geq\begin{cases} \lfloor\frac{n}{2}\rfloor, & \Delta=2\\ \lceil\log_{(\Delta-1)}\frac{n(\Delta-2)+2}{\Delta}\rceil, & \Delta\geq3\\ \end{cases}

        # 路族 / 容器

        x,yV(G)x,y\in V(G) , Cw(x,y)C_w(x,y) 表示 GGww 条内点不交路的路族,ww 称为路族的宽度,Cw(x,y)C_w(x,y) 中最长路的路长成为该路族的长度,记为:l(Cw(x,y))l (C_w(x,y))

        # w 宽距离和最优路族

        x,yV(G)x,y\in V(G) , 定义 xxyy 间所有宽度为 ww 的路族长度的最小值 dw(x,y)d_w(x,y)xxyyww 宽距离,xxyy 间长度等于 ww 宽距离的路族称为 xxyy 间最优路族。

        所以,求 ww 宽距离,就是要找到最优路族。

        # 宽直径

        GGww 连通的,GG 的所有点对间的 ww 宽距离的最大值,称为 GGww 宽直径,记为 dw(G)d_w(G )

        例如:⭐️

        • nn 点圈 CnC_n:(连通度 = 2)
          • d1(Cn)=n2d_1(C_n)=\lfloor\frac{n}{2}\rfloor
          • d2(Cn)=n1d_2(C_n)=n-1
        • kk 阶完全图 KnK_n:(连通度 = n1n-1
          • d1(Kn)=1d_1(K_n)=1
          • dw(Kn)=2d_w(K_n)=2;(2wn12\leq w\leq n-1

        容错直径(了解)

        \ No newline at end of file diff --git a/math/discrete-math/graph/euler_hamilton/index.html b/math/discrete-math/graph/euler_hamilton/index.html index 2d6e1636..0eb5eefa 100644 --- a/math/discrete-math/graph/euler_hamilton/index.html +++ b/math/discrete-math/graph/euler_hamilton/index.html @@ -1 +1 @@ -欧拉图与哈密顿图 - 图论 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 欧拉图

        # 定义

        欧拉图存在欧拉闭迹,半欧拉图只存在欧拉迹

        若能遍历完所有的边但是没法回到起始点,称为非欧拉图但是有欧拉迹

        # 欧拉图

        假定 GG 是一个连通图,且是欧拉图,则下列命题等价:

        • GG 是欧拉图
        • GG 的每个点的度数是偶数
        • GG 的边集能划分为边不重的圈的并
        • GG 存在欧拉闭迹)

        # 半欧拉图

        假定 GG 是一个连通图,且是半欧拉图,则下列命题等价:

        • GG 是半欧拉图
        • GG 有且仅有 2 个点的度数是奇数
        • GG 存在欧拉迹)

        # 小结论💡

        • nn 满足什么条件时,完全图 KnK_n 是欧拉图? nn 为奇数
        • nn 满足什么条件时,nn 方体 QnQ_n 是欧拉图? nn 为偶数
        • 若完全二部图 Ka,bK_{a,b} 为欧拉图,aabb 需满足什么条件? a,ba,b 均为偶数
        • 假设图 GG 恰好有两个连通分支,并且每个连通分支都是欧拉图,若要将 GG 变为欧拉图,最少需要添加几条边? 最少需要添加 2 条边
        • 欧拉图、半欧拉图是否存在割边? 欧拉图不存在割边,半欧拉图存在割边

        # Fleury 算法

        弗勒里算法解决了在欧拉图中求出一条具体欧拉环游的方法,核心思想是尽可能避割边行走

        # 算法流程

        • (1)任意选取一个顶点 v0v_0 ,置 w0=v0w_0=v_0
        • (2)假设迹 wi=v0e1v1eiviw_i=v_0e_1v_1\cdots e_iv_i 已经确定,按如下方法从 E{e1,e2,,ei}E-\{e_1,e_2,\cdots,e_{i}\} 选一个 ei+1e_{i+1}
          • ei+1e_{i+1}viv_i 相关联;
          • 除非没有其他边可选,否则 ei+1e_{i+1} 不能是 Gi=G{e1,e2,,ei}G_i=G-\{e_1,e_2,\cdots,e_{i}\}割边
        • (3)若(2)无法进行则算法停止

        利用 Fluery 算法可以求一个半欧拉图的欧拉迹,半欧拉图有且仅有 2 个奇度顶点 u,vu,v ,在这 2 个点之间增加一条边 mm ,就可以得到一个欧拉图,然后从其中一个奇度顶点,例如 uu ,开始求欧拉环游。得到的欧拉环游去掉 mm 就是从 uuvv 的欧拉迹。

        # 欧拉图中扩展欧拉环游充要条件

        GG 是非平凡的欧拉图,且 vV(G)v\in V(G) ,则 GG 的每条具有起点 vv 的迹都能扩展成 GG 的欧拉环游当且仅当 GvG-v 是森林。

        # 中国邮递员问题

        1962,中国数学家管梅谷提出并解决了 “中国邮路问题”

        即最优环游问题:在一个连通具有非负权的赋权图 GG 中找到一条包含每条边(允许重复)且边权之和最小的闭途径,称之为最优 (欧拉) 环游

        • 若图 GG 是一个欧拉图,则找出 GG 的欧拉环游即可
        • 对一般图,其解法为:添加重复边以使 GG 成为欧拉图 GG^* ,并使添加的重复边的边权之和为最小,再求 GG^* 的欧拉回路

        管梅谷的结论:

        假定 GG^* 是在图 GG 中添加一些重复边得到的欧拉图,则 GG^* 具有最小权值的充要条件是:

        • GG 的每一条边最多被添加一次
        • 对于 GG^* 的每个圈来说,该圈新添加的边的总权值不超过该圈总权值的一半

        证明:必要性

        GG 中某条边在 GG^* 中被添加的次数超过 1,则去掉其中两条重复的边,我们将得到一个总权值更小,且以 GG 为生成子图的欧拉图

        上述与 “ GG^* 总权值最小” 相矛盾,因此每条边最多被添加一次

        假定在 GG^* 中存在某个圈使得该圈新添加的边的总权值大于该圈权值的一半,不妨设为 CC

        那么在 GG^* 中,将 CC 上新添加的边全部去掉,然后将原圈未添加新边的边都添加一条重复边

        这样的操作使得该圈所有点的度数不变或改变 2,相应的图仍然是欧拉图,但是权值更小,这与 “ GG^* 总权值最小” 相矛盾

        这个必要性的证明给出了最优欧拉环游的构造方法

        # 半欧拉图最优欧拉环游

        一个非负权的边赋权图 GG 中只有两个奇度顶点 uuvv , 设计一个求其最优欧拉环游的算法如下:

        • (1) 在 uuvv 间求出一条最短路 PP; (最短路算法)
        • (2) 在最短路 PP 上,给每条边添加一条重复边得 GG 的欧拉图 GG^*
        • (3) 在 GG 的欧拉图 GG^* 中用 Fleury 算法求出一条欧拉环游;

        证明

        uuvvGG 的两个奇度顶点,GG^*GG 的任意一个欧拉图。

        考虑 G[EE]G^*[E^*-E], 显然它只有两个奇数顶点 uuvv, 当然它们必须在 G[EE]G^*[E^*-E] 的同一个分支中,因此,存在 (u,v)(u,v)PP^*

        eEEw(e)w(P)w(P)\sum_{e\in E^*-E}w(e)\geq w(P^*)\geq w(P)

        # 哈密顿图

        # 定义

        # 哈姆顿图

        如果经过图 GG 的每个顶点恰好一次后能够回到出发点,称这样的图为哈密尔顿图,简称 HH。所经过的闭途径是 GG 的一个生成圈,称为 GG哈密尔顿圈,简称 HH

        # 半哈密顿图

        不需回到出发点

        如果存在经过 GG 的每个顶点恰好一次的路,称这样的图为半哈密尔顿图,称该路为 GG哈密尔顿路,简称 HH

        # 小结论💡

        • n3n\geq3 时,完全图 KnK_n 是否为哈密尔顿图?
        • n2n\geq2 时, 方体 QnQ_n 是否为哈密尔顿图?
        • 若完全二部图 Ka,bK_{a,b} 为哈密尔顿图,aabb 需满足什么条件?a=b2\Rightarrow a=b\geq2
        • 是否存在一个具有奇数个顶点的连通图,它既是二部图,也是哈密尔顿图? 不存在,否则二部图中出现了奇圈
        • 若二部图是哈密尔顿图,则它的二部划分 (X,Y)(X,Y) 满足什么条件?X=Y\Rightarrow|X|=|Y|

        哈密顿图目前为止没有充要条件,实际上图的 HH 性判定是 NP - 困难问题,下面介绍一个必要条件和一些充分条件

        # 必要条件

        GGHH 图,则对于 VV 的每个非空真子集 SS,均有:

        ω(GS)S\omega(G-S)\leq|S|

        💡别忘了 ω(...)\omega(...) 的意思是连通分支数,这里还容易得到 HH 图不存在割点

        只是必要条件而不是充分条件,经典例子 —— 彼得森图

        用于判断非 HH 图:SV,ω(GS)>S\exist S\subset V,\ \ \omega(G-S)\gt|S| ,则 GG 不是 HH 图。

        # 充分条件

        # Dirac

        对于 n3n≥3 的简单图 GG , 如果 GG 中有:

        δ(G)n2\delta(G)\geq\frac{n}{2}

        那么 GGHH 图。

        # Ore

        对于 n3n≥3 的单图 GG , 如果 GG 中的任意两个不相邻顶点 uuvv ,有:

        d(u)+d(v)nd(u)+d(v)\geq n

        那么 GGHH 图。

        该定理证明和 Dirac 定理可以完全一致!

        该定理的条件是紧的。例如:设 GG 是由 Kk+1Κ_{k+1} 的一个顶点和另一个 Kk+1Κ_{k+1} 的一个顶点重合得到的图,那么对于 GG 的任意两个不相邻顶点 uuvv,有:

        d(u)+d(v)=2k=n1d(u)+d(v)=2k=n-1

        GG 是非 HH 图。

        # Bondy

        # 引理

        对于单图 GG ,如果 GG 中有两个不相邻顶点 uuvv ,满足:d(u)+d(v)nd(u)+d(v)≥n,那么 GGHH 图当且仅当 G+uvG+uvHH 图。

        # 闭图

        nn 阶简单图中,若对 d(u)+d(v)nd(u)+d(v)≥n 的任意一对顶点 uuvv ,均有 uadjvu\ \text{adj}\ v , 则称 GG 是闭图。

        # 引理

        G1G_1G2G_2 是同一个点集 VV 的两个闭图,则 G1G2G_1\cap G_2 是闭图,但 G1G2G_1\cup G_2 不一定是闭图。

        # 闭包

        G\overline G 是图 GG 的闭包,如果它是包含 GG 的极小闭图,即对 H∀H,若有 GHGG\subseteq H \subset\overline G,则 HH 不是闭图。

        • 如果 GG 本身是闭图,则其闭包是它本身;
        • 否则由定义可以通过在度和 n\geq n 的不相邻顶点对间加边来构造 GG 的闭图;加边之后还要继续检查是否需要加边。

        GG 的闭包是唯一的。

        # 闭包定理

        由闭包定理也可以推出 Dirac 和 Ore 定理

        GGHH 图当且仅当它的闭包是 HH

        # Chavatal

        度序列判定法

        HH 图充分条件

        设简单图 GG 的度序列是 d(d1,d2,,dn)d(d_1,d_2,\cdots,d_n),这里,d1d2dnd_1≤d_2≤⋯≤d_n,并且 n3n≥3。若对任意的 m<n2m<\frac{n}{2}

        • 或有 dm>md_m>m
        • 或有 dnmnmd_{n-m}≥n-m

        GGHH 图。(GG 的闭包是完全图!证明也是证明这个)

        # TSP

        # 边交换

        # 最优 H 圈下界

        可以通过如下方法求出最优 HH 圈的一个下界:(不是下确界,不一定达得到!)

        (1) 在 GG 中删掉一点 vv (任意的) 得图 G1G_1

        (2) 在图 G1G_1 中求出一棵最小生成树 TT

        (3) 在 vv 的关联边中选出两条权值最小者 e1e_1e2e_2

        HHGG 的最优圈,则有权值下界:

        W(H)W(T)+W(e1)+W(e2)W(H)\geq W(T)+W(e_1)+W(e_2)

        \ No newline at end of file +欧拉图与哈密顿图 - 图论 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 欧拉图

        # 定义

        欧拉图存在欧拉闭迹,半欧拉图只存在欧拉迹

        若能遍历完所有的边但是没法回到起始点,称为非欧拉图但是有欧拉迹

        # 欧拉图

        假定 GG 是一个连通图,且是欧拉图,则下列命题等价:

        • GG 是欧拉图
        • GG 的每个点的度数是偶数
        • GG 的边集能划分为边不重的圈的并
        • GG 存在欧拉闭迹)

        # 半欧拉图

        假定 GG 是一个连通图,且是半欧拉图,则下列命题等价:

        • GG 是半欧拉图
        • GG 有且仅有 2 个点的度数是奇数
        • GG 存在欧拉迹)

        # 小结论💡

        • nn 满足什么条件时,完全图 KnK_n 是欧拉图? nn 为奇数
        • nn 满足什么条件时,nn 方体 QnQ_n 是欧拉图? nn 为偶数
        • 若完全二部图 Ka,bK_{a,b} 为欧拉图,aabb 需满足什么条件? a,ba,b 均为偶数
        • 假设图 GG 恰好有两个连通分支,并且每个连通分支都是欧拉图,若要将 GG 变为欧拉图,最少需要添加几条边? 最少需要添加 2 条边
        • 欧拉图、半欧拉图是否存在割边? 欧拉图不存在割边,半欧拉图存在割边

        # Fleury 算法

        弗勒里算法解决了在欧拉图中求出一条具体欧拉环游的方法,核心思想是尽可能避割边行走

        # 算法流程

        • (1)任意选取一个顶点 v0v_0 ,置 w0=v0w_0=v_0
        • (2)假设迹 wi=v0e1v1eiviw_i=v_0e_1v_1\cdots e_iv_i 已经确定,按如下方法从 E{e1,e2,,ei}E-\{e_1,e_2,\cdots,e_{i}\} 选一个 ei+1e_{i+1}
          • ei+1e_{i+1}viv_i 相关联;
          • 除非没有其他边可选,否则 ei+1e_{i+1} 不能是 Gi=G{e1,e2,,ei}G_i=G-\{e_1,e_2,\cdots,e_{i}\}割边
        • (3)若(2)无法进行则算法停止

        利用 Fluery 算法可以求一个半欧拉图的欧拉迹,半欧拉图有且仅有 2 个奇度顶点 u,vu,v ,在这 2 个点之间增加一条边 mm ,就可以得到一个欧拉图,然后从其中一个奇度顶点,例如 uu ,开始求欧拉环游。得到的欧拉环游去掉 mm 就是从 uuvv 的欧拉迹。

        # 欧拉图中扩展欧拉环游充要条件

        GG 是非平凡的欧拉图,且 vV(G)v\in V(G) ,则 GG 的每条具有起点 vv 的迹都能扩展成 GG 的欧拉环游当且仅当 GvG-v 是森林。

        # 中国邮递员问题

        1962,中国数学家管梅谷提出并解决了 “中国邮路问题”

        即最优环游问题:在一个连通具有非负权的赋权图 GG 中找到一条包含每条边(允许重复)且边权之和最小的闭途径,称之为最优 (欧拉) 环游

        • 若图 GG 是一个欧拉图,则找出 GG 的欧拉环游即可
        • 对一般图,其解法为:添加重复边以使 GG 成为欧拉图 GG^* ,并使添加的重复边的边权之和为最小,再求 GG^* 的欧拉回路

        管梅谷的结论:

        假定 GG^* 是在图 GG 中添加一些重复边得到的欧拉图,则 GG^* 具有最小权值的充要条件是:

        • GG 的每一条边最多被添加一次
        • 对于 GG^* 的每个圈来说,该圈新添加的边的总权值不超过该圈总权值的一半

        证明:必要性

        GG 中某条边在 GG^* 中被添加的次数超过 1,则去掉其中两条重复的边,我们将得到一个总权值更小,且以 GG 为生成子图的欧拉图

        上述与 “ GG^* 总权值最小” 相矛盾,因此每条边最多被添加一次

        假定在 GG^* 中存在某个圈使得该圈新添加的边的总权值大于该圈权值的一半,不妨设为 CC

        那么在 GG^* 中,将 CC 上新添加的边全部去掉,然后将原圈未添加新边的边都添加一条重复边

        这样的操作使得该圈所有点的度数不变或改变 2,相应的图仍然是欧拉图,但是权值更小,这与 “ GG^* 总权值最小” 相矛盾

        这个必要性的证明给出了最优欧拉环游的构造方法

        # 半欧拉图最优欧拉环游

        一个非负权的边赋权图 GG 中只有两个奇度顶点 uuvv , 设计一个求其最优欧拉环游的算法如下:

        • (1) 在 uuvv 间求出一条最短路 PP; (最短路算法)
        • (2) 在最短路 PP 上,给每条边添加一条重复边得 GG 的欧拉图 GG^*
        • (3) 在 GG 的欧拉图 GG^* 中用 Fleury 算法求出一条欧拉环游;

        证明

        uuvvGG 的两个奇度顶点,GG^*GG 的任意一个欧拉图。

        考虑 G[EE]G^*[E^*-E], 显然它只有两个奇数顶点 uuvv, 当然它们必须在 G[EE]G^*[E^*-E] 的同一个分支中,因此,存在 (u,v)(u,v)PP^*

        eEEw(e)w(P)w(P)\sum_{e\in E^*-E}w(e)\geq w(P^*)\geq w(P)

        # 哈密顿图

        # 定义

        # 哈姆顿图

        如果经过图 GG 的每个顶点恰好一次后能够回到出发点,称这样的图为哈密尔顿图,简称 HH。所经过的闭途径是 GG 的一个生成圈,称为 GG哈密尔顿圈,简称 HH

        # 半哈密顿图

        不需回到出发点

        如果存在经过 GG 的每个顶点恰好一次的路,称这样的图为半哈密尔顿图,称该路为 GG哈密尔顿路,简称 HH

        # 小结论💡

        • n3n\geq3 时,完全图 KnK_n 是否为哈密尔顿图?
        • n2n\geq2 时, 方体 QnQ_n 是否为哈密尔顿图?
        • 若完全二部图 Ka,bK_{a,b} 为哈密尔顿图,aabb 需满足什么条件?a=b2\Rightarrow a=b\geq2
        • 是否存在一个具有奇数个顶点的连通图,它既是二部图,也是哈密尔顿图? 不存在,否则二部图中出现了奇圈
        • 若二部图是哈密尔顿图,则它的二部划分 (X,Y)(X,Y) 满足什么条件?X=Y\Rightarrow|X|=|Y|

        哈密顿图目前为止没有充要条件,实际上图的 HH 性判定是 NP - 困难问题,下面介绍一个必要条件和一些充分条件

        # 必要条件

        GGHH 图,则对于 VV 的每个非空真子集 SS,均有:

        ω(GS)S\omega(G-S)\leq|S|

        💡别忘了 ω(...)\omega(...) 的意思是连通分支数,这里还容易得到 HH 图不存在割点

        只是必要条件而不是充分条件,经典例子 —— 彼得森图

        用于判断非 HH 图:SV,ω(GS)>S\exist S\subset V,\ \ \omega(G-S)\gt|S| ,则 GG 不是 HH 图。

        # 充分条件

        # Dirac

        对于 n3n≥3 的简单图 GG , 如果 GG 中有:

        δ(G)n2\delta(G)\geq\frac{n}{2}

        那么 GGHH 图。

        # Ore

        对于 n3n≥3 的单图 GG , 如果 GG 中的任意两个不相邻顶点 uuvv ,有:

        d(u)+d(v)nd(u)+d(v)\geq n

        那么 GGHH 图。

        该定理证明和 Dirac 定理可以完全一致!

        该定理的条件是紧的。例如:设 GG 是由 Kk+1Κ_{k+1} 的一个顶点和另一个 Kk+1Κ_{k+1} 的一个顶点重合得到的图,那么对于 GG 的任意两个不相邻顶点 uuvv,有:

        d(u)+d(v)=2k=n1d(u)+d(v)=2k=n-1

        GG 是非 HH 图。

        # Bondy

        # 引理

        对于单图 GG ,如果 GG 中有两个不相邻顶点 uuvv ,满足:d(u)+d(v)nd(u)+d(v)≥n,那么 GGHH 图当且仅当 G+uvG+uvHH 图。

        # 闭图

        nn 阶简单图中,若对 d(u)+d(v)nd(u)+d(v)≥n 的任意一对顶点 uuvv ,均有 uadjvu\ \text{adj}\ v , 则称 GG 是闭图。

        # 引理

        G1G_1G2G_2 是同一个点集 VV 的两个闭图,则 G1G2G_1\cap G_2 是闭图,但 G1G2G_1\cup G_2 不一定是闭图。

        # 闭包

        G\overline G 是图 GG 的闭包,如果它是包含 GG 的极小闭图,即对 H∀H,若有 GHGG\subseteq H \subset\overline G,则 HH 不是闭图。

        • 如果 GG 本身是闭图,则其闭包是它本身;
        • 否则由定义可以通过在度和 n\geq n 的不相邻顶点对间加边来构造 GG 的闭图;加边之后还要继续检查是否需要加边。

        GG 的闭包是唯一的。

        # 闭包定理

        由闭包定理也可以推出 Dirac 和 Ore 定理

        GGHH 图当且仅当它的闭包是 HH

        # Chavatal

        度序列判定法

        HH 图充分条件

        设简单图 GG 的度序列是 d(d1,d2,,dn)d(d_1,d_2,\cdots,d_n),这里,d1d2dnd_1≤d_2≤⋯≤d_n,并且 n3n≥3。若对任意的 m<n2m<\frac{n}{2}

        • 或有 dm>md_m>m
        • 或有 dnmnmd_{n-m}≥n-m

        GGHH 图。(GG 的闭包是完全图!证明也是证明这个)

        # TSP

        # 边交换

        # 最优 H 圈下界

        可以通过如下方法求出最优 HH 圈的一个下界:(不是下确界,不一定达得到!)

        (1) 在 GG 中删掉一点 vv (任意的) 得图 G1G_1

        (2) 在图 G1G_1 中求出一棵最小生成树 TT

        (3) 在 vv 的关联边中选出两条权值最小者 e1e_1e2e_2

        HHGG 的最优圈,则有权值下界:

        W(H)W(T)+W(e1)+W(e2)W(H)\geq W(T)+W(e_1)+W(e_2)

        \ No newline at end of file diff --git a/math/discrete-math/graph/graph-basics/index.html b/math/discrete-math/graph/graph-basics/index.html index 33726cf7..e3827e9a 100644 --- a/math/discrete-math/graph/graph-basics/index.html +++ b/math/discrete-math/graph/graph-basics/index.html @@ -1 +1 @@ -图的基本概念 - 图论 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 图的基础知识

        # 图的定义

        一 个 图 GG 定 义 为 一 个 有 序 对 (V , E), 记 为 G = ( V , E ), 其 中:

        1. V 是一个非空集合,称为顶点集或边集,其元素称为项点或点
        2. E 是由 V 中的点组成的无序点对构成的集合,称为边集,其元素称为边,且同一 点对在 E 中可出现多次。

        GG顶点集也记为 V (G)边集也记为 E (G)

        顶点集和边集都有限的图称为有限图

        只有一个项点而无边的图称为平凡图。其他所有的图都称为非平凡图

        做题往往需要先排除平凡图

        边集为空的图称为空图

        GG顶点数 (或阶数) 和边数可分别用符号 n(G)m(G) 表示。

        连接两个相同顶点 的边的条数,叫做边的重数。重数大于 1 的边称为重边

        端点重合为一点的边称为

        既没有环也没有重边的图称为简单图。其他所有的图都称为复合图

        边记为 uv,也可记 uv 为 e ,即 e = uv。此时称 u 和 v 是 e 的端点,并称 u 和 v 相邻,u (或 v) 与 e 相关联

        若两条边有一个共同的端点,则称这两条边相邻

        若用小圆点代表点, 连线代表边,则可将一个图用 “图形” 来表示。

        # 图的同构

        设 有 两 个 图 G1 = (V1 , E1 ) 和 G2 = (V2 , E2 ), 若 在 其 项 点 集 合 之 间 存 在 双 射 , 即 存
        在一一对应的关系,使得边之间有如下的关系:

        u1u2,v1v2,u1,v1V1,u2,v2V2u1v1E1u2v2E2设 u_1 \lrarr u_2 , v_1 \lrarr v_2 ,\ \ \ u_1 ,v_1\in V_1 ,\ \ \ u_2 , v_2 \in V_2 \\ u_1 v_1 \in E_1 \lrArr u_2 v_2 \in E_2

        u1v1重数=u2v2重数u_1 v_1 重数 = u_2 v_2 重数

        则称两图 G1 , G2 同构 , 记为:

        G1G2G_1 \cong G_2

        同构关系就是一种等价关系,形成等价类

        # 完全图

        每两个不同的顶点之问都有一条边相连的简单图称为完全图

        完全图是简单图的上限,空图则是下限

        在同构意义下,n 个顶点的完全图只有一个,记为 Kn ,也叫 n 阶完全图。

        所谓具有二分类 (X, Y) 的偶图 (或二部图) 是指:一个图,它的点集可以分解为两个 (非空) 子集 X 和 Y ,使得每条边的一个端点在 X 中,另一个端点在 Y 中。

        完全偶图是指具有二分类 (X , Y) 的简单偶图 , 其中 X 的每个项点与 Y 的每个项点相连,若 |X|=m ,|Y|=n ,则这样的偶图记为 Km,n

        # 补图

        对于一个简单图 G= (V, E),令集合

        E1={uvuv,u,vV}E_1=\{uv|u\neq v ,\ \ u,v\in V\}

        则图 H=(VE/E1)H=(VE/E_1) 称为 GG补图 ,记为:

        H=G=(V,E/E1)H=\overline G=(V,E/E_1)

        如果一个图和自己的补图是同构的,则称这个图是自补的,并记为:

        GGG\cong \overline G

        # 自补图性质

        若 n 阶 图 GG 是 自补 的 ( 即 GGG\cong \overline G ), 则 n=0,1(mod4)n=0,1(\mathrm{mod} 4)

        # 正则图

        G 的顶点的度 d (v) 是指 GG 中与 v 关联的边的数目,每个环计算两次。

        δ(G)\delta(G)Δ(G)\Delta(G) 分别表示 G 的项点的最小度最大度

        为方便,奇数度的顶点称为奇点,偶数度的顶点称偶点

        设 G = (V,E) 为简单图,如果对所有 vVv\in V ,有 d (v) = k ,称图 GG k - 正则图

        完全图 Kn 和完全偶图 Kn,m 均是正则图。

        # 握手定理⭐️

        图 G = (V , E) 中 所 有 项 点 的 度 的 和 等 于 边 数 m 的 2 倍 , 即

        vVd(v)=2m\sum_{v\in V}d(v)=2m

        也叫:握手定理,图论第一定理

        # 推论 1

        在任何图中,奇点个数为偶数。

        # 推论 2

        正则图的阶数和度数不同时为奇数。

        # 推论 3⭐️

        由握手定理有:

        nδ<vV(G)d(v)=2mnΔδ2mnΔn\delta\lt\sum_{v\in V(G)}d(v)=2m\leq n\Delta\\ \delta\leq\frac{2m}{n}\leq\Delta

        # 可图数组

        一个图 GG 的各个点的度 d1,d2,,dnd_1,d_2,\cdots,d_n 构 成 的 非 负 整 数 组 (d1,d2,,dn)(d_1,d_2,\cdots,d_n) 称为 GG度序列

        度序列通常以度数非递增的顺序列出?

        若对一个非负整数组 (d1,d2,,dn)(d_1,d_2,\cdots,d_n) 满足 mZ,i=1ndi=2m\exist m \in \mathbb Z,\sum_{i=1}^{n}d_i=2m (数组和为偶数),存在一个简单图 GG 以该数组为度序列,则称这个数组是可图的,也叫这个数组为图数组 / 图序列

        关于图序列我们研究三个问题:

        • 存在问题 —— 什么样的数组是图序列
        • 计数问题 —— 一个图序列对应多少个不同构的图
        • 唯一性问题 —— 怎样的图序列对应唯一一个同构的图

        # 必要条件

        非负整数组 (d1,d2,,dn)(d_1,d_2,\cdots,d_n) 是图序列的一个必要条件是:

        i=1ndi是偶数,或者:2i=1ndi\sum_{i=1}^{n}d_i \text{是偶数,或者:} 2|\sum_{i=1}^{n}d_i

        另一个必要条件是:(可简单图化)

        din1d_i\leq n-1

        # 图序列相关定理

        # Havel-Hakimi ⭐️

        设有非负整数组 Π=(d1,d2,,dn)\Pi =(d_1 , d_2 , \cdots , d_n) ,且 i=1ndi\sum_{i=1}^{n}d_i 是一个偶数,n1d1d2dnn-1\geq d_1\geq d_2 \geq \cdots\geq d_nΠ\Pi 是可图(可简单图化)的充要条件为:

        Π=(d21,d31,,dd1+11,dd1+2,,dn)\Pi'=(d_2-1,d_3-1,\cdots,d_{d_1+1}-1,d_{d_1+2},\cdots,d_n)

        是可图的(是图序列)。


        对任意一个序列,判断是否可图的步骤如下:(当然先确保满足可图的必要条件)

        1. 按递减排序
        2. 去掉第一个元素 d1d_1 ,后面 d1d_1 个元素依次减 1️⃣(1-1
        3. 重复前面两个步骤 ,直到:
          • 只剩一个点,则该序列是可图的,即是图序列
          • 减 1️⃣后出现负数,则该序列是不可图的,即不是图序列

        # 中证明充分性:

        # 情形 1

        v1v_1 与点 v2,v3,,vd+1v_2,v_3,\cdots,v_{d+1} 邻接

        Gv1G-v_1 的度序列正好 Π1\Pi_1

        # 情形 2

        v1v_1 与点 v2,v3,,vd+1v_2,v_3,\cdots,v_{d+1} 的某些顶点邻接

        作如下假设:

        • v1v_1vj0v_{j_0} 邻接,但当 k>j0k\gt j_0 时,v1v_1vkv_k 不邻接;
        • v1v_1vi0v_{i_0} 不邻接,但当 k<i0k\lt i_0 时,v1v_1vkv_k 邻接;

        进行这样的操作:

        • 去掉边 v1vj0v_1 v_{j_0}vi0vmv_{i_0}v_m
        • 加上边 vj0vmv_{j_0}v_m 和 v_1v_

        显然新图与原来的度序列相同,但是,但 j0j_0 减小了, i0i_0 增大了!

        如此循环下去,j0=i0j_0=i_0 时 情形 2 就可以变成情形 1

        # 厄多斯定理

        非负整数组:

        π=(d1,d2,,dn),d1d2dn,i=1ndi=2m\pi = (d_1,d_2,\cdots,d_n),d_1\geq d_2 \geq d_n,\sum_{i=1}^{n}d_i=2m

        是图序列的充分必要条件是:

        i=1rdir(r1)+i=r+1nmin{r,di},1rn1\sum_{i=1}^{r}d_i\leq r(r-1)+\sum_{i=r+1}^{n}\text{min}\{r,d_i\},\ \ \ 1\leq r\leq n-1

        证明非常困难捏

        # 定理?

        一个满足 d2=dn1d_2=d_{n-1}图序列 π=(d1,d2,,dn)\pi=(d_1,d_2,\cdots,d_n)唯一图序列的充分必要条件时下列条件之一满足:

        • d1=dn,dn{1,n1,n2}d_1=d_n,\ \ d_n\in \{ 1,n-1,n-2 \}
        • d1=dn=2,n=5d_1=d_n=2,\ \ n=5
        • d1>d2=dn=1d_1\gt d_2=d_n=1
        • d1>d2=dn=2,d1{n1,n2}d_1\gt d_2=d_n=2,\ \ d_1\in\{n-1,n-2\}
        • n2=d1=dn1>dnn-2=d_1=d_{n-1}\gt d_n
        • n3=d1=dn1>dn=1n-3=d_1=d_{n-1}\gt d_n=1
        • n1=d1>d2=dn=3,n=6n-1=d_1\gt d_2=d_{n}=3,\ \ n=6

        # 定理 - 简单图度数不可能互不相同

        一个简单图 GGnn 个点的度不能互不相同

        证明: 因为图 G 为简单图,所以:Δ(G)n1\Delta(G)\leq n - 1


        情形 1:若 GG 没有孤立点,则 1d(v)n1,vV(G)1 ≤ d(v)≤ n - 1,\ \ \ ∀v \in V(G) , 由鸽笼原理:必有两顶点度数相同;


        情形 2:若 GG 只有一个孤立点,设 G1G_1 表示 G 去掉孤立点后的部分,则:

        1d(v)n2,vV(G1)1 ≤ d(v)≤ n - 2,\ \ \ ∀v \in V(G_1)

        由鸽笼原理:在 G1G_1 里必有两顶点度数相同;


        情形 3:若 GG 只有两个以上的孤立点,则定理显然成立。


        # 频序列

        nn 阶图 GG 的各点的度取了 ss 个不同的非负整数 d1,d2,,dsd_1,d_2,\cdots,d_s 。 又 设 度 为 did_i 的 点 有 bib_i 个 (i=1,2,,s)i=1,2,\cdots,s) , 则有 i=1sbi=n\sum_{i=1}^{s}b_i =n 。故非负整数组 (d1,d2,,dsd_1,d_2,\cdots,d_s) 是 nn 的一个划分,称为 GG频序列

        # 性质

        一个 nn 阶图 GG 和它的补图有相同的频序列

        # 拓扑不变量

        GG拓扑不变量是指与图 GG 有关的一个数或数组 (向量)。它对于与图 GG 同构的所有图来说,不会发生改变。

        一个图 GG 可以对应很多拓扑不变量。如果某组不变量可完全决定一个图,称它为不变量的完全集

        # 图的运算

        # 子图和真子图

        如果 V(H)V(G),E(H)E(G)V(H) \subseteq V(G),E(H)\subseteq E(G) ,且 HH 中边的重数不超过 GG 中对应边的重数 , 则称 HHGG子图,记为 HGH \subseteq G 。有时又称 GGHH母图

        HGH\subseteq G ,但 HGH\neq G 时,则记为 HGH\subset G ,且称 HHGG真子图

        # 生成子图

        GG生成子图是指满足 V(H)=V(G)V (H) = V (G ) 的子图 HH。也就是包含所有顶点但不包含所有边的子图。

        # 计数性质

        简单图 G=(n,m)G=(n,m) 的所有不同的生成子图个数是 2m2^m (包括 GG 本身和空图)。

        # 导出子图

        假设 VV'VV 的一个非空子集。以 VV' 为顶点集,以两端点均在 VV' 中的边的全体为边集所组成的子图,称为 GG 的由 VV' 导出的子图,记为 G[V]G[V'],简称为 GG(点) 导出子图

        假设 EE'EE 的非空子集。以 EE' 为边集,以 EE' 中边的端点全体为顶点集所组成的子图称为 GG 的由 EE' 导出的子图,记为 G[E]G[E'],简称为 GG边导出子图

        # 删点运算

        (点) 导出子图 G[V\V]G[V\backslash V'] 记为 GVG-V'。它是 GG 中删除 VV' 中的项点以及与这些项点相关联的边所得到的子图。若 V={v}V'=\{v\} ,则把 G{v}G-\{v\} 简记为 GvG-v

        # 删边运算

        边集为 E\EE\backslash E'GG 的边导出子图简记为 GEG-E' 。若 E={e}E'=\{e\} ,则把 GeG-{e} 简记为 GeG-e

        ⚠️删边运算会删掉那些度数变为 0 的顶点⚠️


        G1G_1G2G_2GG 的子图。若 G1G_1G2G_2 无公共顶点,则称它们是不相交

        G1G_1G2G_2 无公共边,则称它们是边不重

        # 并运算

        G1G_1G2G_2并图 G1UG2G_1UG_2 是指 GG 的一个子图:

        • 其顶点集为 V(G1)V(G2)V(G_1)\cup V(G_2)
        • 其边集为 E(G1)E(G2)E(G_1)\cup E(G_2)

        如果 G1G_1G2G_2 是不相交的,有时就记其并图为 G1+G2G_1+G_2

        # 交运算

        G1G_1G2G_2交图 G1G2G_1\cap G_2 ,是指 GG 的一个子图:

        • 其顶点集为 V(G1)V(G2)V(G_1)\cap V(G_2)
        • 其边集为 E(G1)E(G2)E(G_1)\cap E(G_2)

        此时 G1G_1G2G_2 至少要有一个公共顶点!

        # 差运算

        G1G_1G2G_2 G1G2G_1-G_2 是由 G1G_1 中去掉 G2G_2 中的边组成的图 。

        没有去掉那些边的端点!即使剩下 0 度数端点仍要保留!

        # 对称差 / 环和差

        G1G_1G2G_2对称差 G1ΔG2G_1 \Delta G_2G1G2G_1 \cup G_2 去掉 G1G2G_1 \cap G_2 所得到的图,即 :

        G1ΔG2=(G1G2)(G1G2)=(G1G2)(G2G1)G_1 \Delta G_2 = (G_1 \cup G_2)-(G_1 \cap G_2) = (G_1 - G_2) \cup (G_2-G_1)

        # 联运算

        在不相交的 G1G_1G2G_2 的并图 G1G2G_1 \cup G_2 中,把 G1G_1 的每个项点和 G2G_2 的每个项点连接起来所得到的 图称为 G1G_1G2G_2联图,记为 $G_1 \lor G_2 $。


        uiadjviu_i\ \mathrm{adj}\ v_i 表示 uiu_iviv_i 邻接

        # 积运算

        G1=(V1,E1),G2=(V2,E2)G_1 =(V_1 , E_1),G_2 =(V_2 ,E_2)对点集 V=V1×V2V=V_1 \times V_2 (笛卡尔集)中任意两个点 u=(u1,u2)u=(u_1,u_2)v=(v1,v2)v=(v_1,v_2) ,当以下式子成立时:

        (u1=v1u2adjv2)(u2=v2u1adjv1)(u_1 =v_1 \land u_2\ \mathrm{adj}\ v_2)\lor(u_2 =v_2 \land u_1\ \mathrm{adj}\ v_1)

        就把 uuvv 连接起来所得到的图 GG 称为 G1G_1G2G_2积图,记为 G=G1×G2G=G_1 \times G_2

        # 合成运算

        Gi=(V1,E1)G_i=(V_1,E_1)G2=(V2,E2)G_2=(V_2,E_2) , 对点集 V=V1×V2V=V_1 \times V_2 (笛卡尔集)中任意两个点 u=(u1,u2)u=(u_1,u_2)v=(v1,v2)v=(v_1,v_2) ,当以下式子成立时:

        (u_1\ \text{adj}\ v_1)\or(u_1=v_1\land u_2\ \text{adj}\ v_2)

        就把 u 和 v 连接起来所得到的图 G 称为 G1G_1G2G_2合成图,记为 G=G1[G2]G=G_1[G_2]

        G1[G2]G_1[G_2]G2[G1]G_2[G_1] 可能是同构的,也可能是不同构的


        积运算和合成运算中,得到的点集 VV 的定义都是相同的,都是笛卡尔集 V=V1×V2V=V_1 \times V_2 ;但是边集的定义有所不同


        运算点的数目边的数目
        G1G2G_1\cup G_2n1+n2n_1 + n_2m1+m2m_1 + m_2
        G1G2G_1\lor G_2n1+n2n_1 + n_2m1+m2+n1n2m_1 + m_2 +n_1 n_2
        G1×G2G_1\times G_2n1n2n_1 n_2n1m2+n2m1n_1m_2 + n_2m_1
        合成 G1[G2]G_1[G_2]n1n2n_1 n_2n1m2+n22m1n_1m_2 + {n_2}^2m_1

        # 方体

        一族特别重要的图称为方体,它用积来定义最为自然。

        nn 方体 QnQ_n 递推地定义为:

        • Q1=K2Q_1=K_2
        • Qn=K2×Qn1Q_n=K_2\times Q_{n-1}

        QnQ_n2n2^n 个点,它的点可以用二进制串 a1a2ana_1 a_2 \cdots a_n 来标定,其中 ai{0,1}a_i\in\{0,1\}

        QnQ_n 两个点的二进制表示式中只有一位不同,则它们邻接

        2方体

        3方体

        # 路与图的连通性

        # 途径、迹、路

        一个有限非空序列:

        w=v0e1v1e2v2ekvkw=v_0\ e_1\ v_1\ e_2\ v_2\cdots e_k\ v_k

        它的项交替地为顶点和边,使得对 1ik1\leq i\leq keie_i ,的端点是 vi1v_{i-1}viv_i、 称 ww 是从 v0v_0vkv_k 的一条途径,或一条 (v0,vk)(v_0,v_k) 途径

        途径中边数称为途径的长度

        v0v_0vkv_k 分别称为途径的起点终点,其余顶点称为途径的内部点

        一条途径的起点与终点重合时叫做闭途径

        边不重复的途径称为图的一条,起点与终点重合时叫做闭迹,也称为回路

        顶点不重复的途径称为图的一条,起点与终点重合时叫做。显然顶点不重复时边也不重复,所以迹是包含路的。

        起点与终点重合的途径、迹、路分别称为图的闭途径、闭迹与圈。闭迹也称为回路

        长度为 kk 的圈称为 kk 圈,kk 为奇数时称为奇圈kk 为偶数时称为偶圈

        # 圈的充分条件

        δ2\delta ≥2,则 GG 中必然含有圈。

        只就连通图证明即可!

        W=v1v2vk1vkW=v_1 v_2\cdots v_{k-1}v_kGG 中的一条最长路。由于 δ2\delta≥2,所以 vkv_k 必然有相异于 vk1v_{k-1} 的邻接顶点。

        WW 是连通图 GG 中最长路,所以,这样的邻接点必然是 v1v2vk2v_1 v_2\cdots v_{k-2} 中之一。设该点为 vmv_m ,则 vmvm+1vkvmv_m v_{m+1}\cdots v_k v_mGG 中圈。

        # 偶图的充要条件

        一个图是二部图 (偶图) 当且当它不包含奇圈


        证明必要性:

        GG 是具有二分类 (X,Y)(X,Y) 的偶图,并且 C=v0v1vkv0C=v_0 v_1\cdots v_k v_0GG 的一个圈

        不失一般性,可假定 v0Xv_0\in X

        一般说来,v2iX,v2i+1Yv_{2i}\in X,v_{2i+1}\in Y

        又因为v0Xv_0\in X ,所以 vkYv_k\in Y,由此即得 CC 是偶圈


        证明充分性:

        GG 中任意选取点 uu,定义 VV 的分类如下:

        • X={xd(u,x)是偶数,xV(G)}X=\{x|d(u,x)\text{是偶数},x\in V(G)\}
        • Y={yd(u,y)是奇数,yV(G)}Y=\{y|d(u,y)\text{是奇数},y\in V(G)\}

        (接下来证明 XX 中任意两点 v,wv,w 不邻接)

        v,wv,wXX 中任意两个顶点,PP 是一条最短 (u,v)(u,v) 路,而 QQ 是一条最短 (u,w)(u,w) 路,设 u1u_1PPQQ 的最后一个交点:

        偶图充分必要条件充分性证明

        由于 P,QP,Q 是最短路,所以,P,QP,Quuu1u_1 段长度相同,因此奇偶性相同

        P,QP,Q 的长均是偶数,所以,P,QP,Qu1vu_1 v 段和 u1wu_1 w 段奇偶性相同

        如果 v,wv,w 邻接则可得到奇圈,矛盾!

        # 两顶点的距离

        图中顶点 uuvv 的距离: uuvv 间最短路的长度称为 uuvv 间距离。记为:

        d(u,v)d(u,v)

        如果 uuvv 间不存在路,定义:

        d(u,v)=0d(u,v)=0

        # 两顶点的连通性

        GG 中点 uuvv 说是连通的,如果 uuvv存在通路。否则称 uuvv 不连通

        点的连通关系是等价关系

        非连通图中每一个极大连通部分,称为 GG连通分支

        GG 的连通分支的个数,称为 GG分支数,记为:

        ω(G)\omega(G)

        # 图的连通性

        如果图 GG 中任意两点是连通的,称 GG连通图,否则,称 GG非连通图

        # 连通图的性质

        nn 阶连通图中:

        1. 至少有 n1n-1 条边
        2. 如果边数大于 n1n-1,则至少有一条闭迹
        3. 如果恰有 n1n-1 条边,则至少有一个奇度点

        (2)证明:

        GG 中没有 11 度顶点,由握手定理:

        2m=vV(G)d(v)2nmnm>n12m=\sum_{v\in V(G)}d(v)\geq 2n \rArr m\geq n \rArr m \gt n-1


        (3)证明:

        若不然,GG 中顶点度数至少为 22,于是由握手定理:

        2m=v(G)d(v)2nmnm>n12m=\sum_{v\in(G)}d(v) \geq 2n \rArr m ≥n \rArr m > n-1

        这与 GG 中恰有 n1n-1 条边矛盾!

        # 补图的连通性

        若图 GG 是不连通的,则它的补图 G\overline G 是连通图.


        证明:设 u,vu,vGG 的任意两个顶点:

        • uuvvGG 中不邻接,则在 G\overline G 中它们邻接,从而 G\overline G 是连通的
        • uuvvGG 中邻接,它们属于 GG 的同一分支;在另一个分支中有一点 ww,在 G\overline Guuvv 均与 w 邻接,即 uwvuwv 是一条通道,从而 G\overline G 是连通的

        # 连通图充分条件 1

        设图 GGnn 阶图,若 GG 中任意两个不相邻的点 uuvv,有:

        d(u)+d(v)n1d(u)+d(v)\geq n-1

        GG 是连通的,且 d(G)2d(G)\leq 2


        证明:对 GG 中任意两点 xxyy,一定存在一条长度至多为 22 的连接 xxyy 的路

        xyE(G)xy\in E(G) 时上述论断显然成立;

        xyE(G)xy\notin E(G) 时:

        可以证明,存在点 ww,它与 xxyy 同时邻接。反证法:

        若不然,在 GG 的剩下 n2n-2 个顶点中,假设:

        • kk 个与 xx 邻接,但与 yy 不邻接;
        • ll 个与 yy 邻接,但与 xx 不邻接;
        • mm 个顶点和 x,yx,y 均不邻接。

        则:d(x)=kd(x)=kd(y)=ld(y)=l

        由于 k+l+m=n2k+l+m=n-2

        所以 d(x)+d(y)=nm2n2d(x)+d(y)=n-m-2\leq n-2 ,矛盾!


        # 连通图充分条件 2

        GGnn 阶图,若 δn12\delta≥\frac{n-1}{2} ,则 GG 连通


        证明:

        对 G 中任意两个不相邻顶点 u 与 v,有:

        d(u)+d(v)n12+n12=n1d(u)+d(v)\geq\frac{n-1}{2}+\frac{n-1}{2}=n-1

        所以,G 是连通的。

        该定理的界是紧的 (Sharpness)。即不能再修改!

        例如:设 G 由两个分支作成的图,两个分支均为 KmK_m ,则 G 中不相邻顶点度数之和恰为 n-2 . (n=2m)

        # 图的直径

        连通图 G 的直径定义为

        d(G)=max{d(u,v)u,vV(G)}d(G) = \text{max}\{d(u,v)|u,v \in V(G)\}

        如果 G 不连通,图 G 的直径定义为 d(G)=d(G) = \infin

        # 直径相关性质

        设 G 是具有 m 条边的 n 阶单图,若 G 的直径为 2 且 δ=n2\delta=n-2 ,则 m2n4m≥2n-4


        证明:

        d(v)=Δ=n2d(v)=\Delta=n-2 ,设 v 的邻接点为 v1,v2,,vn2v_1,v_2,\cdots,v_{n-2} ,u 是剩下的一个项点。

        由于 d(G)=2d(G)=2 且 u 不能和 v 邻接,所以 u 至少和 v1,v2,,vn2v_1,v_2,\cdots,v_{n-2} 中的一个顶点邻接。否则有 d(G)>2d(G)>2 ; 不妨假设 u 和 v1,v2,,vkv_1,v_2,\cdots,v_{k} 邻接。

        为了保证 u 到各点距离不超过 2 , vk+1,,vn2v_{k+1},\cdots,v_{n-2} 这些頂点的每一个必须和前面 v1,v2,,vkv_1,v_2,\cdots,v_{k} 中某点邻接,这样,图中至少又有 n-2 条边。总共至少有 2n-4 条边。

        \ No newline at end of file +图的基本概念 - 图论 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 图的基础知识

        # 图的定义

        一 个 图 GG 定 义 为 一 个 有 序 对 (V , E), 记 为 G = ( V , E ), 其 中:

        1. V 是一个非空集合,称为顶点集或边集,其元素称为项点或点
        2. E 是由 V 中的点组成的无序点对构成的集合,称为边集,其元素称为边,且同一 点对在 E 中可出现多次。

        GG顶点集也记为 V (G)边集也记为 E (G)

        顶点集和边集都有限的图称为有限图

        只有一个项点而无边的图称为平凡图。其他所有的图都称为非平凡图

        做题往往需要先排除平凡图

        边集为空的图称为空图

        GG顶点数 (或阶数) 和边数可分别用符号 n(G)m(G) 表示。

        连接两个相同顶点 的边的条数,叫做边的重数。重数大于 1 的边称为重边

        端点重合为一点的边称为

        既没有环也没有重边的图称为简单图。其他所有的图都称为复合图

        边记为 uv,也可记 uv 为 e ,即 e = uv。此时称 u 和 v 是 e 的端点,并称 u 和 v 相邻,u (或 v) 与 e 相关联

        若两条边有一个共同的端点,则称这两条边相邻

        若用小圆点代表点, 连线代表边,则可将一个图用 “图形” 来表示。

        # 图的同构

        设 有 两 个 图 G1 = (V1 , E1 ) 和 G2 = (V2 , E2 ), 若 在 其 项 点 集 合 之 间 存 在 双 射 , 即 存
        在一一对应的关系,使得边之间有如下的关系:

        u1u2,v1v2,u1,v1V1,u2,v2V2u1v1E1u2v2E2设 u_1 \lrarr u_2 , v_1 \lrarr v_2 ,\ \ \ u_1 ,v_1\in V_1 ,\ \ \ u_2 , v_2 \in V_2 \\ u_1 v_1 \in E_1 \lrArr u_2 v_2 \in E_2

        u1v1重数=u2v2重数u_1 v_1 重数 = u_2 v_2 重数

        则称两图 G1 , G2 同构 , 记为:

        G1G2G_1 \cong G_2

        同构关系就是一种等价关系,形成等价类

        # 完全图

        每两个不同的顶点之问都有一条边相连的简单图称为完全图

        完全图是简单图的上限,空图则是下限

        在同构意义下,n 个顶点的完全图只有一个,记为 Kn ,也叫 n 阶完全图。

        所谓具有二分类 (X, Y) 的偶图 (或二部图) 是指:一个图,它的点集可以分解为两个 (非空) 子集 X 和 Y ,使得每条边的一个端点在 X 中,另一个端点在 Y 中。

        完全偶图是指具有二分类 (X , Y) 的简单偶图 , 其中 X 的每个项点与 Y 的每个项点相连,若 |X|=m ,|Y|=n ,则这样的偶图记为 Km,n

        # 补图

        对于一个简单图 G= (V, E),令集合

        E1={uvuv,u,vV}E_1=\{uv|u\neq v ,\ \ u,v\in V\}

        则图 H=(VE/E1)H=(VE/E_1) 称为 GG补图 ,记为:

        H=G=(V,E/E1)H=\overline G=(V,E/E_1)

        如果一个图和自己的补图是同构的,则称这个图是自补的,并记为:

        GGG\cong \overline G

        # 自补图性质

        若 n 阶 图 GG 是 自补 的 ( 即 GGG\cong \overline G ), 则 n=0,1(mod4)n=0,1(\mathrm{mod} 4)

        # 正则图

        G 的顶点的度 d (v) 是指 GG 中与 v 关联的边的数目,每个环计算两次。

        δ(G)\delta(G)Δ(G)\Delta(G) 分别表示 G 的项点的最小度最大度

        为方便,奇数度的顶点称为奇点,偶数度的顶点称偶点

        设 G = (V,E) 为简单图,如果对所有 vVv\in V ,有 d (v) = k ,称图 GG k - 正则图

        完全图 Kn 和完全偶图 Kn,m 均是正则图。

        # 握手定理⭐️

        图 G = (V , E) 中 所 有 项 点 的 度 的 和 等 于 边 数 m 的 2 倍 , 即

        vVd(v)=2m\sum_{v\in V}d(v)=2m

        也叫:握手定理,图论第一定理

        # 推论 1

        在任何图中,奇点个数为偶数。

        # 推论 2

        正则图的阶数和度数不同时为奇数。

        # 推论 3⭐️

        由握手定理有:

        nδ<vV(G)d(v)=2mnΔδ2mnΔn\delta\lt\sum_{v\in V(G)}d(v)=2m\leq n\Delta\\ \delta\leq\frac{2m}{n}\leq\Delta

        # 可图数组

        一个图 GG 的各个点的度 d1,d2,,dnd_1,d_2,\cdots,d_n 构 成 的 非 负 整 数 组 (d1,d2,,dn)(d_1,d_2,\cdots,d_n) 称为 GG度序列

        度序列通常以度数非递增的顺序列出?

        若对一个非负整数组 (d1,d2,,dn)(d_1,d_2,\cdots,d_n) 满足 mZ,i=1ndi=2m\exist m \in \mathbb Z,\sum_{i=1}^{n}d_i=2m (数组和为偶数),存在一个简单图 GG 以该数组为度序列,则称这个数组是可图的,也叫这个数组为图数组 / 图序列

        关于图序列我们研究三个问题:

        • 存在问题 —— 什么样的数组是图序列
        • 计数问题 —— 一个图序列对应多少个不同构的图
        • 唯一性问题 —— 怎样的图序列对应唯一一个同构的图

        # 必要条件

        非负整数组 (d1,d2,,dn)(d_1,d_2,\cdots,d_n) 是图序列的一个必要条件是:

        i=1ndi是偶数,或者:2i=1ndi\sum_{i=1}^{n}d_i \text{是偶数,或者:} 2|\sum_{i=1}^{n}d_i

        另一个必要条件是:(可简单图化)

        din1d_i\leq n-1

        # 图序列相关定理

        # Havel-Hakimi ⭐️

        设有非负整数组 Π=(d1,d2,,dn)\Pi =(d_1 , d_2 , \cdots , d_n) ,且 i=1ndi\sum_{i=1}^{n}d_i 是一个偶数,n1d1d2dnn-1\geq d_1\geq d_2 \geq \cdots\geq d_nΠ\Pi 是可图(可简单图化)的充要条件为:

        Π=(d21,d31,,dd1+11,dd1+2,,dn)\Pi'=(d_2-1,d_3-1,\cdots,d_{d_1+1}-1,d_{d_1+2},\cdots,d_n)

        是可图的(是图序列)。


        对任意一个序列,判断是否可图的步骤如下:(当然先确保满足可图的必要条件)

        1. 按递减排序
        2. 去掉第一个元素 d1d_1 ,后面 d1d_1 个元素依次减 1️⃣(1-1
        3. 重复前面两个步骤 ,直到:
          • 只剩一个点,则该序列是可图的,即是图序列
          • 减 1️⃣后出现负数,则该序列是不可图的,即不是图序列

        # 中证明充分性:

        # 情形 1

        v1v_1 与点 v2,v3,,vd+1v_2,v_3,\cdots,v_{d+1} 邻接

        Gv1G-v_1 的度序列正好 Π1\Pi_1

        # 情形 2

        v1v_1 与点 v2,v3,,vd+1v_2,v_3,\cdots,v_{d+1} 的某些顶点邻接

        作如下假设:

        • v1v_1vj0v_{j_0} 邻接,但当 k>j0k\gt j_0 时,v1v_1vkv_k 不邻接;
        • v1v_1vi0v_{i_0} 不邻接,但当 k<i0k\lt i_0 时,v1v_1vkv_k 邻接;

        进行这样的操作:

        • 去掉边 v1vj0v_1 v_{j_0}vi0vmv_{i_0}v_m
        • 加上边 vj0vmv_{j_0}v_m 和 v_1v_

        显然新图与原来的度序列相同,但是,但 j0j_0 减小了, i0i_0 增大了!

        如此循环下去,j0=i0j_0=i_0 时 情形 2 就可以变成情形 1

        # 厄多斯定理

        非负整数组:

        π=(d1,d2,,dn),d1d2dn,i=1ndi=2m\pi = (d_1,d_2,\cdots,d_n),d_1\geq d_2 \geq d_n,\sum_{i=1}^{n}d_i=2m

        是图序列的充分必要条件是:

        i=1rdir(r1)+i=r+1nmin{r,di},1rn1\sum_{i=1}^{r}d_i\leq r(r-1)+\sum_{i=r+1}^{n}\text{min}\{r,d_i\},\ \ \ 1\leq r\leq n-1

        证明非常困难捏

        # 定理?

        一个满足 d2=dn1d_2=d_{n-1}图序列 π=(d1,d2,,dn)\pi=(d_1,d_2,\cdots,d_n)唯一图序列的充分必要条件时下列条件之一满足:

        • d1=dn,dn{1,n1,n2}d_1=d_n,\ \ d_n\in \{ 1,n-1,n-2 \}
        • d1=dn=2,n=5d_1=d_n=2,\ \ n=5
        • d1>d2=dn=1d_1\gt d_2=d_n=1
        • d1>d2=dn=2,d1{n1,n2}d_1\gt d_2=d_n=2,\ \ d_1\in\{n-1,n-2\}
        • n2=d1=dn1>dnn-2=d_1=d_{n-1}\gt d_n
        • n3=d1=dn1>dn=1n-3=d_1=d_{n-1}\gt d_n=1
        • n1=d1>d2=dn=3,n=6n-1=d_1\gt d_2=d_{n}=3,\ \ n=6

        # 定理 - 简单图度数不可能互不相同

        一个简单图 GGnn 个点的度不能互不相同

        证明: 因为图 G 为简单图,所以:Δ(G)n1\Delta(G)\leq n - 1


        情形 1:若 GG 没有孤立点,则 1d(v)n1,vV(G)1 ≤ d(v)≤ n - 1,\ \ \ ∀v \in V(G) , 由鸽笼原理:必有两顶点度数相同;


        情形 2:若 GG 只有一个孤立点,设 G1G_1 表示 G 去掉孤立点后的部分,则:

        1d(v)n2,vV(G1)1 ≤ d(v)≤ n - 2,\ \ \ ∀v \in V(G_1)

        由鸽笼原理:在 G1G_1 里必有两顶点度数相同;


        情形 3:若 GG 只有两个以上的孤立点,则定理显然成立。


        # 频序列

        nn 阶图 GG 的各点的度取了 ss 个不同的非负整数 d1,d2,,dsd_1,d_2,\cdots,d_s 。 又 设 度 为 did_i 的 点 有 bib_i 个 (i=1,2,,s)i=1,2,\cdots,s) , 则有 i=1sbi=n\sum_{i=1}^{s}b_i =n 。故非负整数组 (d1,d2,,dsd_1,d_2,\cdots,d_s) 是 nn 的一个划分,称为 GG频序列

        # 性质

        一个 nn 阶图 GG 和它的补图有相同的频序列

        # 拓扑不变量

        GG拓扑不变量是指与图 GG 有关的一个数或数组 (向量)。它对于与图 GG 同构的所有图来说,不会发生改变。

        一个图 GG 可以对应很多拓扑不变量。如果某组不变量可完全决定一个图,称它为不变量的完全集

        # 图的运算

        # 子图和真子图

        如果 V(H)V(G),E(H)E(G)V(H) \subseteq V(G),E(H)\subseteq E(G) ,且 HH 中边的重数不超过 GG 中对应边的重数 , 则称 HHGG子图,记为 HGH \subseteq G 。有时又称 GGHH母图

        HGH\subseteq G ,但 HGH\neq G 时,则记为 HGH\subset G ,且称 HHGG真子图

        # 生成子图

        GG生成子图是指满足 V(H)=V(G)V (H) = V (G ) 的子图 HH。也就是包含所有顶点但不包含所有边的子图。

        # 计数性质

        简单图 G=(n,m)G=(n,m) 的所有不同的生成子图个数是 2m2^m (包括 GG 本身和空图)。

        # 导出子图

        假设 VV'VV 的一个非空子集。以 VV' 为顶点集,以两端点均在 VV' 中的边的全体为边集所组成的子图,称为 GG 的由 VV' 导出的子图,记为 G[V]G[V'],简称为 GG(点) 导出子图

        假设 EE'EE 的非空子集。以 EE' 为边集,以 EE' 中边的端点全体为顶点集所组成的子图称为 GG 的由 EE' 导出的子图,记为 G[E]G[E'],简称为 GG边导出子图

        # 删点运算

        (点) 导出子图 G[V\V]G[V\backslash V'] 记为 GVG-V'。它是 GG 中删除 VV' 中的项点以及与这些项点相关联的边所得到的子图。若 V={v}V'=\{v\} ,则把 G{v}G-\{v\} 简记为 GvG-v

        # 删边运算

        边集为 E\EE\backslash E'GG 的边导出子图简记为 GEG-E' 。若 E={e}E'=\{e\} ,则把 GeG-{e} 简记为 GeG-e

        ⚠️删边运算会删掉那些度数变为 0 的顶点⚠️


        G1G_1G2G_2GG 的子图。若 G1G_1G2G_2 无公共顶点,则称它们是不相交

        G1G_1G2G_2 无公共边,则称它们是边不重

        # 并运算

        G1G_1G2G_2并图 G1UG2G_1UG_2 是指 GG 的一个子图:

        • 其顶点集为 V(G1)V(G2)V(G_1)\cup V(G_2)
        • 其边集为 E(G1)E(G2)E(G_1)\cup E(G_2)

        如果 G1G_1G2G_2 是不相交的,有时就记其并图为 G1+G2G_1+G_2

        # 交运算

        G1G_1G2G_2交图 G1G2G_1\cap G_2 ,是指 GG 的一个子图:

        • 其顶点集为 V(G1)V(G2)V(G_1)\cap V(G_2)
        • 其边集为 E(G1)E(G2)E(G_1)\cap E(G_2)

        此时 G1G_1G2G_2 至少要有一个公共顶点!

        # 差运算

        G1G_1G2G_2 G1G2G_1-G_2 是由 G1G_1 中去掉 G2G_2 中的边组成的图 。

        没有去掉那些边的端点!即使剩下 0 度数端点仍要保留!

        # 对称差 / 环和差

        G1G_1G2G_2对称差 G1ΔG2G_1 \Delta G_2G1G2G_1 \cup G_2 去掉 G1G2G_1 \cap G_2 所得到的图,即 :

        G1ΔG2=(G1G2)(G1G2)=(G1G2)(G2G1)G_1 \Delta G_2 = (G_1 \cup G_2)-(G_1 \cap G_2) = (G_1 - G_2) \cup (G_2-G_1)

        # 联运算

        在不相交的 G1G_1G2G_2 的并图 G1G2G_1 \cup G_2 中,把 G1G_1 的每个项点和 G2G_2 的每个项点连接起来所得到的 图称为 G1G_1G2G_2联图,记为 $G_1 \lor G_2 $。


        uiadjviu_i\ \mathrm{adj}\ v_i 表示 uiu_iviv_i 邻接

        # 积运算

        G1=(V1,E1),G2=(V2,E2)G_1 =(V_1 , E_1),G_2 =(V_2 ,E_2)对点集 V=V1×V2V=V_1 \times V_2 (笛卡尔集)中任意两个点 u=(u1,u2)u=(u_1,u_2)v=(v1,v2)v=(v_1,v_2) ,当以下式子成立时:

        (u1=v1u2adjv2)(u2=v2u1adjv1)(u_1 =v_1 \land u_2\ \mathrm{adj}\ v_2)\lor(u_2 =v_2 \land u_1\ \mathrm{adj}\ v_1)

        就把 uuvv 连接起来所得到的图 GG 称为 G1G_1G2G_2积图,记为 G=G1×G2G=G_1 \times G_2

        # 合成运算

        Gi=(V1,E1)G_i=(V_1,E_1)G2=(V2,E2)G_2=(V_2,E_2) , 对点集 V=V1×V2V=V_1 \times V_2 (笛卡尔集)中任意两个点 u=(u1,u2)u=(u_1,u_2)v=(v1,v2)v=(v_1,v_2) ,当以下式子成立时:

        (u_1\ \text{adj}\ v_1)\or(u_1=v_1\land u_2\ \text{adj}\ v_2)

        就把 u 和 v 连接起来所得到的图 G 称为 G1G_1G2G_2合成图,记为 G=G1[G2]G=G_1[G_2]

        G1[G2]G_1[G_2]G2[G1]G_2[G_1] 可能是同构的,也可能是不同构的


        积运算和合成运算中,得到的点集 VV 的定义都是相同的,都是笛卡尔集 V=V1×V2V=V_1 \times V_2 ;但是边集的定义有所不同


        运算点的数目边的数目
        G1G2G_1\cup G_2n1+n2n_1 + n_2m1+m2m_1 + m_2
        G1G2G_1\lor G_2n1+n2n_1 + n_2m1+m2+n1n2m_1 + m_2 +n_1 n_2
        G1×G2G_1\times G_2n1n2n_1 n_2n1m2+n2m1n_1m_2 + n_2m_1
        合成 G1[G2]G_1[G_2]n1n2n_1 n_2n1m2+n22m1n_1m_2 + {n_2}^2m_1

        # 方体

        一族特别重要的图称为方体,它用积来定义最为自然。

        nn 方体 QnQ_n 递推地定义为:

        • Q1=K2Q_1=K_2
        • Qn=K2×Qn1Q_n=K_2\times Q_{n-1}

        QnQ_n2n2^n 个点,它的点可以用二进制串 a1a2ana_1 a_2 \cdots a_n 来标定,其中 ai{0,1}a_i\in\{0,1\}

        QnQ_n 两个点的二进制表示式中只有一位不同,则它们邻接

        2方体

        3方体

        # 路与图的连通性

        # 途径、迹、路

        一个有限非空序列:

        w=v0e1v1e2v2ekvkw=v_0\ e_1\ v_1\ e_2\ v_2\cdots e_k\ v_k

        它的项交替地为顶点和边,使得对 1ik1\leq i\leq keie_i ,的端点是 vi1v_{i-1}viv_i、 称 ww 是从 v0v_0vkv_k 的一条途径,或一条 (v0,vk)(v_0,v_k) 途径

        途径中边数称为途径的长度

        v0v_0vkv_k 分别称为途径的起点终点,其余顶点称为途径的内部点

        一条途径的起点与终点重合时叫做闭途径

        边不重复的途径称为图的一条,起点与终点重合时叫做闭迹,也称为回路

        顶点不重复的途径称为图的一条,起点与终点重合时叫做。显然顶点不重复时边也不重复,所以迹是包含路的。

        起点与终点重合的途径、迹、路分别称为图的闭途径、闭迹与圈。闭迹也称为回路

        长度为 kk 的圈称为 kk 圈,kk 为奇数时称为奇圈kk 为偶数时称为偶圈

        # 圈的充分条件

        δ2\delta ≥2,则 GG 中必然含有圈。

        只就连通图证明即可!

        W=v1v2vk1vkW=v_1 v_2\cdots v_{k-1}v_kGG 中的一条最长路。由于 δ2\delta≥2,所以 vkv_k 必然有相异于 vk1v_{k-1} 的邻接顶点。

        WW 是连通图 GG 中最长路,所以,这样的邻接点必然是 v1v2vk2v_1 v_2\cdots v_{k-2} 中之一。设该点为 vmv_m ,则 vmvm+1vkvmv_m v_{m+1}\cdots v_k v_mGG 中圈。

        # 偶图的充要条件

        一个图是二部图 (偶图) 当且当它不包含奇圈


        证明必要性:

        GG 是具有二分类 (X,Y)(X,Y) 的偶图,并且 C=v0v1vkv0C=v_0 v_1\cdots v_k v_0GG 的一个圈

        不失一般性,可假定 v0Xv_0\in X

        一般说来,v2iX,v2i+1Yv_{2i}\in X,v_{2i+1}\in Y

        又因为v0Xv_0\in X ,所以 vkYv_k\in Y,由此即得 CC 是偶圈


        证明充分性:

        GG 中任意选取点 uu,定义 VV 的分类如下:

        • X={xd(u,x)是偶数,xV(G)}X=\{x|d(u,x)\text{是偶数},x\in V(G)\}
        • Y={yd(u,y)是奇数,yV(G)}Y=\{y|d(u,y)\text{是奇数},y\in V(G)\}

        (接下来证明 XX 中任意两点 v,wv,w 不邻接)

        v,wv,wXX 中任意两个顶点,PP 是一条最短 (u,v)(u,v) 路,而 QQ 是一条最短 (u,w)(u,w) 路,设 u1u_1PPQQ 的最后一个交点:

        偶图充分必要条件充分性证明

        由于 P,QP,Q 是最短路,所以,P,QP,Quuu1u_1 段长度相同,因此奇偶性相同

        P,QP,Q 的长均是偶数,所以,P,QP,Qu1vu_1 v 段和 u1wu_1 w 段奇偶性相同

        如果 v,wv,w 邻接则可得到奇圈,矛盾!

        # 两顶点的距离

        图中顶点 uuvv 的距离: uuvv 间最短路的长度称为 uuvv 间距离。记为:

        d(u,v)d(u,v)

        如果 uuvv 间不存在路,定义:

        d(u,v)=0d(u,v)=0

        # 两顶点的连通性

        GG 中点 uuvv 说是连通的,如果 uuvv存在通路。否则称 uuvv 不连通

        点的连通关系是等价关系

        非连通图中每一个极大连通部分,称为 GG连通分支

        GG 的连通分支的个数,称为 GG分支数,记为:

        ω(G)\omega(G)

        # 图的连通性

        如果图 GG 中任意两点是连通的,称 GG连通图,否则,称 GG非连通图

        # 连通图的性质

        nn 阶连通图中:

        1. 至少有 n1n-1 条边
        2. 如果边数大于 n1n-1,则至少有一条闭迹
        3. 如果恰有 n1n-1 条边,则至少有一个奇度点

        (2)证明:

        GG 中没有 11 度顶点,由握手定理:

        2m=vV(G)d(v)2nmnm>n12m=\sum_{v\in V(G)}d(v)\geq 2n \rArr m\geq n \rArr m \gt n-1


        (3)证明:

        若不然,GG 中顶点度数至少为 22,于是由握手定理:

        2m=v(G)d(v)2nmnm>n12m=\sum_{v\in(G)}d(v) \geq 2n \rArr m ≥n \rArr m > n-1

        这与 GG 中恰有 n1n-1 条边矛盾!

        # 补图的连通性

        若图 GG 是不连通的,则它的补图 G\overline G 是连通图.


        证明:设 u,vu,vGG 的任意两个顶点:

        • uuvvGG 中不邻接,则在 G\overline G 中它们邻接,从而 G\overline G 是连通的
        • uuvvGG 中邻接,它们属于 GG 的同一分支;在另一个分支中有一点 ww,在 G\overline Guuvv 均与 w 邻接,即 uwvuwv 是一条通道,从而 G\overline G 是连通的

        # 连通图充分条件 1

        设图 GGnn 阶图,若 GG 中任意两个不相邻的点 uuvv,有:

        d(u)+d(v)n1d(u)+d(v)\geq n-1

        GG 是连通的,且 d(G)2d(G)\leq 2


        证明:对 GG 中任意两点 xxyy,一定存在一条长度至多为 22 的连接 xxyy 的路

        xyE(G)xy\in E(G) 时上述论断显然成立;

        xyE(G)xy\notin E(G) 时:

        可以证明,存在点 ww,它与 xxyy 同时邻接。反证法:

        若不然,在 GG 的剩下 n2n-2 个顶点中,假设:

        • kk 个与 xx 邻接,但与 yy 不邻接;
        • ll 个与 yy 邻接,但与 xx 不邻接;
        • mm 个顶点和 x,yx,y 均不邻接。

        则:d(x)=kd(x)=kd(y)=ld(y)=l

        由于 k+l+m=n2k+l+m=n-2

        所以 d(x)+d(y)=nm2n2d(x)+d(y)=n-m-2\leq n-2 ,矛盾!


        # 连通图充分条件 2

        GGnn 阶图,若 δn12\delta≥\frac{n-1}{2} ,则 GG 连通


        证明:

        对 G 中任意两个不相邻顶点 u 与 v,有:

        d(u)+d(v)n12+n12=n1d(u)+d(v)\geq\frac{n-1}{2}+\frac{n-1}{2}=n-1

        所以,G 是连通的。

        该定理的界是紧的 (Sharpness)。即不能再修改!

        例如:设 G 由两个分支作成的图,两个分支均为 KmK_m ,则 G 中不相邻顶点度数之和恰为 n-2 . (n=2m)

        # 图的直径

        连通图 G 的直径定义为

        d(G)=max{d(u,v)u,vV(G)}d(G) = \text{max}\{d(u,v)|u,v \in V(G)\}

        如果 G 不连通,图 G 的直径定义为 d(G)=d(G) = \infin

        # 直径相关性质

        设 G 是具有 m 条边的 n 阶单图,若 G 的直径为 2 且 δ=n2\delta=n-2 ,则 m2n4m≥2n-4


        证明:

        d(v)=Δ=n2d(v)=\Delta=n-2 ,设 v 的邻接点为 v1,v2,,vn2v_1,v_2,\cdots,v_{n-2} ,u 是剩下的一个项点。

        由于 d(G)=2d(G)=2 且 u 不能和 v 邻接,所以 u 至少和 v1,v2,,vn2v_1,v_2,\cdots,v_{n-2} 中的一个顶点邻接。否则有 d(G)>2d(G)>2 ; 不妨假设 u 和 v1,v2,,vkv_1,v_2,\cdots,v_{k} 邻接。

        为了保证 u 到各点距离不超过 2 , vk+1,,vn2v_{k+1},\cdots,v_{n-2} 这些頂点的每一个必须和前面 v1,v2,,vkv_1,v_2,\cdots,v_{k} 中某点邻接,这样,图中至少又有 n-2 条边。总共至少有 2n-4 条边。

        \ No newline at end of file diff --git a/math/discrete-math/graph/match/index.html b/math/discrete-math/graph/match/index.html index 85c15b63..d42c11a9 100644 --- a/math/discrete-math/graph/match/index.html +++ b/math/discrete-math/graph/match/index.html @@ -1 +1 @@ -匹配 - 图论 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 匹配

        # 匹配基础概念

        匹配 MM —— 如果 MM 是图 GG 的边子集 (不含环),且 MM 中的任意两条边没有共同顶点,则称 MMGG 的一个匹配,或对集,或边独立集。

        如果 GG 中顶点是 GG 的匹配 MM 中某条边的端点,称它为 MM 饱和点,否则为 MM 非饱和点

        最大匹配 MM (无向图定义) —— 如果 MM 是图 GG 的包含边数最多的匹配,称 MMGG 的一个最大匹配。

        特别是,若最大匹配饱和了 GG 的所有顶点,称它为 GG 的一个完美匹配

        MM 交错路 —— 如果 MM 是图 GG 的匹配, GG 中一条由 MM 中的边和非 MM 中的边交错形成的路,称为 GG 中的一条 MM 交错路。

        特别地,若 MM 交错路的起点与终点都是 MM 非饱和点,称这种 MM 交错路为 MM 可扩路

        # 贝尔热定理

        Berge

        GG 的匹配 MM 是最大匹配,当且仅当 GG 不包含 MM 可扩路。

        # HALL 定理

        完美匹配的充要条件,将 XX 视为需求,YY 视为供给,可以理解为供给必须满足需求。

        G=(X,Y)G=(X,Y) 是偶图,则 GG 存在饱和 XX 每个顶点的完美匹配的充要条件是:

        SX,N(S)S\forall S\subseteq X,\ \ |N(S)|\geq|S|

        其中 N(S)N(S) 表示 SS 的邻集(图 GG 中所有与 SS 相邻接顶点的集合),N(S)N(S) 一定是 YY 的子集。

        G=(X,Y)G=(X,Y) 存在饱和 XX 每个顶点的匹配也常说成存在由 XXYY 的匹配 。

        ⭐️推论:若 GGk(k>0)k(k\gt0) 正则偶图,则 GG 存在完美匹配。

        # 典例

        # 例 1

        证明每个 kk 方体都是 kk 正则偶图。

        事实上,由 kk 方体的构造: kk 方体有 2k2k 个顶点,每个顶点可以用长度为 kk 的二进制码来表示,两个顶点连线当且仅当代表两个顶点的二进制码只有一位坐标不同。

        如果我们划分 kk 方体的 2k2k 个顶点,把坐标之和为偶数的顶点归入 XX,否则归入 YY 。显然,XX 中顶点互不邻接,YY 中顶点也如此。所以 kk 方体是偶图。

        又不难知道 kk 方体的每个顶点度数为 kk , 所以 kk 方体是 kk 正则偶图。

        由推论: kk 方体存在完美匹配。

        直接在 k 方体中找出完美匹配。

        kk 方体顶点二进制码为 x1,x2,,xkx_1,x_2,\cdots,x_k , 我们取 x1,x2,,xk1,0x_1,x_2,\cdots,x_{k-1},0 , 和

        (x1,x2,,xk1,1)(x-1,x_2,\cdots,x_{k-1},1) 之间的全体边所成之集为 MM

        显然,MM 中的边均不相邻接,所以作成 kk 方体的匹配,又容易知道:M=2k1|M|=2^{k-1},所以 MM 是完美匹配。

        # 例 2

        𝐾2𝑛𝐾_{2𝑛}𝐾𝑛,𝑛𝐾_{𝑛,𝑛} 中不同的完美匹配的个数

        𝐾2𝑛𝐾_{2𝑛} 的任意一个顶点有 2𝑛12𝑛-1 种不同的方法被匹配。所以 𝐾2𝑛𝐾_{2𝑛} 的不同完美匹配个数等于 (2𝑛1)𝐾2𝑛2(2𝑛-1)𝐾_{2𝑛-2} , 如此推下去,可以归纳出 𝐾2𝑛𝐾_{2𝑛} 的不同完美匹配个数为:(2𝑛1)!!(2𝑛-1)!!

        同样的推导方法可归纳出 𝐾𝑛,𝑛𝐾_{𝑛,𝑛} 的不同完美匹配个数为:𝑛!𝑛!

        # 例 3

        证明树至多存在一个完美匹配。

        证明:若不然,设 𝑀1𝑀_1𝑀2𝑀_2 是树 TT 的两个不同的完美匹配,那么 𝑀1Δ𝑀2𝑀_1\Delta𝑀_2≠\varnothing, 容易知道:𝑇[𝑀1Δ𝑀2]𝑇[𝑀_1\Delta 𝑀_2] 每个非空部分顶点度数为 2 ,即它存在圈,于是推出 TT 中有圈,矛盾。

        # 图的点覆盖

        GG 的一个顶点子集 KK 称为 GG 的一个点覆盖,如果 GG 的每条边都至少有一个端点在 KK 中。GG 的一个包含点数最少的点覆盖称为 GG 的最小点覆盖,其包含的点数称为 GG 的覆盖数,记为 α(G)\alpha(G)

        ⭐️设 MMGG 的匹配,KKGG 的覆盖,若 M=K|M|=|K| , 则 MM 是最大匹配,而 KK 是最小覆盖。

        # 哥尼定理

        偶图的点覆盖与偶图匹配间的关系

        在偶图中,最大匹配的边数等于最小覆盖的顶点数。

        # 托特定理

        GG 有完美匹配当且仅当对 VV 的任意非空真子集 SS , 有:

        o(GS)So(G-S)\leq|S|

        o(GS)o(G-S) 表示奇分支数目(有奇数个顶点的分支)

        # 彼得森定理

        哥尼定理的推论,是完美匹配的充分条件而不是必要条件⚠️

        没有割边的 3 正则图存在完美匹配

        # 匈牙利算法

        可以解决问题:

        • 在无权偶图中寻找完美匹配
        • 在有权偶图中寻找最优匹配
          • 在有权偶图中寻找最小匹配(权值和最小)
          • 在有权偶图中寻找最大匹配(权值和最大,通过所有权值取反转化为最小匹配问题)。

        G=(X,Y)G=(X,Y),匈牙利算法需要满足约束:X=Y|X|=|Y|

        基本思想

        从任意初始匹配 M0M_0 出发,通过寻求一条 M0M_0 可扩路 PP ,令 M1=M0ΔE(P)M_1=M_0\Delta E(P) ,得到比 M0M_0 更大的匹配 M1M_1

        1965 年,Edmonds 首先提出:用扎根于 MM 非饱和点 uuMM 交错树的生长来求 MM 可扩路。

        # 无权偶图完美匹配

        # M 交错树

        定义 设 G=(X,Y)G=(X,Y)MMGG 的匹配,uuMM 非饱和点。称树 HHGG 的扎根于点 uuMM 交错树,如果:

        • uV(H)u\in V(H)
        • 对任意 vV(H)v\in V(H)(u,v)(u,v) 路是 MM 交错路;

        扎根于 MM 非饱和点 uuMM 交错树 HH 有两种情形:

        • 情形 1: 除点 uu 外,HH 中所有点为 MM 饱和点,且在 MM 上配对;
        • 情形 2: HH 包含除 uu 外的 MM 非饱和点。

        # 匈牙利算法

        MM 是初始匹配。HH 是扎根于 MM 非饱和点 uu 的交错树。令:S=V(H)XS=V(H)\cap XT=V(H)YT=V(H)\cap Y

        • (a)
          • MM 饱和 XX 所有顶点,停止;
          • 否则,设 uuXXMM 非饱和顶点,置 S={u}S=\{u\}T=T=\varnothing
        • (b)
          • N(S)=TN(S)=T,则 GG 中不存在完美匹配;
          • 否则设 yN(S)Ty\in N(S)-T
        • (c)
          • yyMM 饱和点,且 yzMyz\in M,置 S=S{z}S=S\cup\{z\}T=T{y}T=T\cup \{y\},转(b);
          • 否则,设 PPMM 可扩路,置 M=MΔE(P)M=M\Delta E(P),转(a)

        # 有权图最优匹配

        # 可行顶点标号

        G=(X,Y)G=(X,Y) ,若对 xX,yY\forall x\in X,\forall y\in Y,有:

        l(x)+l(y)w(xy)l(x)+l(y)\geq w(xy)

        则称 ll 是赋权完全偶图 GG可行顶点标号

        一个简单常用的可行顶点标号:

        {l(x)=maxyYw(xy),xXl(y)=0,yY\begin{cases} l(x)=\max_{y\in Y}w(xy),&x\in X\\ l(y)=0,&y\in Y \end{cases}

        # 相等子图

        ll 是赋权完全偶图 G=(X,Y)G=(X,Y) 的可行顶点标号,令:

        El={xyE(G)l(x)+l(y)=w(xy)}E_l=\{xy\in E(G)|l(x)+l(y)=w(xy)\}

        Gl=G[El]G_l=G[E_l]GG 的对应于 ll相等子图

        # Kuhn 算法

        Kuhn 采用顶点标号修改策略,找到了求最优匹配好算法如下

        ⚠️ 求的是权值和最大的匹配⚠️

        给一初始顶点标号 ll ,在相等子图 GlG_l 中任选一个匹配 MM

        (1)

        • XXMM 饱和的,则 MM 是最优匹配;
        • 否则,令 uu 是一个 MM 非饱和点,置 S={u}S=\{u\}T=T=\varnothing

        (2)

        • TNGl(S)T\sub N_{G_l}(S),转(3);
        • 否则,计算:

        αl=minxS,yT{l(x)+l(y)w(xy)}\alpha_l=\min_{x\in S,y\notin T}\{l(x)+l(y)-w(xy)\}

        修改 ll 如下:

        l={l(v)αl,vSl(v)+αl,vTl(v),othersl=\begin{cases} l(v)-\alpha_l,&v\in S\\ l(v)+\alpha_l,&v\in T\\ l(v),&\text{others}\\ \end{cases}

        给出新的可行顶点标号,在新标号下重新开始。

        (3)在 NGl(S)\TN_{G_l}(S)\backslash T 中选择点 yy

        • yyMM 饱和的,yzMyz\in M,则置 S=S{z}S=S\cup\{z\}T=T{y}T=T\cup\{y\},转(2);
        • 否则,设 PPGlG_lMM 可扩路,置 M=MΔE(P)M=M\Delta E(P),转(1)

        该算法把上面原来的匈牙利算法用于其中,主要是用来判定和求完美匹配。

        \ No newline at end of file +匹配 - 图论 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 匹配

        # 匹配基础概念

        匹配 MM —— 如果 MM 是图 GG 的边子集 (不含环),且 MM 中的任意两条边没有共同顶点,则称 MMGG 的一个匹配,或对集,或边独立集。

        如果 GG 中顶点是 GG 的匹配 MM 中某条边的端点,称它为 MM 饱和点,否则为 MM 非饱和点

        最大匹配 MM (无向图定义) —— 如果 MM 是图 GG 的包含边数最多的匹配,称 MMGG 的一个最大匹配。

        特别是,若最大匹配饱和了 GG 的所有顶点,称它为 GG 的一个完美匹配

        MM 交错路 —— 如果 MM 是图 GG 的匹配, GG 中一条由 MM 中的边和非 MM 中的边交错形成的路,称为 GG 中的一条 MM 交错路。

        特别地,若 MM 交错路的起点与终点都是 MM 非饱和点,称这种 MM 交错路为 MM 可扩路

        # 贝尔热定理

        Berge

        GG 的匹配 MM 是最大匹配,当且仅当 GG 不包含 MM 可扩路。

        # HALL 定理

        完美匹配的充要条件,将 XX 视为需求,YY 视为供给,可以理解为供给必须满足需求。

        G=(X,Y)G=(X,Y) 是偶图,则 GG 存在饱和 XX 每个顶点的完美匹配的充要条件是:

        SX,N(S)S\forall S\subseteq X,\ \ |N(S)|\geq|S|

        其中 N(S)N(S) 表示 SS 的邻集(图 GG 中所有与 SS 相邻接顶点的集合),N(S)N(S) 一定是 YY 的子集。

        G=(X,Y)G=(X,Y) 存在饱和 XX 每个顶点的匹配也常说成存在由 XXYY 的匹配 。

        ⭐️推论:若 GGk(k>0)k(k\gt0) 正则偶图,则 GG 存在完美匹配。

        # 典例

        # 例 1

        证明每个 kk 方体都是 kk 正则偶图。

        事实上,由 kk 方体的构造: kk 方体有 2k2k 个顶点,每个顶点可以用长度为 kk 的二进制码来表示,两个顶点连线当且仅当代表两个顶点的二进制码只有一位坐标不同。

        如果我们划分 kk 方体的 2k2k 个顶点,把坐标之和为偶数的顶点归入 XX,否则归入 YY 。显然,XX 中顶点互不邻接,YY 中顶点也如此。所以 kk 方体是偶图。

        又不难知道 kk 方体的每个顶点度数为 kk , 所以 kk 方体是 kk 正则偶图。

        由推论: kk 方体存在完美匹配。

        直接在 k 方体中找出完美匹配。

        kk 方体顶点二进制码为 x1,x2,,xkx_1,x_2,\cdots,x_k , 我们取 x1,x2,,xk1,0x_1,x_2,\cdots,x_{k-1},0 , 和

        (x1,x2,,xk1,1)(x-1,x_2,\cdots,x_{k-1},1) 之间的全体边所成之集为 MM

        显然,MM 中的边均不相邻接,所以作成 kk 方体的匹配,又容易知道:M=2k1|M|=2^{k-1},所以 MM 是完美匹配。

        # 例 2

        𝐾2𝑛𝐾_{2𝑛}𝐾𝑛,𝑛𝐾_{𝑛,𝑛} 中不同的完美匹配的个数

        𝐾2𝑛𝐾_{2𝑛} 的任意一个顶点有 2𝑛12𝑛-1 种不同的方法被匹配。所以 𝐾2𝑛𝐾_{2𝑛} 的不同完美匹配个数等于 (2𝑛1)𝐾2𝑛2(2𝑛-1)𝐾_{2𝑛-2} , 如此推下去,可以归纳出 𝐾2𝑛𝐾_{2𝑛} 的不同完美匹配个数为:(2𝑛1)!!(2𝑛-1)!!

        同样的推导方法可归纳出 𝐾𝑛,𝑛𝐾_{𝑛,𝑛} 的不同完美匹配个数为:𝑛!𝑛!

        # 例 3

        证明树至多存在一个完美匹配。

        证明:若不然,设 𝑀1𝑀_1𝑀2𝑀_2 是树 TT 的两个不同的完美匹配,那么 𝑀1Δ𝑀2𝑀_1\Delta𝑀_2≠\varnothing, 容易知道:𝑇[𝑀1Δ𝑀2]𝑇[𝑀_1\Delta 𝑀_2] 每个非空部分顶点度数为 2 ,即它存在圈,于是推出 TT 中有圈,矛盾。

        # 图的点覆盖

        GG 的一个顶点子集 KK 称为 GG 的一个点覆盖,如果 GG 的每条边都至少有一个端点在 KK 中。GG 的一个包含点数最少的点覆盖称为 GG 的最小点覆盖,其包含的点数称为 GG 的覆盖数,记为 α(G)\alpha(G)

        ⭐️设 MMGG 的匹配,KKGG 的覆盖,若 M=K|M|=|K| , 则 MM 是最大匹配,而 KK 是最小覆盖。

        # 哥尼定理

        偶图的点覆盖与偶图匹配间的关系

        在偶图中,最大匹配的边数等于最小覆盖的顶点数。

        # 托特定理

        GG 有完美匹配当且仅当对 VV 的任意非空真子集 SS , 有:

        o(GS)So(G-S)\leq|S|

        o(GS)o(G-S) 表示奇分支数目(有奇数个顶点的分支)

        # 彼得森定理

        哥尼定理的推论,是完美匹配的充分条件而不是必要条件⚠️

        没有割边的 3 正则图存在完美匹配

        # 匈牙利算法

        可以解决问题:

        • 在无权偶图中寻找完美匹配
        • 在有权偶图中寻找最优匹配
          • 在有权偶图中寻找最小匹配(权值和最小)
          • 在有权偶图中寻找最大匹配(权值和最大,通过所有权值取反转化为最小匹配问题)。

        G=(X,Y)G=(X,Y),匈牙利算法需要满足约束:X=Y|X|=|Y|

        基本思想

        从任意初始匹配 M0M_0 出发,通过寻求一条 M0M_0 可扩路 PP ,令 M1=M0ΔE(P)M_1=M_0\Delta E(P) ,得到比 M0M_0 更大的匹配 M1M_1

        1965 年,Edmonds 首先提出:用扎根于 MM 非饱和点 uuMM 交错树的生长来求 MM 可扩路。

        # 无权偶图完美匹配

        # M 交错树

        定义 设 G=(X,Y)G=(X,Y)MMGG 的匹配,uuMM 非饱和点。称树 HHGG 的扎根于点 uuMM 交错树,如果:

        • uV(H)u\in V(H)
        • 对任意 vV(H)v\in V(H)(u,v)(u,v) 路是 MM 交错路;

        扎根于 MM 非饱和点 uuMM 交错树 HH 有两种情形:

        • 情形 1: 除点 uu 外,HH 中所有点为 MM 饱和点,且在 MM 上配对;
        • 情形 2: HH 包含除 uu 外的 MM 非饱和点。

        # 匈牙利算法

        MM 是初始匹配。HH 是扎根于 MM 非饱和点 uu 的交错树。令:S=V(H)XS=V(H)\cap XT=V(H)YT=V(H)\cap Y

        • (a)
          • MM 饱和 XX 所有顶点,停止;
          • 否则,设 uuXXMM 非饱和顶点,置 S={u}S=\{u\}T=T=\varnothing
        • (b)
          • N(S)=TN(S)=T,则 GG 中不存在完美匹配;
          • 否则设 yN(S)Ty\in N(S)-T
        • (c)
          • yyMM 饱和点,且 yzMyz\in M,置 S=S{z}S=S\cup\{z\}T=T{y}T=T\cup \{y\},转(b);
          • 否则,设 PPMM 可扩路,置 M=MΔE(P)M=M\Delta E(P),转(a)

        # 有权图最优匹配

        # 可行顶点标号

        G=(X,Y)G=(X,Y) ,若对 xX,yY\forall x\in X,\forall y\in Y,有:

        l(x)+l(y)w(xy)l(x)+l(y)\geq w(xy)

        则称 ll 是赋权完全偶图 GG可行顶点标号

        一个简单常用的可行顶点标号:

        {l(x)=maxyYw(xy),xXl(y)=0,yY\begin{cases} l(x)=\max_{y\in Y}w(xy),&x\in X\\ l(y)=0,&y\in Y \end{cases}

        # 相等子图

        ll 是赋权完全偶图 G=(X,Y)G=(X,Y) 的可行顶点标号,令:

        El={xyE(G)l(x)+l(y)=w(xy)}E_l=\{xy\in E(G)|l(x)+l(y)=w(xy)\}

        Gl=G[El]G_l=G[E_l]GG 的对应于 ll相等子图

        # Kuhn 算法

        Kuhn 采用顶点标号修改策略,找到了求最优匹配好算法如下

        ⚠️ 求的是权值和最大的匹配⚠️

        给一初始顶点标号 ll ,在相等子图 GlG_l 中任选一个匹配 MM

        (1)

        • XXMM 饱和的,则 MM 是最优匹配;
        • 否则,令 uu 是一个 MM 非饱和点,置 S={u}S=\{u\}T=T=\varnothing

        (2)

        • TNGl(S)T\sub N_{G_l}(S),转(3);
        • 否则,计算:

        αl=minxS,yT{l(x)+l(y)w(xy)}\alpha_l=\min_{x\in S,y\notin T}\{l(x)+l(y)-w(xy)\}

        修改 ll 如下:

        l={l(v)αl,vSl(v)+αl,vTl(v),othersl=\begin{cases} l(v)-\alpha_l,&v\in S\\ l(v)+\alpha_l,&v\in T\\ l(v),&\text{others}\\ \end{cases}

        给出新的可行顶点标号,在新标号下重新开始。

        (3)在 NGl(S)\TN_{G_l}(S)\backslash T 中选择点 yy

        • yyMM 饱和的,yzMyz\in M,则置 S=S{z}S=S\cup\{z\}T=T{y}T=T\cup\{y\},转(2);
        • 否则,设 PPGlG_lMM 可扩路,置 M=MΔE(P)M=M\Delta E(P),转(1)

        该算法把上面原来的匈牙利算法用于其中,主要是用来判定和求完美匹配。

        \ No newline at end of file diff --git a/math/discrete-math/graph/planar-graph/index.html b/math/discrete-math/graph/planar-graph/index.html index cdee5468..d099a187 100644 --- a/math/discrete-math/graph/planar-graph/index.html +++ b/math/discrete-math/graph/planar-graph/index.html @@ -1 +1 @@ -平面图 - 图论 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        如果能把图 GG 画在平面上,使得除顶点外,边与边之间没有交叉,称 GG 可嵌入平面,或称 GG 是可平面图

        可平面图 GG 的边不交叉的一种画法,称为 GG 的一种平面嵌入GG 的平面嵌入表示的图称为平面图

        可平面图概念和平面图概念有时可以等同看待;

        图的平面性问题主要涉及如下几个方面:

        • 平面图的性质
        • 平面图的判定
        • 平面嵌入方法 (平面性算法)
        • 涉及图的平面性问题的拓扑不变量

        # 平面图的性质

        一个平面图 GG 把平面分成若干连通片,这些连通片称为 GG区域,或 GG 的一个GG 的面组成的集合用 Φ\Phi 表示。

        面积有限的区域称为平面图 GG内部面,否则称为 GG外部面

        GG 中,顶点和边都与某个给定区域关联的子图,称为该面的边界。某面 ff 的边界中含有的边数 (割边计算 2 次⚠️) 称为该面 ff次数,记 degf\deg f

        # 平面图的次数公式

        G=(n,m)G=(n,m) 是平面图,则:

        fΦdeg(f)=2m\sum_{f\in\Phi}\deg(f)=2m


        证明:

        GG 的任意一条边 ee, 如果 ee 是某面割边,那么由面的次数定义,该边给 GG 的总次数贡献 2 次;如果 ee 不是割边,那么,它必然是两个面的公共边,因此,由面的次数定义,它也给总次数贡献 2 次。

        # 连通平面图的欧拉公式

        G=(n,m)G=(n,m)连通平面图,φ\varphiGG 的面数,则:

        nm+φ=2n-m+\varphi=2


        证明:

        情形 1,如果 GG 是树。

        那么 m=n1m=n-1, 𝜑=1。在这种情况下,容易验证,定理中的恒等式是成立的。

        情形 2, GG 不是树的连通平面图。

        假设在这种情形下,欧拉恒等式不成立。则存在一个含有最少边数的连通平面图 GG,使得它不满足欧拉恒等式。设这个最少边数连通平面图 G=(n,m)G=(n,m),面数为 𝜑,则:

        nm+φ2n-m+\varphi\neq2

        因为 GG 不是树,所以存在非割边 ee。显然,GeG-e 是连通平面图,边数为 m1m-1,顶点数为 nn,面数为 𝜑-1。由最少性假设GeG-e 满足欧拉等式:

        n(m1)+(φ1)=2nm+φ=2\begin{aligned} n-(m-1)+(\varphi-1)&=2\\ n-m+\varphi&=2 \end{aligned}

        于是矛盾,假设不成立。

        # 非连通平面图的欧拉公式

        G=(n,m)G=(n,m)kk 个连通分支的平面图,φ\varphiGG 的面数,则:

        nm+φ=k+1n-m+\varphi=k+1


        证明:

        对第 i(1ik)i(1≤i≤k) 个分支来说,设顶点数为 nin_i,边数为 mim_i,面数为 𝜑_i,由欧拉公式:

        nimi+φi=2i=1k(nimi+φi)=2ki=1knii=1kmi+i=1kφi=2k\begin{aligned} n_i-m_i+\varphi_i&=2\\ \sum_{i=1}^{k}(n_i-m_i+\varphi_i)&=2k\\ \sum_{i=1}^{k}n_i-\sum_{i=1}^{k}m_i+\sum_{i=1}^{k}\varphi_i&=2k \end{aligned}

        同时:(扣除重复计算的 k1k-1 个外部面)

        i=1kni=n,i=1kmi=m,i=1kφi=φ+k1\sum_{i=1}^{k}n_i=n,\ \ \sum_{i=1}^{k}m_i=m,\ \ \sum_{i=1}^{k}\varphi_i=\varphi+k-1

        于是得到:

        nm+(φ+k1)=2knm+φ=k+1\begin{aligned} n-m+(\varphi+k-1)&=2k\\ n-m+\varphi&=k+1 \end{aligned}

        GG 是具有 nn 个点 mm 条边 𝜑 个面的连通平面图,如果对 GG 的每个面 ff,有 degfl3\deg f≥l≥3,则:

        mll2(n2)m\leq\frac{l}{l-2}(n-2)

        这是连通平面图的必要条件,不是充分条件!

        GG 是具有 nn 个点 mm 条边 𝜑 个面的连通平面图,有:

        m3n6m\leq3n-6

        GG 是具有 nn 个点 mm 条边 𝜑 个面的连通平面图,如果对 GG 的每个面 ff,有 degf=l\deg f=l,则:

        m(l2)=l(n2)m(l-2)=l(n-2)

        GG 是具有 nn 个点 mm 条边 𝜑 个面的连通平面图,有:

        δ5\delta\leq5

        该结论是证明 “5 色定理” 的出发点

        一个连通平面图是 22 连通的,当且仅当它的每个面的边界是圈。

        # 图的嵌入性问题

        # 曲面嵌入

        # 球面嵌入

        ⭐️ GG 可球面嵌入与 GG 可平面嵌入等价

        我们实际上对图的嵌入性问题也只考虑这两个。

        # 环面嵌入

        # 定向曲面嵌入

        # 3 维空间嵌入

        所有图均可嵌入 R3R^3 中。

        # 凸多面体与平面图

        一个多面体称为凸多面体,如果在体上任取两点,其连线均在体上。

        凸多面体的一维骨架:把一个凸多面体压缩在平面上,得到一个对应的平面图,该平面图称为该凸多面体的一维骨架。

        存在且只存在 5 种正多面体:它们是正四、六、八、十二、二十面体

        # 平面图的判定

        # 同胚

        • 在图 GG 的边上插入一个新的 22 度顶点,使一条边分成两条边,则称将图 GG 在 2 度顶点内扩充
        • 去掉图 的一个 22 度顶点,使这个 22 度顶点关联的两条边合成一条边,则称将 GG 在 2 度顶点内收缩

        两个图 G1G_1G2G_2,如果 G1G2G_1\cong G_2,或者通过反复在 22 度顶点内扩充和收缩它们能变成同构的,则称 G2G_2G2G_2同胚的,或 G1G_1G2G_2 22 度顶点内是同构的

        ⭐️图的可平面性在同胚意义下不变。

        # 库拉托夫斯基定理

        GG 是可平面的当且仅当它不含与 K5K_5K3,3K_{3,3} 同胚的子图

        库拉托夫斯基定理可以等价叙述为:「图 GG 是非可平面的当且仅当它含有与 K5K_5K3,3K_{3,3} 同胚的子图」

        给定图 GG,去掉 GG 中的环 (若有的话),将 GG 中的重边 (若有的话) 用单边代替,称这样所得的图为 GG基础简单图

        • GG 是可平面图当且仅当其基础简单图是可平面图
        • GG 是可平面图当且仅当它的每个块是可平面图

        # 初等收缩

        uvuv 是简单图 GG 的一条边。去掉该边,重合其端点,再删去由此产生的环和重边。这一过程称为图 GG初等收缩收缩边 uvuv 的运算,并记所得之图为 G/uvG/uv

        一个图 GG 可收缩到 HH,是指 HH 可从 GG 通过一系列初等收缩而得到。

        # 瓦格纳定理

        简单图 GG 是可平面图当且仅当它不含可收缩到 K5K_5K3,3K_{3,3} 的子图。

        该定理和库拉托夫斯基定理是等价的。

        至少有 9 个点的简单可平面图的补图是不可平面的,而 9 是这种数目中最小的一个

        # 平面性的拓扑不变量

        # 亏格

        # 环柄

        在边交叉处建立的 “立交桥”,通过环柄使得一条边经过 “桥下”,另一条边经过 “桥上”,使二者分开。

        # 亏格定义

        若通过加上 kk 个环柄可将图 GG 嵌入球面,则 kk 得最小值称为 GG亏格,记为:

        γ(G)\gamma(G)

        • GG 是可平面图,则 γ(G)=0\gamma(G)=0
        • 同胚的图是亏格相等;
        • γ(K5)=1\gamma(K_5)=1

        # 引入亏格的欧拉公式

        对一个亏格为 γ\gamma,有 nn 个顶点,mm 条边和 φ\varphi 个面的多面体或连通图,有:

        nm+φ=22γn-m+\varphi=2-2\gamma

        # 结论

        GG 是一个有 nn 个顶点,mm 条边和 φ\varphi 个面的多面体或连通图,有:

        • γm6n22\gamma\geq\frac{m}{6}-\frac{n-2}{2}
        • GG 无三角形,则 γm4n22\gamma\geq\frac{m}{4}-\frac{n-2}{2}
        • GG 每个面是三角形,则 m=3(n2+2γ)m=3(n-2+2\gamma)
        • GG 每个面是四边形,则 m=2(n2+2γ)m=2(n-2+2\gamma)

        # 厚度

        若图 GGkk 个可平面子图的并等于 ,则称 kk 的最小值为 GG厚度,记为

        GG 是可平面图当且仅当 θ(G)=1\theta(G)=1

        # 叉数

        GG 画在平面上时相交的边对的最少数目称为 GG叉数,记为:

        η(G)\eta(G)

        GG 是可平面图当且仅当 η(G)=0\eta(G)=0

        # 糙度

        GG 中边不重的不可平面子图的最多数目称为 GG糙度,记为:

        ζ(G)\zeta(G)

        # 平面性算法

        HH 是图 GG 的一个子图,在 E(G)\E(H)E(G)\backslash E(H) 上定义关系 \sim 如下:e1e2e_1\sim e_2 当且仅当存在一条途径 WW,使得:

        • WW 的第一条边与最后一条边分别是 e1e_1e2e_2
        • WW 的内部顶点与 HH 不相交

        易证关系 \sim 具有自反性,对称性和传递性,从而是 E(G)\E(H)E(G)\backslash E(H) 上的一个等价关系

        此等价关系的等价类导出的 GE(H)G-E(H) 的边导出子图称为 HH - 片段,或者 GG 关于 HH 的桥。片段或桥与 HH 的公共顶点称为附着顶点

        • BBHH - 片段,则 BB 是连通图
        • BB 的任何两个顶点都有和 HH 的内部不相交的路相连接
        • 不计 HH 的顶点,任意两 HH - 片段没有公共顶点

        HH 是图 GG 的一个圈,图 GG 的两个 HH - 片段 B1B_1B2B_2冲突的,如果

        • B1B_1B2B_2HH 上有三个公共的附着顶点;或
        • HH 上存在四个顺序排列的顶点 v1,v2,v3,v4v_1,v_2,v_3,v_4 使得 v1,v3v_1,v_3B1B_1 的附着顶点并且 v2,v4v_2,v_4B2B_2 的附着顶点

        冲突的两个片段无法同时平面嵌入到同一个面中

        HH 是图 GG 的一个圈,以图 GGHH - 片段为顶点,顶点间连一条边当且仅当对应的 HH - 片段是冲突的,把这样得到的图称为 HH 的冲突图

        HH 是图 GG 的可平面子图,是 H~\tilde H 的一种平面嵌入。若 GG 也是可平面图,且存在 GG 的一个平面嵌入 H~\tilde H,使得:

        H~G~\tilde H\subseteq\tilde G

        则称 H~\tilde HGG 容许的。

        GG 是可平面的当且仅当对 GG 的每个圈 HH,其冲突图是二部图

        HH 是图 GG 的一个可平面子图,HH'HH 的一个平面嵌入。假定 BB 是子图 HH 的任一片段,若 BBHH 的所有附着顶点都位于 HH' 中某个面 ff 的边界上,则称 BBHH' 的面 ff 中是可画入的,否则,称为不可画入的。令:

        F(B,H)={ffH的面且Bf中可画入}F(B,H)=\{f|f\text{是}H'\text{的面且}B\text{在}f\text{中可画入}\}

        假定 HH' 如实线所示,则片段 B3B_3 在外部面上是可画入的,而 B2B_2 对面 f2f_2f3f_3 均为不可画入的,且 F(B1,H)={f1,f3}F(B_1,H)=\{f_1,f_3\}

        可画入示例

        # 算法流程

        如果 GG 非可平面,通过算法可以得到判定;如果 GG 是可平面图,通过算法,可以给出一种平面嵌入形式。

        GG 是至少有三个顶点的简单块,取 GG 的任意一个圈 HH,求出 HH 的一个平面嵌入。若无圈,则 GG 是一棵树,一定可平面。

        初始取 GG 的一个圈 H1H _1,并作平面嵌入 H~1\tilde H_1。假设 HiH_i 已确定,下面确定 Hi+1H_{i+1},算法流程如下:

        1. 判断 E(G)\E(Hi)=E(G)\backslash E(H_i)=\varnothing
          • 若是空集
            • 🖐则停,返回 GG 可平面。
          • 否则
            1. 确定 HiH_i 的所有片段 (桥),
            2. 对每个片段 (桥) BB,求集合 F(B,Hi)F(B,H_i)
        2. 判断是否存在片段 BB,使 F(B,H~i)=F(B,\tilde H_i)=\varnothing
          • 若存在
            • 🖐则停 ,返回 GG 不可平面。
          • 否则
            1. HiH_i 的所有片段中确定一个使 F(B,H~i)|F(B,\tilde H_i)| 最小的片段 BB
              • (⭐️这 2 步的选取具有随机性,算法的随机性来自于这一步)
            2. 并取 fF(B,H~i)f\in F(B,\tilde H_i)
            3. 在片段 BB 上取一条连接 HiH_i 上两个附着点的路 PiP_i
            4. PiP_i 画在 H~i\tilde H_i 的面 ff 内得到 Hi+1=HiPiH_{i+1}=H_i\cup P_i
        3. i=i+1i=i+1,转第 1
        \ No newline at end of file +平面图 - 图论 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        如果能把图 GG 画在平面上,使得除顶点外,边与边之间没有交叉,称 GG 可嵌入平面,或称 GG 是可平面图

        可平面图 GG 的边不交叉的一种画法,称为 GG 的一种平面嵌入GG 的平面嵌入表示的图称为平面图

        可平面图概念和平面图概念有时可以等同看待;

        图的平面性问题主要涉及如下几个方面:

        • 平面图的性质
        • 平面图的判定
        • 平面嵌入方法 (平面性算法)
        • 涉及图的平面性问题的拓扑不变量

        # 平面图的性质

        一个平面图 GG 把平面分成若干连通片,这些连通片称为 GG区域,或 GG 的一个GG 的面组成的集合用 Φ\Phi 表示。

        面积有限的区域称为平面图 GG内部面,否则称为 GG外部面

        GG 中,顶点和边都与某个给定区域关联的子图,称为该面的边界。某面 ff 的边界中含有的边数 (割边计算 2 次⚠️) 称为该面 ff次数,记 degf\deg f

        # 平面图的次数公式

        G=(n,m)G=(n,m) 是平面图,则:

        fΦdeg(f)=2m\sum_{f\in\Phi}\deg(f)=2m


        证明:

        GG 的任意一条边 ee, 如果 ee 是某面割边,那么由面的次数定义,该边给 GG 的总次数贡献 2 次;如果 ee 不是割边,那么,它必然是两个面的公共边,因此,由面的次数定义,它也给总次数贡献 2 次。

        # 连通平面图的欧拉公式

        G=(n,m)G=(n,m)连通平面图,φ\varphiGG 的面数,则:

        nm+φ=2n-m+\varphi=2


        证明:

        情形 1,如果 GG 是树。

        那么 m=n1m=n-1, 𝜑=1。在这种情况下,容易验证,定理中的恒等式是成立的。

        情形 2, GG 不是树的连通平面图。

        假设在这种情形下,欧拉恒等式不成立。则存在一个含有最少边数的连通平面图 GG,使得它不满足欧拉恒等式。设这个最少边数连通平面图 G=(n,m)G=(n,m),面数为 𝜑,则:

        nm+φ2n-m+\varphi\neq2

        因为 GG 不是树,所以存在非割边 ee。显然,GeG-e 是连通平面图,边数为 m1m-1,顶点数为 nn,面数为 𝜑-1。由最少性假设GeG-e 满足欧拉等式:

        n(m1)+(φ1)=2nm+φ=2\begin{aligned} n-(m-1)+(\varphi-1)&=2\\ n-m+\varphi&=2 \end{aligned}

        于是矛盾,假设不成立。

        # 非连通平面图的欧拉公式

        G=(n,m)G=(n,m)kk 个连通分支的平面图,φ\varphiGG 的面数,则:

        nm+φ=k+1n-m+\varphi=k+1


        证明:

        对第 i(1ik)i(1≤i≤k) 个分支来说,设顶点数为 nin_i,边数为 mim_i,面数为 𝜑_i,由欧拉公式:

        nimi+φi=2i=1k(nimi+φi)=2ki=1knii=1kmi+i=1kφi=2k\begin{aligned} n_i-m_i+\varphi_i&=2\\ \sum_{i=1}^{k}(n_i-m_i+\varphi_i)&=2k\\ \sum_{i=1}^{k}n_i-\sum_{i=1}^{k}m_i+\sum_{i=1}^{k}\varphi_i&=2k \end{aligned}

        同时:(扣除重复计算的 k1k-1 个外部面)

        i=1kni=n,i=1kmi=m,i=1kφi=φ+k1\sum_{i=1}^{k}n_i=n,\ \ \sum_{i=1}^{k}m_i=m,\ \ \sum_{i=1}^{k}\varphi_i=\varphi+k-1

        于是得到:

        nm+(φ+k1)=2knm+φ=k+1\begin{aligned} n-m+(\varphi+k-1)&=2k\\ n-m+\varphi&=k+1 \end{aligned}

        GG 是具有 nn 个点 mm 条边 𝜑 个面的连通平面图,如果对 GG 的每个面 ff,有 degfl3\deg f≥l≥3,则:

        mll2(n2)m\leq\frac{l}{l-2}(n-2)

        这是连通平面图的必要条件,不是充分条件!

        GG 是具有 nn 个点 mm 条边 𝜑 个面的连通平面图,有:

        m3n6m\leq3n-6

        GG 是具有 nn 个点 mm 条边 𝜑 个面的连通平面图,如果对 GG 的每个面 ff,有 degf=l\deg f=l,则:

        m(l2)=l(n2)m(l-2)=l(n-2)

        GG 是具有 nn 个点 mm 条边 𝜑 个面的连通平面图,有:

        δ5\delta\leq5

        该结论是证明 “5 色定理” 的出发点

        一个连通平面图是 22 连通的,当且仅当它的每个面的边界是圈。

        # 图的嵌入性问题

        # 曲面嵌入

        # 球面嵌入

        ⭐️ GG 可球面嵌入与 GG 可平面嵌入等价

        我们实际上对图的嵌入性问题也只考虑这两个。

        # 环面嵌入

        # 定向曲面嵌入

        # 3 维空间嵌入

        所有图均可嵌入 R3R^3 中。

        # 凸多面体与平面图

        一个多面体称为凸多面体,如果在体上任取两点,其连线均在体上。

        凸多面体的一维骨架:把一个凸多面体压缩在平面上,得到一个对应的平面图,该平面图称为该凸多面体的一维骨架。

        存在且只存在 5 种正多面体:它们是正四、六、八、十二、二十面体

        # 平面图的判定

        # 同胚

        • 在图 GG 的边上插入一个新的 22 度顶点,使一条边分成两条边,则称将图 GG 在 2 度顶点内扩充
        • 去掉图 的一个 22 度顶点,使这个 22 度顶点关联的两条边合成一条边,则称将 GG 在 2 度顶点内收缩

        两个图 G1G_1G2G_2,如果 G1G2G_1\cong G_2,或者通过反复在 22 度顶点内扩充和收缩它们能变成同构的,则称 G2G_2G2G_2同胚的,或 G1G_1G2G_2 22 度顶点内是同构的

        ⭐️图的可平面性在同胚意义下不变。

        # 库拉托夫斯基定理

        GG 是可平面的当且仅当它不含与 K5K_5K3,3K_{3,3} 同胚的子图

        库拉托夫斯基定理可以等价叙述为:「图 GG 是非可平面的当且仅当它含有与 K5K_5K3,3K_{3,3} 同胚的子图」

        给定图 GG,去掉 GG 中的环 (若有的话),将 GG 中的重边 (若有的话) 用单边代替,称这样所得的图为 GG基础简单图

        • GG 是可平面图当且仅当其基础简单图是可平面图
        • GG 是可平面图当且仅当它的每个块是可平面图

        # 初等收缩

        uvuv 是简单图 GG 的一条边。去掉该边,重合其端点,再删去由此产生的环和重边。这一过程称为图 GG初等收缩收缩边 uvuv 的运算,并记所得之图为 G/uvG/uv

        一个图 GG 可收缩到 HH,是指 HH 可从 GG 通过一系列初等收缩而得到。

        # 瓦格纳定理

        简单图 GG 是可平面图当且仅当它不含可收缩到 K5K_5K3,3K_{3,3} 的子图。

        该定理和库拉托夫斯基定理是等价的。

        至少有 9 个点的简单可平面图的补图是不可平面的,而 9 是这种数目中最小的一个

        # 平面性的拓扑不变量

        # 亏格

        # 环柄

        在边交叉处建立的 “立交桥”,通过环柄使得一条边经过 “桥下”,另一条边经过 “桥上”,使二者分开。

        # 亏格定义

        若通过加上 kk 个环柄可将图 GG 嵌入球面,则 kk 得最小值称为 GG亏格,记为:

        γ(G)\gamma(G)

        • GG 是可平面图,则 γ(G)=0\gamma(G)=0
        • 同胚的图是亏格相等;
        • γ(K5)=1\gamma(K_5)=1

        # 引入亏格的欧拉公式

        对一个亏格为 γ\gamma,有 nn 个顶点,mm 条边和 φ\varphi 个面的多面体或连通图,有:

        nm+φ=22γn-m+\varphi=2-2\gamma

        # 结论

        GG 是一个有 nn 个顶点,mm 条边和 φ\varphi 个面的多面体或连通图,有:

        • γm6n22\gamma\geq\frac{m}{6}-\frac{n-2}{2}
        • GG 无三角形,则 γm4n22\gamma\geq\frac{m}{4}-\frac{n-2}{2}
        • GG 每个面是三角形,则 m=3(n2+2γ)m=3(n-2+2\gamma)
        • GG 每个面是四边形,则 m=2(n2+2γ)m=2(n-2+2\gamma)

        # 厚度

        若图 GGkk 个可平面子图的并等于 ,则称 kk 的最小值为 GG厚度,记为

        GG 是可平面图当且仅当 θ(G)=1\theta(G)=1

        # 叉数

        GG 画在平面上时相交的边对的最少数目称为 GG叉数,记为:

        η(G)\eta(G)

        GG 是可平面图当且仅当 η(G)=0\eta(G)=0

        # 糙度

        GG 中边不重的不可平面子图的最多数目称为 GG糙度,记为:

        ζ(G)\zeta(G)

        # 平面性算法

        HH 是图 GG 的一个子图,在 E(G)\E(H)E(G)\backslash E(H) 上定义关系 \sim 如下:e1e2e_1\sim e_2 当且仅当存在一条途径 WW,使得:

        • WW 的第一条边与最后一条边分别是 e1e_1e2e_2
        • WW 的内部顶点与 HH 不相交

        易证关系 \sim 具有自反性,对称性和传递性,从而是 E(G)\E(H)E(G)\backslash E(H) 上的一个等价关系

        此等价关系的等价类导出的 GE(H)G-E(H) 的边导出子图称为 HH - 片段,或者 GG 关于 HH 的桥。片段或桥与 HH 的公共顶点称为附着顶点

        • BBHH - 片段,则 BB 是连通图
        • BB 的任何两个顶点都有和 HH 的内部不相交的路相连接
        • 不计 HH 的顶点,任意两 HH - 片段没有公共顶点

        HH 是图 GG 的一个圈,图 GG 的两个 HH - 片段 B1B_1B2B_2冲突的,如果

        • B1B_1B2B_2HH 上有三个公共的附着顶点;或
        • HH 上存在四个顺序排列的顶点 v1,v2,v3,v4v_1,v_2,v_3,v_4 使得 v1,v3v_1,v_3B1B_1 的附着顶点并且 v2,v4v_2,v_4B2B_2 的附着顶点

        冲突的两个片段无法同时平面嵌入到同一个面中

        HH 是图 GG 的一个圈,以图 GGHH - 片段为顶点,顶点间连一条边当且仅当对应的 HH - 片段是冲突的,把这样得到的图称为 HH 的冲突图

        HH 是图 GG 的可平面子图,是 H~\tilde H 的一种平面嵌入。若 GG 也是可平面图,且存在 GG 的一个平面嵌入 H~\tilde H,使得:

        H~G~\tilde H\subseteq\tilde G

        则称 H~\tilde HGG 容许的。

        GG 是可平面的当且仅当对 GG 的每个圈 HH,其冲突图是二部图

        HH 是图 GG 的一个可平面子图,HH'HH 的一个平面嵌入。假定 BB 是子图 HH 的任一片段,若 BBHH 的所有附着顶点都位于 HH' 中某个面 ff 的边界上,则称 BBHH' 的面 ff 中是可画入的,否则,称为不可画入的。令:

        F(B,H)={ffH的面且Bf中可画入}F(B,H)=\{f|f\text{是}H'\text{的面且}B\text{在}f\text{中可画入}\}

        假定 HH' 如实线所示,则片段 B3B_3 在外部面上是可画入的,而 B2B_2 对面 f2f_2f3f_3 均为不可画入的,且 F(B1,H)={f1,f3}F(B_1,H)=\{f_1,f_3\}

        可画入示例

        # 算法流程

        如果 GG 非可平面,通过算法可以得到判定;如果 GG 是可平面图,通过算法,可以给出一种平面嵌入形式。

        GG 是至少有三个顶点的简单块,取 GG 的任意一个圈 HH,求出 HH 的一个平面嵌入。若无圈,则 GG 是一棵树,一定可平面。

        初始取 GG 的一个圈 H1H _1,并作平面嵌入 H~1\tilde H_1。假设 HiH_i 已确定,下面确定 Hi+1H_{i+1},算法流程如下:

        1. 判断 E(G)\E(Hi)=E(G)\backslash E(H_i)=\varnothing
          • 若是空集
            • 🖐则停,返回 GG 可平面。
          • 否则
            1. 确定 HiH_i 的所有片段 (桥),
            2. 对每个片段 (桥) BB,求集合 F(B,Hi)F(B,H_i)
        2. 判断是否存在片段 BB,使 F(B,H~i)=F(B,\tilde H_i)=\varnothing
          • 若存在
            • 🖐则停 ,返回 GG 不可平面。
          • 否则
            1. HiH_i 的所有片段中确定一个使 F(B,H~i)|F(B,\tilde H_i)| 最小的片段 BB
              • (⭐️这 2 步的选取具有随机性,算法的随机性来自于这一步)
            2. 并取 fF(B,H~i)f\in F(B,\tilde H_i)
            3. 在片段 BB 上取一条连接 HiH_i 上两个附着点的路 PiP_i
            4. PiP_i 画在 H~i\tilde H_i 的面 ff 内得到 Hi+1=HiPiH_{i+1}=H_i\cup P_i
        3. i=i+1i=i+1,转第 1
        \ No newline at end of file diff --git a/math/discrete-math/graph/shortest-path-algorithms/index.html b/math/discrete-math/graph/shortest-path-algorithms/index.html index 7cb1d267..3debf29e 100644 --- a/math/discrete-math/graph/shortest-path-algorithms/index.html +++ b/math/discrete-math/graph/shortest-path-algorithms/index.html @@ -1 +1 @@ -最短路径算法 - 图论 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        This article ism't finished yet...

        # 顶点标号法

        Dantzig 算法,1959 年,旦捷希 (Dantzig) 发现了在赋权图中求由点 a 到点 b 的最短路好算法,称为顶点标号法。

        和弗洛伊德算法非常非常像... ,在这里补充一下 Floyid 算法对邻接矩阵求最短距离矩阵过程,这个过程和 Dantzig 几乎一致(完全一致好吧,考试可能会考?):

        • 按顺序一列一列看,看第 ii 列时:
          • 按顺序一行一行看,看第 jj 行时,若 ajia_{ji}\neq\infin 则:
            • 看第 jj 列,看 aji+akja_{ji}+a_{kj} 是否是新的 iikk 的最短距离

        # 数据结构

        • t(an)t(a_n) : 点 ana_n 的标号值,表示点 a1=aa_1=aana_n 的最短路长度
        • Ai={a1,a2,,ai}A_i =\{a_1,a_2,\cdots,a_i\} : 已经标号的顶点集合
        • TiT_i : aaaia_i 的最短路上的边集合

        # 算法步骤

        1. 记:

          • a=a1a=a_1 ,
          • t(a1)=0t(a_1)=0 ,
          • A1={a1}A_1=\{a_1\} ,
          • T=T=\varnothing ;
        2. 若已经得到 Ai={a,a2,,ai}A_i=\{a,a_2,\cdots,a_i\} , 且对于 1ni1≤n\leq i , 已知 t(an)t(a_n)每一个 anAia_n\in A_i , 求一点:

          bn(i)N(an)Ai=Bn(i)b_n^{(i)}\in N(a_n)-A_i=B_n^{(i)}

          使得:

          l(anbn(i))=minvBn(i)l(anv)l(a_n b_n^{(i)})=\min_{v\in B_n^{(i)}}l(a_n v)

          这里 ii 表示第 ii 轮,N(an)N(a_n) 表示与 ana_n 邻接的点组成的集合,N 表示 Neighbour

        3. 设有 jj , 1ji1≤j\leq i , 而 bj(i)b_{j}^{(i)} 是使 t(aj)+l(ajbj(i))t(a_{j})+l(a_{j}b_{j}^{(i)}) 取最小值,令:

          • ai+1=bj(i)a_{i+1}=b_{j}^{(i)}
          • t(ai+1)=t(aj)+l(ajai+1)t(a_{i+1})= t(a_{j})+l(a_{j}a_{i+1})
          • Ti+1=Ti{ajai+1}T_{i+1}=T_i\cup\{a_{j}a_{i+1}\}
          • ai+1=ba_{i+1}=b , 停止;
          • 否则,记 Ai+1=Ai{ai+1}A_{i+1}=A_i\cup\{a_{i+1}\}, 转 (2)

        # 时间复杂度

        对第 ii 次循环:

        • 步骤 (2) 要进行 ii 次比较运算
        • 步骤 (3) 要进行 ii 次加法与 ii 次比较运算

        所以,该次循环运算量为 3i3i

        所以,在最坏的情况下,运算量为 n2n^2 级,是好算法

        # 完备性证明

        定理 1:算法中的函数 t(an)t(a_n) 给出了 aaaia_i 的距离
        证明:对 ii 作数学归纳法

        1. i=1i=1 时结论显然成立;
        2. 设对所有的 jj1ji1\leq j\leq i 时,t(aj)=d(a,aj)t(a_j)= d(a,a_j)
        3. 考虑 j=i+1j=i+1

        顶点标记法完备性证明

        于是 d=d(a,ai+1)d=d(a,a_{i+1})

        又令 vnv_n 是 P 中第一个不在 AiA_{i} 中的点。由于 ai+1Aia_{i+1} \notin A_{i} , 所以这样的点存在

        因为 v0Aiv_0\in A_{i} ,所以 n1n\geq 1

        记 P 中 aavn+1v_{n+1} 一段长为 ll , 而 aavnv_{n} 的一段长为 l1l_1

        由归纳假设有: l1t(vn)l_1\geq t(v_{n}) ,进而有:

        d=d(a,ai+1)l=l1+l(vnvn+1)t(vn)+l(vnvn+1)d=d(a,a_{i+1})\geq l=l_1+l(v_{n}v_{n+1})\geq t(v_{n})+l(v_{n}v_{n+1})

        算法中,当第 ii 轮中已知 Ai={a1,a2,,ai}A_{i}=\{a_1,a_2,\cdots,a_{i}\} 要给 ai+1a_{i+1} 标号时,其中要选择 bn(i)b_{n}^{(i)} ,满足:

        l(vnbn(i))l(vnvn+1)l(v_{n}b_{n}^{(i)})\leq l(v_{n}v_{n+1})

        だから:

        dt(vn)+l(vnvn+1)t(vn)+l(vnbn(i))d\geq t(v_{n})+l(v_{n}v_{n+1})\geq t(v_{n})+l(v_{n}b_{n}^{(i)})

        又由算法最终对点 aia_i 的标号值的选择方法知;

        dt(vn)+l(vnbn(i))t(aj)+l(ajai+1)t(ai+1)d\geq t(v_{n})+l(v_{n}b_{n}^{(i)})\geq t(a_{j})+l(a_{j}a_{i+1})\geq t(a_{i+1})

        另一方面,由算法可知存在一条长度为 t(ai)t(a_i) 的联结 aaaia_i 的路,所以:

        t(ai+1)d(a,ai+1)t(a_{i+1})\geq d(a,a_{i+1})

        从而得到 t(ai+1)d(a,ai+1)t(ai+1)t(a_{i+1})\geq d(a,a_{i+1})\geq t(a_{i+1}) ,即得到:t(ai+1)=d(a,ai+1)t(a_{i+1})=d(a,a_{i+1})

        # 应用举例

        # 分酒

        某两人有一只 8 升的酒壶装满了酒,还有两只空酒壶,分别为 5 升和 3 升,求最少的操作次数能均分酒

        解:设 x1,x2,x3x_1,x_2,x_3 分别表示 8,5,3 升酒壶中的酒量。则

        x1+x2+x3=8,x18,x25,x33x_1+x_2+x_3=8,\ \ \ x_1≤8,x_2≤5,x_3≤3

        容易算出 (x1,x2,x3)(x_1,x_2,x_3) 的组合形式共 24 种

        每种组合用一个点表示,两点连线,当且仅当可通过倒酒的方式相互变换。若各边赋权为 1,则问题转化为在该图中求 (8,0,0)(8,0,0)(4,4,0)(4,4,0) 的一条最短路。结果如下:

        (8,0,0)(3,5,0)(3,2,3)(6,2,0)(6,0,2)(1,5,2)(1,4,3)(4,4,0)(8,0,0)\to(3,5,0)\to(3,2,3)\to(6,2,0)\to(6,0,2)\to(1,5,2)\to(1,4,3)\to(4,4,0)

        这个问题给定了初始状态和目标状态,也可以转化为人工智能中的智能规划问题

        # 狼羊过河

        在一河岸有狼,羊和卷心菜。摆渡人要将它们渡过河去,由于船太小,每次只能载一样东西。由于狼羊,羊卷心菜不能单独相处。问摆渡人至少要多少次才能将其渡过河?

        人,狼,羊,菜所有组合形式为:

        C40+C41+C42+C43+C44=24=16C_4^0+C_4^1+C_4^2+C_4^3+C_4^4=2^4=16

        但是以下组合不能允许出现:狼羊菜,羊菜,狼羊,人,人狼,人菜,共 6 种;

        岸上只能允许出现 10 种组合:人狼羊菜,人狼羊,人狼菜,人羊,空,菜,羊,狼,狼菜,人羊菜;

        每种情况用点表示,两点连线,当且仅当两种情况可用载人 (或加一物) 的渡船相互转变,每条边赋权为 1 。于是,问题转化为求由顶点 “人狼羊菜” 到顶点 “空” 的一条最短路问题。结果为:

        人狼羊菜狼菜人狼菜人狼羊人羊\text{人狼羊菜}\to\text{狼菜}\to\text{人狼菜}\to\text{狼}\to\text{人狼羊}\to\text{羊}\to\text{人羊}\to\text{空}

        \ No newline at end of file +最短路径算法 - 图论 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        This article ism't finished yet...

        # 顶点标号法

        Dantzig 算法,1959 年,旦捷希 (Dantzig) 发现了在赋权图中求由点 a 到点 b 的最短路好算法,称为顶点标号法。

        和弗洛伊德算法非常非常像... ,在这里补充一下 Floyid 算法对邻接矩阵求最短距离矩阵过程,这个过程和 Dantzig 几乎一致(完全一致好吧,考试可能会考?):

        • 按顺序一列一列看,看第 ii 列时:
          • 按顺序一行一行看,看第 jj 行时,若 ajia_{ji}\neq\infin 则:
            • 看第 jj 列,看 aji+akja_{ji}+a_{kj} 是否是新的 iikk 的最短距离

        # 数据结构

        • t(an)t(a_n) : 点 ana_n 的标号值,表示点 a1=aa_1=aana_n 的最短路长度
        • Ai={a1,a2,,ai}A_i =\{a_1,a_2,\cdots,a_i\} : 已经标号的顶点集合
        • TiT_i : aaaia_i 的最短路上的边集合

        # 算法步骤

        1. 记:

          • a=a1a=a_1 ,
          • t(a1)=0t(a_1)=0 ,
          • A1={a1}A_1=\{a_1\} ,
          • T=T=\varnothing ;
        2. 若已经得到 Ai={a,a2,,ai}A_i=\{a,a_2,\cdots,a_i\} , 且对于 1ni1≤n\leq i , 已知 t(an)t(a_n)每一个 anAia_n\in A_i , 求一点:

          bn(i)N(an)Ai=Bn(i)b_n^{(i)}\in N(a_n)-A_i=B_n^{(i)}

          使得:

          l(anbn(i))=minvBn(i)l(anv)l(a_n b_n^{(i)})=\min_{v\in B_n^{(i)}}l(a_n v)

          这里 ii 表示第 ii 轮,N(an)N(a_n) 表示与 ana_n 邻接的点组成的集合,N 表示 Neighbour

        3. 设有 jj , 1ji1≤j\leq i , 而 bj(i)b_{j}^{(i)} 是使 t(aj)+l(ajbj(i))t(a_{j})+l(a_{j}b_{j}^{(i)}) 取最小值,令:

          • ai+1=bj(i)a_{i+1}=b_{j}^{(i)}
          • t(ai+1)=t(aj)+l(ajai+1)t(a_{i+1})= t(a_{j})+l(a_{j}a_{i+1})
          • Ti+1=Ti{ajai+1}T_{i+1}=T_i\cup\{a_{j}a_{i+1}\}
          • ai+1=ba_{i+1}=b , 停止;
          • 否则,记 Ai+1=Ai{ai+1}A_{i+1}=A_i\cup\{a_{i+1}\}, 转 (2)

        # 时间复杂度

        对第 ii 次循环:

        • 步骤 (2) 要进行 ii 次比较运算
        • 步骤 (3) 要进行 ii 次加法与 ii 次比较运算

        所以,该次循环运算量为 3i3i

        所以,在最坏的情况下,运算量为 n2n^2 级,是好算法

        # 完备性证明

        定理 1:算法中的函数 t(an)t(a_n) 给出了 aaaia_i 的距离
        证明:对 ii 作数学归纳法

        1. i=1i=1 时结论显然成立;
        2. 设对所有的 jj1ji1\leq j\leq i 时,t(aj)=d(a,aj)t(a_j)= d(a,a_j)
        3. 考虑 j=i+1j=i+1

        顶点标记法完备性证明

        于是 d=d(a,ai+1)d=d(a,a_{i+1})

        又令 vnv_n 是 P 中第一个不在 AiA_{i} 中的点。由于 ai+1Aia_{i+1} \notin A_{i} , 所以这样的点存在

        因为 v0Aiv_0\in A_{i} ,所以 n1n\geq 1

        记 P 中 aavn+1v_{n+1} 一段长为 ll , 而 aavnv_{n} 的一段长为 l1l_1

        由归纳假设有: l1t(vn)l_1\geq t(v_{n}) ,进而有:

        d=d(a,ai+1)l=l1+l(vnvn+1)t(vn)+l(vnvn+1)d=d(a,a_{i+1})\geq l=l_1+l(v_{n}v_{n+1})\geq t(v_{n})+l(v_{n}v_{n+1})

        算法中,当第 ii 轮中已知 Ai={a1,a2,,ai}A_{i}=\{a_1,a_2,\cdots,a_{i}\} 要给 ai+1a_{i+1} 标号时,其中要选择 bn(i)b_{n}^{(i)} ,满足:

        l(vnbn(i))l(vnvn+1)l(v_{n}b_{n}^{(i)})\leq l(v_{n}v_{n+1})

        だから:

        dt(vn)+l(vnvn+1)t(vn)+l(vnbn(i))d\geq t(v_{n})+l(v_{n}v_{n+1})\geq t(v_{n})+l(v_{n}b_{n}^{(i)})

        又由算法最终对点 aia_i 的标号值的选择方法知;

        dt(vn)+l(vnbn(i))t(aj)+l(ajai+1)t(ai+1)d\geq t(v_{n})+l(v_{n}b_{n}^{(i)})\geq t(a_{j})+l(a_{j}a_{i+1})\geq t(a_{i+1})

        另一方面,由算法可知存在一条长度为 t(ai)t(a_i) 的联结 aaaia_i 的路,所以:

        t(ai+1)d(a,ai+1)t(a_{i+1})\geq d(a,a_{i+1})

        从而得到 t(ai+1)d(a,ai+1)t(ai+1)t(a_{i+1})\geq d(a,a_{i+1})\geq t(a_{i+1}) ,即得到:t(ai+1)=d(a,ai+1)t(a_{i+1})=d(a,a_{i+1})

        # 应用举例

        # 分酒

        某两人有一只 8 升的酒壶装满了酒,还有两只空酒壶,分别为 5 升和 3 升,求最少的操作次数能均分酒

        解:设 x1,x2,x3x_1,x_2,x_3 分别表示 8,5,3 升酒壶中的酒量。则

        x1+x2+x3=8,x18,x25,x33x_1+x_2+x_3=8,\ \ \ x_1≤8,x_2≤5,x_3≤3

        容易算出 (x1,x2,x3)(x_1,x_2,x_3) 的组合形式共 24 种

        每种组合用一个点表示,两点连线,当且仅当可通过倒酒的方式相互变换。若各边赋权为 1,则问题转化为在该图中求 (8,0,0)(8,0,0)(4,4,0)(4,4,0) 的一条最短路。结果如下:

        (8,0,0)(3,5,0)(3,2,3)(6,2,0)(6,0,2)(1,5,2)(1,4,3)(4,4,0)(8,0,0)\to(3,5,0)\to(3,2,3)\to(6,2,0)\to(6,0,2)\to(1,5,2)\to(1,4,3)\to(4,4,0)

        这个问题给定了初始状态和目标状态,也可以转化为人工智能中的智能规划问题

        # 狼羊过河

        在一河岸有狼,羊和卷心菜。摆渡人要将它们渡过河去,由于船太小,每次只能载一样东西。由于狼羊,羊卷心菜不能单独相处。问摆渡人至少要多少次才能将其渡过河?

        人,狼,羊,菜所有组合形式为:

        C40+C41+C42+C43+C44=24=16C_4^0+C_4^1+C_4^2+C_4^3+C_4^4=2^4=16

        但是以下组合不能允许出现:狼羊菜,羊菜,狼羊,人,人狼,人菜,共 6 种;

        岸上只能允许出现 10 种组合:人狼羊菜,人狼羊,人狼菜,人羊,空,菜,羊,狼,狼菜,人羊菜;

        每种情况用点表示,两点连线,当且仅当两种情况可用载人 (或加一物) 的渡船相互转变,每条边赋权为 1 。于是,问题转化为求由顶点 “人狼羊菜” 到顶点 “空” 的一条最短路问题。结果为:

        人狼羊菜狼菜人狼菜人狼羊人羊\text{人狼羊菜}\to\text{狼菜}\to\text{人狼菜}\to\text{狼}\to\text{人狼羊}\to\text{羊}\to\text{人羊}\to\text{空}

        \ No newline at end of file diff --git a/math/discrete-math/graph/spectral-clustering/index.html b/math/discrete-math/graph/spectral-clustering/index.html index 99f278bd..47cbbcb6 100644 --- a/math/discrete-math/graph/spectral-clustering/index.html +++ b/math/discrete-math/graph/spectral-clustering/index.html @@ -1,4 +1,4 @@ -Spectral Clustering (谱聚类) - 图论 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        This article isn't finished yet...

        Graph Partitioning

        Bi-partitioning task

        # Minmal-cut

        # Normalized-cut

        Criterion: Normalized-cut [Shi-Malik, '97]

        Define: Connectivity between groups relative to the density of each group

        ncut(A,B)=cut(A,B)vol(A)+cut(A,B)vol(B)ncut(\mathbf{A},\mathbf{B})=\frac{cut(\mathbf{A},\mathbf{B})}{vol(\mathbf{A})}+\frac{cut(\mathbf{A},\mathbf{B})}{vol(\mathbf{B})}

        vol(A)vol(\mathbf{A}) : total weight of the edges with at least one endpoint in A\mathbf{A} :

        vol(A)=iAkivol(\mathbf{A})=\sum_{i\in\mathbf{A}}k_i

        The purpose of this criterion is to produce more balanced partitions.

        Spectral Graph Theory:

        Analyze the "spectrum" of matrix(s) representing GG

        Spectrum: Eigenvectors xix_i of a graph, ordered by the magnitude(strength) of their corresponding eigenvalues λi\lambda_i :

        Λ={λ1,λ2,,λn}λ1λ2λn\Lambda=\{\lambda_1,\lambda_2,\cdots,\lambda_n\}\\ \lambda_1\leq\lambda_2\leq\cdots\leq\lambda_n

        dd-regular : the degree of each vertex in the graph is dd

        If GG is not connected, for example, GG has 2 components, each dd-regular.

        每一个特征向量表现了图的划分方式

        What are some eigenvectors?

        # Spectral Partitioning Algorithm

        # 算法流程

        # Pre-processing

        根据图构建拉普拉斯矩阵 LL

        # Decomposition

        * 分解

        LL 的第 2 小的特征值 λ1\lambda_1 (λ0=0\lambda_0=0) 和特征向量 q1\vec{q_1}

        This article isn't finished yet...

        Graph Partitioning

        Bi-partitioning task

        # Minmal-cut

        # Normalized-cut

        Criterion: Normalized-cut [Shi-Malik, '97]

        Define: Connectivity between groups relative to the density of each group

        ncut(A,B)=cut(A,B)vol(A)+cut(A,B)vol(B)ncut(\mathbf{A},\mathbf{B})=\frac{cut(\mathbf{A},\mathbf{B})}{vol(\mathbf{A})}+\frac{cut(\mathbf{A},\mathbf{B})}{vol(\mathbf{B})}

        vol(A)vol(\mathbf{A}) : total weight of the edges with at least one endpoint in A\mathbf{A} :

        vol(A)=iAkivol(\mathbf{A})=\sum_{i\in\mathbf{A}}k_i

        The purpose of this criterion is to produce more balanced partitions.

        Spectral Graph Theory:

        Analyze the "spectrum" of matrix(s) representing GG

        Spectrum: Eigenvectors xix_i of a graph, ordered by the magnitude(strength) of their corresponding eigenvalues λi\lambda_i :

        Λ={λ1,λ2,,λn}λ1λ2λn\Lambda=\{\lambda_1,\lambda_2,\cdots,\lambda_n\}\\ \lambda_1\leq\lambda_2\leq\cdots\leq\lambda_n

        dd-regular : the degree of each vertex in the graph is dd

        If GG is not connected, for example, GG has 2 components, each dd-regular.

        每一个特征向量表现了图的划分方式

        What are some eigenvectors?

        # Spectral Partitioning Algorithm

        # 算法流程

        # Pre-processing

        根据图构建拉普拉斯矩阵 LL

        # Decomposition

        * 分解

        LL 的第 2 小的特征值 λ1\lambda_1 (λ0=0\lambda_0=0) 和特征向量 q1\vec{q_1}nn 个参数对应到 nn 个顶点,可以画图,将参数作为纵坐标,顶点标号作为横坐标,结合图像进行分组 (grouping)

        \ No newline at end of file +c-16-25.333-24-45-24-59z"/>nn 个参数对应到 nn 个顶点,可以画图,将参数作为纵坐标,顶点标号作为横坐标,结合图像进行分组 (grouping)

        \ No newline at end of file diff --git a/math/discrete-math/graph/tree/index.html b/math/discrete-math/graph/tree/index.html index 72d83530..cbdda303 100644 --- a/math/discrete-math/graph/tree/index.html +++ b/math/discrete-math/graph/tree/index.html @@ -1 +1 @@ -树 - 图论 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 树的基本性质

        # 树的叶子数

        TT(n,m)(n,m) 树,TT 中有 nin_i 个度为 i,(1ik)i,(1\leq i\leq k) 的点,且有 ni=n\sum n_i=n ,则有:n1=2+n3+2n4++(k2)nkn_1=2+n_3+2n_4+\cdots+(k-2)n_k

        证明:由 m=n1m=n-1 得:

        m=(n1+n2++np)1m=(n_1+n_2+\cdots+n_p)-1

        又由握手定理得:

        2m=n+2n2++knk2m=n+2n_2+\cdots+kn_k

        由上面两等式得:

        n1=2+n3+2n4++(k2)nkn_1=2+n_3+2n_4+\cdots+(k-2)n_k

        # 森林的分支数

        具有 kk 个分支的森林有 nkn-k 条边。


        证明:

        设森林 GGkk 个分支为 Ti(1ik)T_i(1≤i\leq k) ,对每个分支,使用定理 3 得

        m(Ti)=ni1(ni=V(Ti))m(T_i)=n_i-1(n_i=|V(T_i)|)

        所以:

        m(G)=i=1km(Ti)=nkm(G)=\sum_{i=1}^{k}m(T_i)=n-k

        # 图的边数下限

        每个 nn 阶连通图的边数至少为 n1n-1.

        证明:

        (1) 如果 nn 阶连通图 GG 没有一度顶点,那么由握手定理有:

        m(G)=12vV(G)d(v)nm(G)=\frac{1}{2}\sum_{v\in V(G)}d(v)\geq n


        (2) 如果 GG 有一度顶点:

        对顶点数作数学归纳。

        n=1n=1 时,结论显然

        设当 n=kn=k 时,结论成立。

        n=k+1n=k+1 时,设 uuGG 的一度顶点,则 GuG-u 为具有 kk 个顶点的连通图

        (2.1) 若 GuG-u 有一度顶点,则由归纳假设,其边数至少 k1k-1 ,于是 GG 的边数至少有 kk 条;

        (2.2) 如果 GuG-u 没有一度顶点,由握手定理有:

        m(Gu)=12vV(Gu)d(v)km(G-u)=\frac{1}{2}\sum_{v\in V(G-u)}d(v)\geq k

        所以,GG 至少有 k+1k+1 条边。

        而当 GG 是树时,边数恰为 n1n-1 .

        所以 n 阶连通图 G 至少有 n-1 条边。

        所以,树也被称为最小连通图。

        任意树 TT 的两个不邻接顶点之间添加一条边后,可以得到唯一圈


        证明:

        uuvv 是树 TT 的任意两个不邻接顶点,由定理 2 知:有唯一路 PP 连接 uuvv ;于是 P{uv}P\cup\{uv\} 是一圈。

        显然,由 PP 的唯一性也就决定了 P{uv}P\cup\{uv\} 的唯一性。

        # 树的叶子数下限

        GG 是树且 Δk\Delta≥k ,则 GG 至少有 kk 个一度顶点(叶子顶点)


        证明:反证法,假设 GG 有至多 k1k-1 个叶子顶点

        2m(G)=vV(G)d(v)k1+k+2(nk)=2n1>2n22m(G)=\sum_{v\in V(G)}d(v)\geq k-1+k+2(n-k)=2n-1\gt 2n-2

        所以 m(G)>n1m(G)\gt n-1 ,与 GG 是树矛盾!

        # 森林的路分解

        GG 是森林且恰有 2k2k奇度顶点,则在 GG 中有 kk 条边不重合的路 P1,P2,,PkP_1,P_2,\cdots,P_k 使得:

        E(G)=E(P1)E(P2)E(Pk)E(G)=E(P_1)\cup E(P_2)\cup\cdots\cup E(P_k)

        证明:对 kk 作数学归纳。

        k=1k=1 时,GG 只有两个奇数度顶点,此时,容易证明,GG 是一条路;

        设当 k=tk=t 时,结论成立。令 k=t+1k=t+1

        GG 中一个分支中取两个一度项点 uuvv ,令 PP 是连接该两个项点的唯一路,则 GPG-P 是有 2t2t 个奇数顶点的森林,由归纳假设,它可以分解 tt 条边不重合的路之并,所以 GG 可以分解为 t+1t+1 条边不重合的路之并。

        对图作某种形式的分解,是图论的一个研究对象,它在网络结构分析中具有重要作用。

        # 树的同构子图

        TTkk 阶树。若图 GG 满足 δk1\delta≥k-1 ,则 TT 同构于 GG 的某个子图。

        证明:对 kk 作数学归纳

        k=1k=1 时,结论显然。

        假设对 k1,(k3)k-1,(k ≥3) 的每颗树 TT ,以及最小度至少为 k2k-2 的每个图 HHT1T_1 同构于 HH 的某个子图 FF

        现在设 TTkk 阶树,且图 GG 满足 δk1\delta≥k-1 ,证明 TT 同构于 GG 的某个子图:

        uuTT 的树叶,vvuu 的邻接顶点。则 TuT-uk1k-1 阶树。

        由于 δ(G)k1>k2\delta(G)≥k-1>k-2 ,由归纳假设,TuT-u 同构于 GG 的某个子图 FF

        树的同构子图证明

        vv 是与 TTvv 相对应的 FF 中的点,由于 dG(v1)k1d_{G(v_1)}\geq k-1 ,所以 vvGG 中一定有相异于 FF 中的邻点 ww ,作 F{u1w}F\cup\{u_1w\} ,则该子图和 TT 同构。

        # 树的度序列问题

        在第一章中,介绍了判定一个非增非负序列是否为简单图的度序列定理。下面介绍一个判定非增非负序列是否为树的度序列的简单方法。

        S={d1,d2,,dn}S=\{d_1,d_2,\cdots,d_n\}nn 个正整数序列,它们满足:

        • d1d2dnd_1\geq d_2\geq\cdots\geq d_n
        • di=2(n1)\sum di=2(n-1)

        则存在一颗树 TT ,其度序列为 SS


        证明:对 nn 作数学归纳。

        n=1,2n=1,2 时,结论显然。

        假设对 n=kn=k 时结论成立。设 n=k+1n=k+1,

        首先,序列中至少一个数为 1,否则,序列和大于 2k2k ,与条件相矛盾!

        所以,dk+1=1d_{k+1}=1

        我们从序列中删掉 d1d_1dk+1d_{k+1} ,增加数 d=di1d^*=d_i-1 放在它应该在的位置。得到序列 SS ,该序列含 kk 个数,序列和为 2(k1)2(k-1)

        由归纳假设,存在树 TT ,它的度序列为 SiS_i

        现在,增加结点 vv ,把它和 T1T_1 中点 dd^* 相连得到树 TT 为所求

        # 树的中心与形心

        # 中心

        1. 图的顶点的离心率 e(v)=max{d(u,v)uV(G)}e(v)=\max\{d(u,v)|u\in V(G)\}
          • 图的直径最大离心率
        2. 图的半径 r(G)=min{e(v)vV(G)}r(G)=\min\{e(v)|v\in V(G)\}
        3. 图的中心点离心率等于半径的点
        4. 图的中心:全体中心点的集合

        对树 TT 的阶数 nn 作归纳证明。

        n=1,2n=1,2 时,结论显然成立

        设对 n<k,(k3)n<k,(k≥3) 的树结论成立,设 TTkk 阶树

        容易知道,删掉 TT 的所有叶,得到的树 TT ,的每个点的离心率比它们在 TT 中离心率减少 1

        又因 TT 的叶不能是中心点,所以 TT 的中心点在 T1T_1

        这样,若点 uu 的离心率在 TT 中最小,则在 T1T_1 中依然最小,即说明 TT 的中心点是 T1T_1 的中心点,反之亦然。

        # 形心

        uu 是树 TT 的任意一个顶点,树 TT 在顶点 uu 的分支是指包含 uu 作为一个叶点的极大子树,其分支数为顶点 uu 的度数;

        1. uu:树 TTuu 点的分支中边的最大数目称为;
        2. TT形心点:树 TT权值最小的点
        3. TT形心:全体形心点的集合

        每一棵树有一个由一个点两个邻接的点组成的形心。

        # 生成树

        # 生成树的性质

        每个连通图至少包含一棵生成树


        证明:

        如果连通图 GG 是树,则其本身是一棵生成树;

        若连通图 GG 中有圈 CC ,则去掉 CC 中一条边后得到的图仍然是连通的,这样不断去掉 GG 中圈,最后得到一个 GG 的无圈连通子图 TT ,它为 GG 的一棵生成树。

        定理 1 的证明实际上给出了连通图 GG 的生成树的求法,该方法称为破圈法

        推论

        GG(n,m)(n,m) 连通图,则 mn1m≥n-1

        连通图 GG 的生成树一般不唯一⚠️

        # 生成树的计数

        # 凯莱递推计数法

        凯莱 (Cayley 1821-1895): 剑桥大学数学教授,著名代数学家,发表论文数仅次于 Erdos,Euler,Cauchy. 著名成果是 1854 年定义了抽象群,并且得到著名定理:任意一个群都和一个变换群同构。同时,他也是一名出色的律师,作律师 14 年期间,发表 200 多篇数学论文,著名定理也是在该期间发表的。

        GG 的边 ee 称为被收缩,是指删掉 ee 后,把 ee 的两个端点重合,如此得到的图记为G\cdote;

        τ(G)\tau(G) 表示 GG 的生成树颗数。

        τ(G)=τ(Ge)+τ(Ge)\tau(G)=\tau(G-e)+\tau(G\cdot e)


        证明:对于 GG 的一条边 ee 来说,GG 的生成树中包含边 ee 的棵数为 GeG\cdot e ,而不包含 ee 的棵数为 GeG-e

        凯莱公式的缺点是:

        • 计算量很大
        • 不能具体指出每棵生成树

        # 关联矩阵计数法

        n×m 矩阵的一个阶数为 min{n,m}\min\{n,m\} 的子方阵,称为它的一个主子阵;主子阵的行列式称为主子行列式

        显然,当 n<mn<m 时,n×m 矩阵 CmnC_m^n 个主子阵。

        AmA_m 是连通图 GG 的基本关联矩阵的主子阵,则 AmA_m 非奇异的充分必要条件是相应于 AmA_m 的列的那些边构成 GG 的一棵生成树。

        • 该方法的优点是不仅指出生成树棵数,而且能绘出所有不同生成树;
        • 缺点是找所有非奇异主子阵计算量太大!

        # 矩阵树定理

        该定理是由物理学家克希荷夫提出的。他于 1824 年出生于普鲁士的哥尼斯堡。1845 年因宣布著名的克希荷夫电流电压定律而闻名,1847 年大学毕业时发表了生成树计数文章,给出了矩阵树定理。他的一生主要花在实验物理上。担任过德国柏林数学物理会主席职务

        GG 的生成树棵数为拉普拉斯矩阵 LL 的任意一个元素的代数余子式。

        # 最小生成树

        # Kruskal 算法

        克鲁斯克尔 (Kruskal):1928 年生,一家 3 弟兄都是数学家 1954 年在普林斯顿大学获博士学位,导师是 ErdÖs, 他大部分研究工作数学和语言学,主要在贝尔实验室工作。1956 年发表包含克鲁斯克尔算论文,使他名声大振

        GG 中的最小边开始,进行避圈式扩张。

        # 算法流程

        1. 选择边 e1e_1 , 使得其权值最小;
        2. 若已经选定边 e1,e2,,eke_1,e_2,\cdots,e_k ,则从 E{e1,e2,,ek}E-\{e_1,e_2,\cdots,e_k\} 中选择边 ek+1e_{k+1} , 使得:
          • [e1,e2,,ek+1][e_1,e_2,\cdots,e_{k+1}] 为无圈图
          • ek+1e_{k+1} 的权值 w(ek+1)w(e_{k+1}) 尽可能小。
        3. 当 (2) 不能进行时,停止。

        # 完备性证明

        克鲁斯克尔算法得到的任何生成树一定是最小生成树。证明:

        GG 是一个 nn 阶连通赋权图,用 T=G[{e1,e2,,en1}]T^*=G[\{e_1,e_2,\cdots,e_{n-1}\}](导出子图)表示由克鲁斯克尔算法得到的一棵生成树,我们证明:它是最小生成树

        TTGG 的一棵最小生成树。若 TTT^*≠T.

        由克鲁斯克尔算法容易知道:TTT\cap T^*≠\varnothing.

        于是令 f(T)=kf(T)=k 表示 TT^* 中的边 eie_i 不在 TT 中的最小 ii 值。即可令 T=G[{e1,e2,,ek1,ek,,en1}]T=G[\{e_1,e_2,\cdots,e_{k-1},e'_k,\cdots,e'_{n-1}\}](导出子图)

        考虑:TekT∪e_k , 则由树的性质,它必然为 GG 中圈 CC .

        T1=TekeT_1=T∪e_k-e , 容易知道:T1T_1 还为 GG 的一棵生成树。

        ee 是圈 CC 的在 TT 中,但不在 TT^* 中的边。

        由克鲁斯克尔算法知道:w(e)w(ek)w(e)≥w(e_k) .

        所以:w(T)w(T1)w(T)≥w(T_1) .

        这说明 T1T_1 是最小树,但这与 f(T)f(T) 的选取假设矛盾!所以:T=TT=T^*

        # 算法案例

        1. 选一条边 e1e_1 , 使得 w(e1)w(e_1) 尽可能小;
        2. 若边 e1,e2,,eie_1,e_2,\cdots,e_i 已经选定,则用下述方法从 E\{e1,e2,,ei}E\backslash\{e_1,e_2,\cdots,e_i\} 中选取边 e_
          • (a) G[{e1,e2,,ei+1}]G[\{e_1,e_2,\cdots,e_{i+1}\}] 为不相交路之并;
          • (b) w(ei+1)w(e_{i+1}) 是满足 (a) 的尽可能小的权。
        3. 当 (2) 不能继续执行时停止。

        该方法不能得到一条最小生成路

        # 管梅谷的破圈法

        在克鲁斯克尔算法基础上,我国著名数学家管梅谷教授于 1975 年提出了最小生成树的破圈法。

        从赋权图 GG 的任意圈开始,去掉该圈中权值最大的一条边,称为破圈。

        不断破圈,直到 GG 中没有圈为止,最后剩下的 GG 的子图为 GG 的最小生成树

        # Prim 算法

        Prim 算法是由 Prim 在 1957 年提出的一个著名算法。作者因此而出名。Prim (1921---) 1949 年在普林斯顿大学获博士学位,是 Sandia 公司副总裁。

        Kruskal 关注边,Prim 关注点

        对于连通赋权图 GG 的任意一个顶点 uu ,选择与点 uu 关联的且权值最小的边作为最小生成树的第一条边 e1e_1 ;

        在接下来的边 e2,e3,,en1,e_2,e_3,\cdots,e_{n-1}, 在与一条已经选取的边只有一个公共端点的的所有边中,选取权值最小的边。

        # 树图

        连通图 GG 的树图是指这样的图,它的顶点是 GG 的生成树 T1,T2,,TτT_1,T_2,\cdots,T_\tauTiT_iTjT_j 相连当且仅当它们恰有 n2n-2 条公共边。

        # 连通性

        任何连通图的树图是连通图。


        证明:只需证明,对任意 TiT_iTjT_j ,在树图中存在连接它们的路即可!

        对任意 TiT_iTjT_j , 设 e1,e2,,ek(k<n2)e_1,e_2,\cdots,e_k\ (k<n-2) 是它们的公共边。

        由树的性质:ek+1E(Ti)∃e'_{k+1}\in E(T_i) , 但 ek+1∉E(Tj)e'_{k+1}\not\in E(T_j) , 使得:Tj+ek+1T_j +e'_{k+1} 有唯一圈。

        该圈中: ek+1E(Tj)∃e_{k+1}\in E(T_j) , 但 ek+1∉E(Ti)e_{k+1}\not\in E(T_i)

        作: Ti+1=Tiek+1+ek+1T_{i+1}=T_i-e'_{k+1}+e_{k+1} ,则 TiT_iTi+1T_{i+1}n2n-2 条边相同,于是,它们邻接。

        此时,Ti+1T_{i+1}TjT_jk+1k+1 条边相同。

        如此这样作下去,可以得到连接 TiT_iTjT_j 的一条路为:Ti,Ti+1,,TjT_i, T_{i+1},\cdots,T_j

        所以,连通图 GG 的树图是连通的。

        \ No newline at end of file +树 - 图论 - 离散数学 - 数学 | Balloon Party = 風船のパーティー = fuusen no party

        # 树的基本性质

        # 树的叶子数

        TT(n,m)(n,m) 树,TT 中有 nin_i 个度为 i,(1ik)i,(1\leq i\leq k) 的点,且有 ni=n\sum n_i=n ,则有:n1=2+n3+2n4++(k2)nkn_1=2+n_3+2n_4+\cdots+(k-2)n_k

        证明:由 m=n1m=n-1 得:

        m=(n1+n2++np)1m=(n_1+n_2+\cdots+n_p)-1

        又由握手定理得:

        2m=n+2n2++knk2m=n+2n_2+\cdots+kn_k

        由上面两等式得:

        n1=2+n3+2n4++(k2)nkn_1=2+n_3+2n_4+\cdots+(k-2)n_k

        # 森林的分支数

        具有 kk 个分支的森林有 nkn-k 条边。


        证明:

        设森林 GGkk 个分支为 Ti(1ik)T_i(1≤i\leq k) ,对每个分支,使用定理 3 得

        m(Ti)=ni1(ni=V(Ti))m(T_i)=n_i-1(n_i=|V(T_i)|)

        所以:

        m(G)=i=1km(Ti)=nkm(G)=\sum_{i=1}^{k}m(T_i)=n-k

        # 图的边数下限

        每个 nn 阶连通图的边数至少为 n1n-1.

        证明:

        (1) 如果 nn 阶连通图 GG 没有一度顶点,那么由握手定理有:

        m(G)=12vV(G)d(v)nm(G)=\frac{1}{2}\sum_{v\in V(G)}d(v)\geq n


        (2) 如果 GG 有一度顶点:

        对顶点数作数学归纳。

        n=1n=1 时,结论显然

        设当 n=kn=k 时,结论成立。

        n=k+1n=k+1 时,设 uuGG 的一度顶点,则 GuG-u 为具有 kk 个顶点的连通图

        (2.1) 若 GuG-u 有一度顶点,则由归纳假设,其边数至少 k1k-1 ,于是 GG 的边数至少有 kk 条;

        (2.2) 如果 GuG-u 没有一度顶点,由握手定理有:

        m(Gu)=12vV(Gu)d(v)km(G-u)=\frac{1}{2}\sum_{v\in V(G-u)}d(v)\geq k

        所以,GG 至少有 k+1k+1 条边。

        而当 GG 是树时,边数恰为 n1n-1 .

        所以 n 阶连通图 G 至少有 n-1 条边。

        所以,树也被称为最小连通图。

        任意树 TT 的两个不邻接顶点之间添加一条边后,可以得到唯一圈


        证明:

        uuvv 是树 TT 的任意两个不邻接顶点,由定理 2 知:有唯一路 PP 连接 uuvv ;于是 P{uv}P\cup\{uv\} 是一圈。

        显然,由 PP 的唯一性也就决定了 P{uv}P\cup\{uv\} 的唯一性。

        # 树的叶子数下限

        GG 是树且 Δk\Delta≥k ,则 GG 至少有 kk 个一度顶点(叶子顶点)


        证明:反证法,假设 GG 有至多 k1k-1 个叶子顶点

        2m(G)=vV(G)d(v)k1+k+2(nk)=2n1>2n22m(G)=\sum_{v\in V(G)}d(v)\geq k-1+k+2(n-k)=2n-1\gt 2n-2

        所以 m(G)>n1m(G)\gt n-1 ,与 GG 是树矛盾!

        # 森林的路分解

        GG 是森林且恰有 2k2k奇度顶点,则在 GG 中有 kk 条边不重合的路 P1,P2,,PkP_1,P_2,\cdots,P_k 使得:

        E(G)=E(P1)E(P2)E(Pk)E(G)=E(P_1)\cup E(P_2)\cup\cdots\cup E(P_k)

        证明:对 kk 作数学归纳。

        k=1k=1 时,GG 只有两个奇数度顶点,此时,容易证明,GG 是一条路;

        设当 k=tk=t 时,结论成立。令 k=t+1k=t+1

        GG 中一个分支中取两个一度项点 uuvv ,令 PP 是连接该两个项点的唯一路,则 GPG-P 是有 2t2t 个奇数顶点的森林,由归纳假设,它可以分解 tt 条边不重合的路之并,所以 GG 可以分解为 t+1t+1 条边不重合的路之并。

        对图作某种形式的分解,是图论的一个研究对象,它在网络结构分析中具有重要作用。

        # 树的同构子图

        TTkk 阶树。若图 GG 满足 δk1\delta≥k-1 ,则 TT 同构于 GG 的某个子图。

        证明:对 kk 作数学归纳

        k=1k=1 时,结论显然。

        假设对 k1,(k3)k-1,(k ≥3) 的每颗树 TT ,以及最小度至少为 k2k-2 的每个图 HHT1T_1 同构于 HH 的某个子图 FF

        现在设 TTkk 阶树,且图 GG 满足 δk1\delta≥k-1 ,证明 TT 同构于 GG 的某个子图:

        uuTT 的树叶,vvuu 的邻接顶点。则 TuT-uk1k-1 阶树。

        由于 δ(G)k1>k2\delta(G)≥k-1>k-2 ,由归纳假设,TuT-u 同构于 GG 的某个子图 FF

        树的同构子图证明

        vv 是与 TTvv 相对应的 FF 中的点,由于 dG(v1)k1d_{G(v_1)}\geq k-1 ,所以 vvGG 中一定有相异于 FF 中的邻点 ww ,作 F{u1w}F\cup\{u_1w\} ,则该子图和 TT 同构。

        # 树的度序列问题

        在第一章中,介绍了判定一个非增非负序列是否为简单图的度序列定理。下面介绍一个判定非增非负序列是否为树的度序列的简单方法。

        S={d1,d2,,dn}S=\{d_1,d_2,\cdots,d_n\}nn 个正整数序列,它们满足:

        • d1d2dnd_1\geq d_2\geq\cdots\geq d_n
        • di=2(n1)\sum di=2(n-1)

        则存在一颗树 TT ,其度序列为 SS


        证明:对 nn 作数学归纳。

        n=1,2n=1,2 时,结论显然。

        假设对 n=kn=k 时结论成立。设 n=k+1n=k+1,

        首先,序列中至少一个数为 1,否则,序列和大于 2k2k ,与条件相矛盾!

        所以,dk+1=1d_{k+1}=1

        我们从序列中删掉 d1d_1dk+1d_{k+1} ,增加数 d=di1d^*=d_i-1 放在它应该在的位置。得到序列 SS ,该序列含 kk 个数,序列和为 2(k1)2(k-1)

        由归纳假设,存在树 TT ,它的度序列为 SiS_i

        现在,增加结点 vv ,把它和 T1T_1 中点 dd^* 相连得到树 TT 为所求

        # 树的中心与形心

        # 中心

        1. 图的顶点的离心率 e(v)=max{d(u,v)uV(G)}e(v)=\max\{d(u,v)|u\in V(G)\}
          • 图的直径最大离心率
        2. 图的半径 r(G)=min{e(v)vV(G)}r(G)=\min\{e(v)|v\in V(G)\}
        3. 图的中心点离心率等于半径的点
        4. 图的中心:全体中心点的集合

        对树 TT 的阶数 nn 作归纳证明。

        n=1,2n=1,2 时,结论显然成立

        设对 n<k,(k3)n<k,(k≥3) 的树结论成立,设 TTkk 阶树

        容易知道,删掉 TT 的所有叶,得到的树 TT ,的每个点的离心率比它们在 TT 中离心率减少 1

        又因 TT 的叶不能是中心点,所以 TT 的中心点在 T1T_1

        这样,若点 uu 的离心率在 TT 中最小,则在 T1T_1 中依然最小,即说明 TT 的中心点是 T1T_1 的中心点,反之亦然。

        # 形心

        uu 是树 TT 的任意一个顶点,树 TT 在顶点 uu 的分支是指包含 uu 作为一个叶点的极大子树,其分支数为顶点 uu 的度数;

        1. uu:树 TTuu 点的分支中边的最大数目称为;
        2. TT形心点:树 TT权值最小的点
        3. TT形心:全体形心点的集合

        每一棵树有一个由一个点两个邻接的点组成的形心。

        # 生成树

        # 生成树的性质

        每个连通图至少包含一棵生成树


        证明:

        如果连通图 GG 是树,则其本身是一棵生成树;

        若连通图 GG 中有圈 CC ,则去掉 CC 中一条边后得到的图仍然是连通的,这样不断去掉 GG 中圈,最后得到一个 GG 的无圈连通子图 TT ,它为 GG 的一棵生成树。

        定理 1 的证明实际上给出了连通图 GG 的生成树的求法,该方法称为破圈法

        推论

        GG(n,m)(n,m) 连通图,则 mn1m≥n-1

        连通图 GG 的生成树一般不唯一⚠️

        # 生成树的计数

        # 凯莱递推计数法

        凯莱 (Cayley 1821-1895): 剑桥大学数学教授,著名代数学家,发表论文数仅次于 Erdos,Euler,Cauchy. 著名成果是 1854 年定义了抽象群,并且得到著名定理:任意一个群都和一个变换群同构。同时,他也是一名出色的律师,作律师 14 年期间,发表 200 多篇数学论文,著名定理也是在该期间发表的。

        GG 的边 ee 称为被收缩,是指删掉 ee 后,把 ee 的两个端点重合,如此得到的图记为G\cdote;

        τ(G)\tau(G) 表示 GG 的生成树颗数。

        τ(G)=τ(Ge)+τ(Ge)\tau(G)=\tau(G-e)+\tau(G\cdot e)


        证明:对于 GG 的一条边 ee 来说,GG 的生成树中包含边 ee 的棵数为 GeG\cdot e ,而不包含 ee 的棵数为 GeG-e

        凯莱公式的缺点是:

        • 计算量很大
        • 不能具体指出每棵生成树

        # 关联矩阵计数法

        n×m 矩阵的一个阶数为 min{n,m}\min\{n,m\} 的子方阵,称为它的一个主子阵;主子阵的行列式称为主子行列式

        显然,当 n<mn<m 时,n×m 矩阵 CmnC_m^n 个主子阵。

        AmA_m 是连通图 GG 的基本关联矩阵的主子阵,则 AmA_m 非奇异的充分必要条件是相应于 AmA_m 的列的那些边构成 GG 的一棵生成树。

        • 该方法的优点是不仅指出生成树棵数,而且能绘出所有不同生成树;
        • 缺点是找所有非奇异主子阵计算量太大!

        # 矩阵树定理

        该定理是由物理学家克希荷夫提出的。他于 1824 年出生于普鲁士的哥尼斯堡。1845 年因宣布著名的克希荷夫电流电压定律而闻名,1847 年大学毕业时发表了生成树计数文章,给出了矩阵树定理。他的一生主要花在实验物理上。担任过德国柏林数学物理会主席职务

        GG 的生成树棵数为拉普拉斯矩阵 LL 的任意一个元素的代数余子式。

        # 最小生成树

        # Kruskal 算法

        克鲁斯克尔 (Kruskal):1928 年生,一家 3 弟兄都是数学家 1954 年在普林斯顿大学获博士学位,导师是 ErdÖs, 他大部分研究工作数学和语言学,主要在贝尔实验室工作。1956 年发表包含克鲁斯克尔算论文,使他名声大振

        GG 中的最小边开始,进行避圈式扩张。

        # 算法流程

        1. 选择边 e1e_1 , 使得其权值最小;
        2. 若已经选定边 e1,e2,,eke_1,e_2,\cdots,e_k ,则从 E{e1,e2,,ek}E-\{e_1,e_2,\cdots,e_k\} 中选择边 ek+1e_{k+1} , 使得:
          • [e1,e2,,ek+1][e_1,e_2,\cdots,e_{k+1}] 为无圈图
          • ek+1e_{k+1} 的权值 w(ek+1)w(e_{k+1}) 尽可能小。
        3. 当 (2) 不能进行时,停止。

        # 完备性证明

        克鲁斯克尔算法得到的任何生成树一定是最小生成树。证明:

        GG 是一个 nn 阶连通赋权图,用 T=G[{e1,e2,,en1}]T^*=G[\{e_1,e_2,\cdots,e_{n-1}\}](导出子图)表示由克鲁斯克尔算法得到的一棵生成树,我们证明:它是最小生成树

        TTGG 的一棵最小生成树。若 TTT^*≠T.

        由克鲁斯克尔算法容易知道:TTT\cap T^*≠\varnothing.

        于是令 f(T)=kf(T)=k 表示 TT^* 中的边 eie_i 不在 TT 中的最小 ii 值。即可令 T=G[{e1,e2,,ek1,ek,,en1}]T=G[\{e_1,e_2,\cdots,e_{k-1},e'_k,\cdots,e'_{n-1}\}](导出子图)

        考虑:TekT∪e_k , 则由树的性质,它必然为 GG 中圈 CC .

        T1=TekeT_1=T∪e_k-e , 容易知道:T1T_1 还为 GG 的一棵生成树。

        ee 是圈 CC 的在 TT 中,但不在 TT^* 中的边。

        由克鲁斯克尔算法知道:w(e)w(ek)w(e)≥w(e_k) .

        所以:w(T)w(T1)w(T)≥w(T_1) .

        这说明 T1T_1 是最小树,但这与 f(T)f(T) 的选取假设矛盾!所以:T=TT=T^*

        # 算法案例

        1. 选一条边 e1e_1 , 使得 w(e1)w(e_1) 尽可能小;
        2. 若边 e1,e2,,eie_1,e_2,\cdots,e_i 已经选定,则用下述方法从 E\{e1,e2,,ei}E\backslash\{e_1,e_2,\cdots,e_i\} 中选取边 e_
          • (a) G[{e1,e2,,ei+1}]G[\{e_1,e_2,\cdots,e_{i+1}\}] 为不相交路之并;
          • (b) w(ei+1)w(e_{i+1}) 是满足 (a) 的尽可能小的权。
        3. 当 (2) 不能继续执行时停止。

        该方法不能得到一条最小生成路

        # 管梅谷的破圈法

        在克鲁斯克尔算法基础上,我国著名数学家管梅谷教授于 1975 年提出了最小生成树的破圈法。

        从赋权图 GG 的任意圈开始,去掉该圈中权值最大的一条边,称为破圈。

        不断破圈,直到 GG 中没有圈为止,最后剩下的 GG 的子图为 GG 的最小生成树

        # Prim 算法

        Prim 算法是由 Prim 在 1957 年提出的一个著名算法。作者因此而出名。Prim (1921---) 1949 年在普林斯顿大学获博士学位,是 Sandia 公司副总裁。

        Kruskal 关注边,Prim 关注点

        对于连通赋权图 GG 的任意一个顶点 uu ,选择与点 uu 关联的且权值最小的边作为最小生成树的第一条边 e1e_1 ;

        在接下来的边 e2,e3,,en1,e_2,e_3,\cdots,e_{n-1}, 在与一条已经选取的边只有一个公共端点的的所有边中,选取权值最小的边。

        # 树图

        连通图 GG 的树图是指这样的图,它的顶点是 GG 的生成树 T1,T2,,TτT_1,T_2,\cdots,T_\tauTiT_iTjT_j 相连当且仅当它们恰有 n2n-2 条公共边。

        # 连通性

        任何连通图的树图是连通图。


        证明:只需证明,对任意 TiT_iTjT_j ,在树图中存在连接它们的路即可!

        对任意 TiT_iTjT_j , 设 e1,e2,,ek(k<n2)e_1,e_2,\cdots,e_k\ (k<n-2) 是它们的公共边。

        由树的性质:ek+1E(Ti)∃e'_{k+1}\in E(T_i) , 但 ek+1∉E(Tj)e'_{k+1}\not\in E(T_j) , 使得:Tj+ek+1T_j +e'_{k+1} 有唯一圈。

        该圈中: ek+1E(Tj)∃e_{k+1}\in E(T_j) , 但 ek+1∉E(Ti)e_{k+1}\not\in E(T_i)

        作: Ti+1=Tiek+1+ek+1T_{i+1}=T_i-e'_{k+1}+e_{k+1} ,则 TiT_iTi+1T_{i+1}n2n-2 条边相同,于是,它们邻接。

        此时,Ti+1T_{i+1}TjT_jk+1k+1 条边相同。

        如此这样作下去,可以得到连接 TiT_iTjT_j 的一条路为:Ti,Ti+1,,TjT_i, T_{i+1},\cdots,T_j

        所以,连通图 GG 的树图是连通的。

        \ No newline at end of file diff --git a/page/10/index.html b/page/10/index.html index 2fe2a462..0bc83d64 100644 --- a/page/10/index.html +++ b/page/10/index.html @@ -1 +1 @@ -Balloon Party = 風船のパーティー = fuusen no party
        9.3k words 8 mins.

        YatCPU - - - lab1 - - - lab2 - - - lab3 - - - lab4 # 测试用例及其波形图分析 # SayGoodbyeTest src/test/scala/riscv/BoardTest.scala 其中 Top 模块来自: 测试执行程序来自 csrc/say_goodbye.c : # 测试用例的功能 与 lab2 中 BoardSayGoodbyeTest 功能相似 测试能否正常执行打印 &quot;Never gonna give you up~ Never gonna let you down~\nNever gonna...
        4.9k words 4 mins.

        YatCPU - - - lab1 - - - lab2 - - - lab3 - - - lab4 # 测试用例及其波形图分析 # SayGoodbyeTest src/test/scala/riscv/BoardTest.scala 其中 Top 模块来自: 测试执行程序来自 csrc/say_goodbye.c : # 测试用例的功能 测试能否正常执行打印 &quot;Never gonna give you up~ Never gonna let you down~\nNever gonna run around and~ desert...
        148 words 1 mins.

        这里是我 2023 年大二上半学期的离散数学基础课程期末复习的几天中整理的一些知识的思维导图 逻辑、图论的部分没有整理,主要包括集合、关系、代数等内容 离散数学基础 - 整理: --- 排列与组合 --- 组合计数基本原理 --- 排列组合计数问题 --- 递推关系式 --- 关系 --- 集合与关系的运算 --- 函数 --- 代数系统
        8.2k words 7 mins.

        离散数学基础 - 整理: --- 排列与组合 --- 组合计数基本原理 --- 排列组合计数问题 --- 递推关系式 --- 关系 --- 集合与关系的运算 --- 函数 --- 代数系统 # 思维导图 # 二元运算 封闭性 # 定义 f:S×S→S是集合S的运算,则称集合S对运算f封闭\begin{aligned}&amp; f:S\times S\to S...
        3.5k words 3 mins.

        离散数学基础 - 整理: --- 排列与组合 --- 组合计数基本原理 --- 排列组合计数问题 --- 递推关系式 --- 关系 --- 集合与关系的运算 --- 函数 --- 代数系统 # 思维导图 # 递推关系式相关概念 # 定义 用序列中某些前面的项ai,0≤i&lt;n表示第n项an的等式\text{用序列中某些前面的项}a_i,0\leq i\lt n\text{表示第}n\text{项}a_n\text{的等式} 用序列中某些前面的项ai​,0≤i&lt;n表示第n项an​的等式 #...
        7.2k words 7 mins.

        离散数学基础 - 整理: --- 排列与组合 --- 组合计数基本原理 --- 排列组合计数问题 --- 递推关系式 --- 关系 --- 集合与关系的运算 --- 函数 --- 代数系统 # 函数 (全函数) # 描述 集合A到B的函数,是A×笛卡尔积的一类子集集合A到A的函数也称为A上的函数定义域的每个元素都存在陪域的唯一元素与之对应\begin{aligned} &amp;\text{集合A到B的函数,是}A\times...
        3.4k words 3 mins.

        离散数学基础 - 整理: --- 排列与组合 --- 组合计数基本原理 --- 排列组合计数问题 --- 递推关系式 --- 关系 --- 集合与关系的运算 --- 函数 --- 代数系统 # 思维导图 # 排列 # 记号 P(n,r)=Pnr=AnrP(n,r)=P_n^r=A_n^r P(n,r)=Pnr​=Anr​ # 描述 n 个 (可区别的) 物体 n 个物体的 r - 排列 n 个物体的 n - 排列,或 n 个物体的全排列 集合 S,|S|=n S 的 r - 排列 S 的 n - 排列,或 S 的全排列 #...
        5.2k words 5 mins.

        离散数学基础 - 整理: --- 排列与组合 --- 组合计数基本原理 --- 排列组合计数问题 --- 递推关系式 --- 关系 --- 集合与关系的运算 --- 函数 --- 代数系统 # 思维导图 # 笛卡尔积 # 定义 A×B={⟨a,b⟩∣a∈A∧b∈B}\begin{aligned}&amp; A\times B=\{\langle a,b\rangle|a\in A\land b\in B\} \end{aligned} ​A×B={⟨a,b⟩∣a∈A∧b∈B}​ #...
        6.6k words 6 mins.

        离散数学基础 - 整理: --- 排列与组合 --- 组合计数基本原理 --- 排列组合计数问题 --- 递推关系式 --- 关系 --- 集合与关系的运算 --- 函数 --- 代数系统 # 思维导图 # 并 # 定义 R∪S={&lt;a,b&gt;∣&lt;a,b&gt;∈R  ∨&lt;a,b&gt;∈S}\begin{aligned}&amp; R\cup S =\{&lt;a,b&gt;|&lt;a,b&gt;\in R\ \ \lor...
        \ No newline at end of file +Balloon Party = 風船のパーティー = fuusen no party
        14k words 13 mins.

        YatCPU - - - lab1 - - - lab2 - - - lab3 - - - lab4 # 测试用例及其波形图分析 # SayGoodbyeTest src/test/scala/riscv/BoardTest.scala 其中 Top 模块来自: # 测试用例的功能 用于烧板验证,但由于 CPU 还未能实现中断,该程序只能通过 LED 灯闪烁判断 CPU 能否正常工作 在 lab1 中该测试用例可以用于检查 CPU 及其内部各个模块、模块内的端口信号是否有完整地实现、初始化,即 &quot;Fully...
        9.3k words 8 mins.

        YatCPU - - - lab1 - - - lab2 - - - lab3 - - - lab4 # 测试用例及其波形图分析 # SayGoodbyeTest src/test/scala/riscv/BoardTest.scala 其中 Top 模块来自: 测试执行程序来自 csrc/say_goodbye.c : # 测试用例的功能 与 lab2 中 BoardSayGoodbyeTest 功能相似 测试能否正常执行打印 &quot;Never gonna give you up~ Never gonna let you down~\nNever gonna...
        148 words 1 mins.

        这里是我 2023 年大二上半学期的离散数学基础课程期末复习的几天中整理的一些知识的思维导图 逻辑、图论的部分没有整理,主要包括集合、关系、代数等内容 离散数学基础 - 整理: --- 排列与组合 --- 组合计数基本原理 --- 排列组合计数问题 --- 递推关系式 --- 关系 --- 集合与关系的运算 --- 函数 --- 代数系统
        8.2k words 7 mins.

        离散数学基础 - 整理: --- 排列与组合 --- 组合计数基本原理 --- 排列组合计数问题 --- 递推关系式 --- 关系 --- 集合与关系的运算 --- 函数 --- 代数系统 # 思维导图 # 二元运算 封闭性 # 定义 f:S×S→S是集合S的运算,则称集合S对运算f封闭\begin{aligned}&amp; f:S\times S\to S...
        3.5k words 3 mins.

        离散数学基础 - 整理: --- 排列与组合 --- 组合计数基本原理 --- 排列组合计数问题 --- 递推关系式 --- 关系 --- 集合与关系的运算 --- 函数 --- 代数系统 # 思维导图 # 递推关系式相关概念 # 定义 用序列中某些前面的项ai,0≤i&lt;n表示第n项an的等式\text{用序列中某些前面的项}a_i,0\leq i\lt n\text{表示第}n\text{项}a_n\text{的等式} 用序列中某些前面的项ai​,0≤i&lt;n表示第n项an​的等式 #...
        7.2k words 7 mins.

        离散数学基础 - 整理: --- 排列与组合 --- 组合计数基本原理 --- 排列组合计数问题 --- 递推关系式 --- 关系 --- 集合与关系的运算 --- 函数 --- 代数系统 # 函数 (全函数) # 描述 集合A到B的函数,是A×笛卡尔积的一类子集集合A到A的函数也称为A上的函数定义域的每个元素都存在陪域的唯一元素与之对应\begin{aligned} &amp;\text{集合A到B的函数,是}A\times...
        6.6k words 6 mins.

        离散数学基础 - 整理: --- 排列与组合 --- 组合计数基本原理 --- 排列组合计数问题 --- 递推关系式 --- 关系 --- 集合与关系的运算 --- 函数 --- 代数系统 # 思维导图 # 并 # 定义 R∪S={&lt;a,b&gt;∣&lt;a,b&gt;∈R  ∨&lt;a,b&gt;∈S}\begin{aligned}&amp; R\cup S =\{&lt;a,b&gt;|&lt;a,b&gt;\in R\ \ \lor...
        3.4k words 3 mins.

        离散数学基础 - 整理: --- 排列与组合 --- 组合计数基本原理 --- 排列组合计数问题 --- 递推关系式 --- 关系 --- 集合与关系的运算 --- 函数 --- 代数系统 # 思维导图 # 排列 # 记号 P(n,r)=Pnr=AnrP(n,r)=P_n^r=A_n^r P(n,r)=Pnr​=Anr​ # 描述 n 个 (可区别的) 物体 n 个物体的 r - 排列 n 个物体的 n - 排列,或 n 个物体的全排列 集合 S,|S|=n S 的 r - 排列 S 的 n - 排列,或 S 的全排列 #...
        5.2k words 5 mins.

        离散数学基础 - 整理: --- 排列与组合 --- 组合计数基本原理 --- 排列组合计数问题 --- 递推关系式 --- 关系 --- 集合与关系的运算 --- 函数 --- 代数系统 # 思维导图 # 笛卡尔积 # 定义 A×B={⟨a,b⟩∣a∈A∧b∈B}\begin{aligned}&amp; A\times B=\{\langle a,b\rangle|a\in A\land b\in B\} \end{aligned} ​A×B={⟨a,b⟩∣a∈A∧b∈B}​ #...
        \ No newline at end of file diff --git a/page/11/index.html b/page/11/index.html index ac050593..8ca5dad6 100644 --- a/page/11/index.html +++ b/page/11/index.html @@ -1 +1 @@ -Balloon Party = 風船のパーティー = fuusen no party
        4.8k words 4 mins.

        离散数学基础 - 整理: --- 排列与组合 --- 组合计数基本原理 --- 排列组合计数问题 --- 递推关系式 --- 关系 --- 集合与关系的运算 --- 函数 --- 代数系统 # 思维导图 # 常用引理 # 引理 1 # 公式 ∣AB‾∣=∣A−B∣=∣A∣−∣AB∣| A\overline{B} |=| A-B | = |A| -| A B| ∣AB∣=∣A−B∣=∣A∣−∣AB∣ AB‾=A−B=A−ABA\overline{B} = A - B = A - A B AB=A−B=A−AB # 证明 略 # 容斥原理 # 2...
        928 words 1 mins.

        # 思维导图 # 为什么要刷新? # DRAM 存储位元的特点 # 靠电容上的电荷存储效应记忆信息,虽然有 MOS 高电阻 (1012∼1015Ω10^{12} \sim 10^{15} \Omega1012∼1015Ω),仍会泄漏电荷 # 为保证所存信息的正确性,需要用充电的方法及时使所有位元的电容上电荷恢复到泄漏前的状态 # DRAM 刷新的有关参数 # 刷新周期 Trc (refresh cycle) 对同一存储位元连续两次刷新,仍能保证鉴别出原存信息的最大允许间隔时间,即在 Trc 内必须对每个单元刷新一遍 一般为 ms 级,亦称为刷新间隔时间 在 Trc...
        2.4k words 2 mins.

        # 思维导图 # 分区方案 # 简单分区: 原理:使用长度不等的固定长分区,当一个进程调入主存时,分配给它一个能容纳的最小分区 缺点:固定长度的分区,可能会浪费 主存空间。多数情况下,进程对分区大小的需求不可能和提供的分区大小一样 # 可变长分区(Partitioning) 原理:分配的分区大小与进程所需大小一样 特点:开始较好 ,但到最后存储器中会有许多碎片。时间越长,存储器的碎片会越来越多,存储器的利用率下降 程序员自己手动管理内存,可以改善内存碎片化问题(例如一次性 malloc 或 new、stl 新建容器好几 kb 大小…… #...
        3.6k words 3 mins.

        # 思维导图 # 基本术语 # 记忆单元 / 存储位元 / 位元 (Cell) 具有两种稳态的能够表示二进制数 0 和 1 的物理器件 # 存储单元 / 编址单位 / 寻址单位 (Addressing Unit) 存储器中具有相同地址的那些位构成一个存储单元,又称为一个编址单位 # 存储体 / 存储矩阵 / 存储阵列 (Bank) 所有存储单元构成一个存储阵列 # 存储器地址寄存器 (Memory Address Register – MAR) 用于存放主存单元地址的寄存器 # 存储器数据寄存器 (Memory Data Register – MDR) 用于存放主存单元中数据的寄存器 #...
        4.7k words 4 mins.

        # 思维导图 # Cache 的结构 # Cache 是小容量、高速缓冲存储器,由 SRAM 组成 # 一般将 Cache 和主存的存储空间都划分为若干大小相同的块并映射 把主存划分成大小相等的主存块 (Block) Cache 中存放一个主存块的对应单位称为行 (line) 或槽 (Slot) 或项 (Entry) 或块 (Block),一段 Cache 行包括: Valid 位,1 位 LRU 位 Dirty 位,1 位,采用 Write Back (写回、一次性写、回写) 时需要 Tag 位 Data 数据 主存块 (Block) 与 Cache 中的 行 / 槽 /...
        208 words 1 mins.

        # 2023 看番年度总结 * 星号是没完全看完 # 寒假: EVA 旧版 凉宫春日 Angel Beats! Little Busters Air 86 - 不存在的战区 ISLAND 斩赤红之瞳 可塑性记忆 虚构推理 冰海战记 魔圆 魔纪 想要成为影之实力者 # 一些短的: 快要坏掉的八音盒 星之梦 + 星之人 # 大一下: Canon * 时光沙漏 安达与岛村 终将成为你 天国大魔境 # 暑假: 处刑少女的生存手册 Happy Sugar Life # 大二上: mygo * 黑之契约者 * Fate Zero
        1.2k words 1 mins.

        # 思维导图 # CPU (Processor) # Control 控制器 指令执行过程中,数据所经过的路径 (包括路径中的部件)—— 指令的执行部件 控制器的基本功能 取指令 (控制指令流出 — PC) 分析指令 (控制指令分析 — IR) 执行指令,发出各种操作命令 (控制指令执行 — REGs 和 ALU) 确定下一条指令的地址 (控制指令流向 —PC) 执行环境的建立与保护 (控制执行环境的维护 —FLAGs/PSW) # Datapath 数据通路 对指令进行译码,生成指令对应的控制信号,控制数据通路的动作,能对指令的执行部件发出控制信号 ——...
        2.8k words 3 mins.

        # 思维导图 CPU # 单周期 CPU # 单周期控制器 控制信号在整个指令执行过程中保持不变 真值表反映了指令和控制信号的关系 按照真值表就可以实现控制器! # 多周期 CPU # 多周期控制器功能描述方式 有限状态机 采用组合逻辑设计方法用硬连线路 (PLA) 实现 微程序 用 ROM 存放微程序实现 # MIPS 各指令的周期数: 划分关键是: 所有指令均需要 IF,ID,EX 这三个周期 需要访存的指令额外需要一个周期 MEM(ID 内的访存不算,所以只有 Load/Store) 需要写回寄存器的指令额外需要一个周期 WB R-Type...
        2.1k words 2 mins.

        # 思维导图 # 异常和中断的概念 # 异常: 异常来自于 cpu 内部的正在执行的指令本身 # 中断: 中断来自于 cpu 外部的硬件触发 # 统称为 “中断”: 内中断 (内部异常) CPU 自己产生而不通过中断请求线请求,皆为不可屏蔽中断 外中断 (外部中断) 通过中断请求线 INTR 和 NMI 来实现 # 内部 “异常”:CPU 内发生的意外事件或特殊事件 # 按发生原因分为硬故障中断和程序性中断两类 硬故障中断:如电源掉电、硬件线路故障等 程序性中断:执行某条指令时发生的 “异常 (Exception)”,如 溢出 缺页 越界 越权 非法指令 除数为...
        2.7k words 2 mins.

        # 思维导图 # 补码 # 符号部分同原码 数的最高位为符号位,0 表示正数,1 表示负数 # 数字部分与它的符号位有关 对于正数,补码数值部分与原码数值部分相同 对于负数,补码数值部分是将原码数值部分按位取反再加 1 ,即在反码数值部分基础上加 1 # 阶码 # 补码和移码的关系:符号位相反、数值位相同 阶码用移码表示,阶码加减运算后得到补码,再将符号位取反即可得到运算后的阶码 移码的和、差 = 真值的和、差的补码 # 补码 E_1 的 (mod 2^n) n 位移码: [E_1] 移 = 2^(n-1) + E_1 # 补码 E_1 的 (mod 2^n) n...
        \ No newline at end of file +Balloon Party = 風船のパーティー = fuusen no party
        4.8k words 4 mins.

        离散数学基础 - 整理: --- 排列与组合 --- 组合计数基本原理 --- 排列组合计数问题 --- 递推关系式 --- 关系 --- 集合与关系的运算 --- 函数 --- 代数系统 # 思维导图 # 常用引理 # 引理 1 # 公式 ∣AB‾∣=∣A−B∣=∣A∣−∣AB∣| A\overline{B} |=| A-B | = |A| -| A B| ∣AB∣=∣A−B∣=∣A∣−∣AB∣ AB‾=A−B=A−ABA\overline{B} = A - B = A - A B AB=A−B=A−AB # 证明 略 # 容斥原理 # 2...
        928 words 1 mins.

        # 思维导图 # 为什么要刷新? # DRAM 存储位元的特点 # 靠电容上的电荷存储效应记忆信息,虽然有 MOS 高电阻 (1012∼1015Ω10^{12} \sim 10^{15} \Omega1012∼1015Ω),仍会泄漏电荷 # 为保证所存信息的正确性,需要用充电的方法及时使所有位元的电容上电荷恢复到泄漏前的状态 # DRAM 刷新的有关参数 # 刷新周期 Trc (refresh cycle) 对同一存储位元连续两次刷新,仍能保证鉴别出原存信息的最大允许间隔时间,即在 Trc 内必须对每个单元刷新一遍 一般为 ms 级,亦称为刷新间隔时间 在 Trc...
        3.6k words 3 mins.

        # 思维导图 # 基本术语 # 记忆单元 / 存储位元 / 位元 (Cell) 具有两种稳态的能够表示二进制数 0 和 1 的物理器件 # 存储单元 / 编址单位 / 寻址单位 (Addressing Unit) 存储器中具有相同地址的那些位构成一个存储单元,又称为一个编址单位 # 存储体 / 存储矩阵 / 存储阵列 (Bank) 所有存储单元构成一个存储阵列 # 存储器地址寄存器 (Memory Address Register – MAR) 用于存放主存单元地址的寄存器 # 存储器数据寄存器 (Memory Data Register – MDR) 用于存放主存单元中数据的寄存器 #...
        2.4k words 2 mins.

        # 思维导图 # 分区方案 # 简单分区: 原理:使用长度不等的固定长分区,当一个进程调入主存时,分配给它一个能容纳的最小分区 缺点:固定长度的分区,可能会浪费 主存空间。多数情况下,进程对分区大小的需求不可能和提供的分区大小一样 # 可变长分区(Partitioning) 原理:分配的分区大小与进程所需大小一样 特点:开始较好 ,但到最后存储器中会有许多碎片。时间越长,存储器的碎片会越来越多,存储器的利用率下降 程序员自己手动管理内存,可以改善内存碎片化问题(例如一次性 malloc 或 new、stl 新建容器好几 kb 大小…… #...
        4.7k words 4 mins.

        # 思维导图 # Cache 的结构 # Cache 是小容量、高速缓冲存储器,由 SRAM 组成 # 一般将 Cache 和主存的存储空间都划分为若干大小相同的块并映射 把主存划分成大小相等的主存块 (Block) Cache 中存放一个主存块的对应单位称为行 (line) 或槽 (Slot) 或项 (Entry) 或块 (Block),一段 Cache 行包括: Valid 位,1 位 LRU 位 Dirty 位,1 位,采用 Write Back (写回、一次性写、回写) 时需要 Tag 位 Data 数据 主存块 (Block) 与 Cache 中的 行 / 槽 /...
        208 words 1 mins.

        # 2023 看番年度总结 * 星号是没完全看完 # 寒假: EVA 旧版 凉宫春日 Angel Beats! Little Busters Air 86 - 不存在的战区 ISLAND 斩赤红之瞳 可塑性记忆 虚构推理 冰海战记 魔圆 魔纪 想要成为影之实力者 # 一些短的: 快要坏掉的八音盒 星之梦 + 星之人 # 大一下: Canon * 时光沙漏 安达与岛村 终将成为你 天国大魔境 # 暑假: 处刑少女的生存手册 Happy Sugar Life # 大二上: mygo * 黑之契约者 * Fate Zero
        1.2k words 1 mins.

        # 思维导图 # CPU (Processor) # Control 控制器 指令执行过程中,数据所经过的路径 (包括路径中的部件)—— 指令的执行部件 控制器的基本功能 取指令 (控制指令流出 — PC) 分析指令 (控制指令分析 — IR) 执行指令,发出各种操作命令 (控制指令执行 — REGs 和 ALU) 确定下一条指令的地址 (控制指令流向 —PC) 执行环境的建立与保护 (控制执行环境的维护 —FLAGs/PSW) # Datapath 数据通路 对指令进行译码,生成指令对应的控制信号,控制数据通路的动作,能对指令的执行部件发出控制信号 ——...
        2.8k words 3 mins.

        # 思维导图 CPU # 单周期 CPU # 单周期控制器 控制信号在整个指令执行过程中保持不变 真值表反映了指令和控制信号的关系 按照真值表就可以实现控制器! # 多周期 CPU # 多周期控制器功能描述方式 有限状态机 采用组合逻辑设计方法用硬连线路 (PLA) 实现 微程序 用 ROM 存放微程序实现 # MIPS 各指令的周期数: 划分关键是: 所有指令均需要 IF,ID,EX 这三个周期 需要访存的指令额外需要一个周期 MEM(ID 内的访存不算,所以只有 Load/Store) 需要写回寄存器的指令额外需要一个周期 WB R-Type...
        2.1k words 2 mins.

        # 思维导图 # 异常和中断的概念 # 异常: 异常来自于 cpu 内部的正在执行的指令本身 # 中断: 中断来自于 cpu 外部的硬件触发 # 统称为 “中断”: 内中断 (内部异常) CPU 自己产生而不通过中断请求线请求,皆为不可屏蔽中断 外中断 (外部中断) 通过中断请求线 INTR 和 NMI 来实现 # 内部 “异常”:CPU 内发生的意外事件或特殊事件 # 按发生原因分为硬故障中断和程序性中断两类 硬故障中断:如电源掉电、硬件线路故障等 程序性中断:执行某条指令时发生的 “异常 (Exception)”,如 溢出 缺页 越界 越权 非法指令 除数为...
        2.7k words 2 mins.

        # 思维导图 # 补码 # 符号部分同原码 数的最高位为符号位,0 表示正数,1 表示负数 # 数字部分与它的符号位有关 对于正数,补码数值部分与原码数值部分相同 对于负数,补码数值部分是将原码数值部分按位取反再加 1 ,即在反码数值部分基础上加 1 # 阶码 # 补码和移码的关系:符号位相反、数值位相同 阶码用移码表示,阶码加减运算后得到补码,再将符号位取反即可得到运算后的阶码 移码的和、差 = 真值的和、差的补码 # 补码 E_1 的 (mod 2^n) n 位移码: [E_1] 移 = 2^(n-1) + E_1 # 补码 E_1 的 (mod 2^n) n...
        \ No newline at end of file diff --git a/page/12/index.html b/page/12/index.html index b252e6ef..85cfe3df 100644 --- a/page/12/index.html +++ b/page/12/index.html @@ -1 +1 @@ -Balloon Party = 風船のパーティー = fuusen no party
        2.1k words 2 mins.

        # 思维导图 # 两大类型 # CISC (1) 指令系统复杂 (2) 指令周期长 (3) 各种指令都能访问存储器 (4) 有专用寄存器(寻址快) (5) 采用微程序控制 (6) 难以进行编译优化生成高效目标代码 (7) 指令长度不等长, 变长指令字 (Expanding Opcodes) # RISC (1) 简化的指令系统 (2) 指令周期短 (3) 有专门的访存指令如 load、store, 以寄存器 - 寄存器方式工作 (三地址指令是 RISC 风格) (4) 采用大量通用寄存器,以减少访存次数 (5) 采用组合逻辑电路控制,不用或少用微程序控制 (6)...
        2k words 2 mins.

        # 思维导图 # 完成单个任务的时间 # 响应时间 (response time): 完成单个任务所需的总时间,由 2 部分组成 响应时间短的场合: 事务处理系统 (存 / 取款的速度要快) CPU 执行时间 (CPU time) 执行时间 (Execution time) CPU 真正花费在程序执行上的时间 用户 CPU 时间: 用来运行用户代码的时间 系统 CPU 时间: 为执行用户程序而需运行一些操作系统代码的时间 其他时间 等待 I/O 操作完成 CPU 花在其他用户程序的时间 # 单位时间完成的任务量 # 吞吐率...
        124 words 1 mins.

        # 思维导图 # 采用二进制表示机器指令和数据 # 硬件系统由 5 大部分组成 # 输入设备 # 控制器 # 运算器 # 存储器 # 输出设备 # 程序和数据预先存放在存储器中,按地址访问存储器 # 操作时根据程序中指令的执行顺序,从存储器中取出指令或数据,由控制器解释执行、运算器完成运算
        1.4k words 1 mins.

        # 定义 最小生成树 Minimum Spanning Tree 问题 等价问题:寻找一个带权无向图中某一点到另一个点的最短路径 / 权最小的路径 注意:当各边有相同权值时,由于选择的随意性,产生的生成树可能不唯一。 # Kruskal (克鲁斯卡尔) 算法 以点为主导,利用最小堆和并查集 对所有边按权值从小到大排序,逐个添加不形成回路的边,直到包含所有的顶点。 # Prim (普里姆) 算法 以边为主导 贪心思想,不断选最小权值的边 从某一顶点出发,选择与它关联的具有最小权值的边,将其顶点加入到生成树顶点集合 U1U_1U1​ 中,剩余顶点集合为...
        12k words 11 mins.

        # 思维导图 # 绪论部分 # 第一章 世界的物质性及发展规律 # 第二章 实践与认识及其发展规律 # 第三章 人类社会及其发展规律 # 第四章 资本主义的本质及规律 # 第六章社会主义的发展及其规律(本章仅供学习参考,不作考试要求) # 第七章 共产主义是人类最崇高的社会理想 # 大纲 # 绪论部分 #...
        32k words 30 mins.

        # 思维导图 # 绪论部分 # 第一章 世界的物质性及发展规律 # 第二章 实践与认识及其发展规律 # 第三章 人类社会及其发展规律 # 第四章 资本主义的本质及规律 # 第五章 资本主义的发展及其趋势 # 第六章社会主义的发展及其规律(本章仅供学习参考,不作考试要求) # 第七章 共产主义是人类最崇高的社会理想 # 大纲 # 绪论部分 #...
        450 words 1 mins.

        # 三种表达式 前缀表达式 / Prefix Notation 波兰式 / Polish Notation 中缀表达式 / Infix Notation (中缀表达式才需要有括号) 后缀表达式 / Postfix Notation 逆波兰式 / Reverse Polish Notation # 三种表达式的相互转化 # 括号法 通用方法,略 # 表达式树法 通用方法,略 # 栈方法 专用于: 中缀 -&gt; 前缀 / 后缀 很像,注意区别,对照记忆 # 中缀 -&gt;...
        2k words 2 mins.

        # 定义 寻找一个带权有向图中某一点到另一个点的最短路径 / 权最小的路径 当然以下算法也适用于无向图 # Floyd (弗洛伊德) 算法 可以算出所有点之间的最短距离 c++//dist 初始化为邻接矩阵,若 i 没有指向 j 的边,则应有 dist [i][j]=MAX_INT (表示无穷大的数)void floyd()&#123; for(int k = 0; k &lt; n; k ++)&#123; // 作为循环中间点的 k 必须放在最外一层循环 for(int i = 0; i &lt; n; i ++)&#123; for(int j...
        448 words 1 mins.

        # 循环队列 对于一个容量为 mmm 的循环队列 实际存储空间为 m+1m+1m+1 初始状态 / 判空条件: rear==frontrear == frontrear==front 判满条件:(rear+1)%m==front(rear+1)\%m==front(rear+1)%m==front 入队操作: 伪代码新元素加入队尾;队尾指针 + 1; 出队操作: 伪代码访问队头元素;队头指针 + 1; 任意时刻队列内包含的元素数量为: (m+rear−front)%m(m+rear-front)\%m(m+rear−front)%m #...
        1.1k words 1 mins.

        # 图的存储结构: AM: Adjacent Matrix 邻接矩阵 AL: Adjacent List 邻接表 RAL: Reverse Adjacent List 逆邻接表 TL: Triple List 三元组顺序表 (往往是行逻辑链接的,性能更好) CL: Cross List 十字链表 # 图按存储结构分为: AMGraph: Adjacent Matrix Graph ALGraph: Adjacent List Graph TLGraph: Triple List Graph RALGraph: Reverse Adjacent List Graph CLGraph:...
        \ No newline at end of file +Balloon Party = 風船のパーティー = fuusen no party
        2.1k words 2 mins.

        # 思维导图 # 两大类型 # CISC (1) 指令系统复杂 (2) 指令周期长 (3) 各种指令都能访问存储器 (4) 有专用寄存器(寻址快) (5) 采用微程序控制 (6) 难以进行编译优化生成高效目标代码 (7) 指令长度不等长, 变长指令字 (Expanding Opcodes) # RISC (1) 简化的指令系统 (2) 指令周期短 (3) 有专门的访存指令如 load、store, 以寄存器 - 寄存器方式工作 (三地址指令是 RISC 风格) (4) 采用大量通用寄存器,以减少访存次数 (5) 采用组合逻辑电路控制,不用或少用微程序控制 (6)...
        2k words 2 mins.

        # 思维导图 # 完成单个任务的时间 # 响应时间 (response time): 完成单个任务所需的总时间,由 2 部分组成 响应时间短的场合: 事务处理系统 (存 / 取款的速度要快) CPU 执行时间 (CPU time) 执行时间 (Execution time) CPU 真正花费在程序执行上的时间 用户 CPU 时间: 用来运行用户代码的时间 系统 CPU 时间: 为执行用户程序而需运行一些操作系统代码的时间 其他时间 等待 I/O 操作完成 CPU 花在其他用户程序的时间 # 单位时间完成的任务量 # 吞吐率...
        124 words 1 mins.

        # 思维导图 # 采用二进制表示机器指令和数据 # 硬件系统由 5 大部分组成 # 输入设备 # 控制器 # 运算器 # 存储器 # 输出设备 # 程序和数据预先存放在存储器中,按地址访问存储器 # 操作时根据程序中指令的执行顺序,从存储器中取出指令或数据,由控制器解释执行、运算器完成运算
        1.4k words 1 mins.

        # 定义 最小生成树 Minimum Spanning Tree 问题 等价问题:寻找一个带权无向图中某一点到另一个点的最短路径 / 权最小的路径 注意:当各边有相同权值时,由于选择的随意性,产生的生成树可能不唯一。 # Kruskal (克鲁斯卡尔) 算法 以点为主导,利用最小堆和并查集 对所有边按权值从小到大排序,逐个添加不形成回路的边,直到包含所有的顶点。 # Prim (普里姆) 算法 以边为主导 贪心思想,不断选最小权值的边 从某一顶点出发,选择与它关联的具有最小权值的边,将其顶点加入到生成树顶点集合 U1U_1U1​ 中,剩余顶点集合为...
        32k words 30 mins.

        # 思维导图 # 绪论部分 # 第一章 世界的物质性及发展规律 # 第二章 实践与认识及其发展规律 # 第三章 人类社会及其发展规律 # 第四章 资本主义的本质及规律 # 第五章 资本主义的发展及其趋势 # 第六章社会主义的发展及其规律(本章仅供学习参考,不作考试要求) # 第七章 共产主义是人类最崇高的社会理想 # 大纲 # 绪论部分 #...
        12k words 11 mins.

        # 思维导图 # 绪论部分 # 第一章 世界的物质性及发展规律 # 第二章 实践与认识及其发展规律 # 第三章 人类社会及其发展规律 # 第四章 资本主义的本质及规律 # 第六章社会主义的发展及其规律(本章仅供学习参考,不作考试要求) # 第七章 共产主义是人类最崇高的社会理想 # 大纲 # 绪论部分 #...
        450 words 1 mins.

        # 三种表达式 前缀表达式 / Prefix Notation 波兰式 / Polish Notation 中缀表达式 / Infix Notation (中缀表达式才需要有括号) 后缀表达式 / Postfix Notation 逆波兰式 / Reverse Polish Notation # 三种表达式的相互转化 # 括号法 通用方法,略 # 表达式树法 通用方法,略 # 栈方法 专用于: 中缀 -&gt; 前缀 / 后缀 很像,注意区别,对照记忆 # 中缀 -&gt;...
        2k words 2 mins.

        # 定义 寻找一个带权有向图中某一点到另一个点的最短路径 / 权最小的路径 当然以下算法也适用于无向图 # Floyd (弗洛伊德) 算法 可以算出所有点之间的最短距离 c++//dist 初始化为邻接矩阵,若 i 没有指向 j 的边,则应有 dist [i][j]=MAX_INT (表示无穷大的数)void floyd()&#123; for(int k = 0; k &lt; n; k ++)&#123; // 作为循环中间点的 k 必须放在最外一层循环 for(int i = 0; i &lt; n; i ++)&#123; for(int j...
        448 words 1 mins.

        # 循环队列 对于一个容量为 mmm 的循环队列 实际存储空间为 m+1m+1m+1 初始状态 / 判空条件: rear==frontrear == frontrear==front 判满条件:(rear+1)%m==front(rear+1)\%m==front(rear+1)%m==front 入队操作: 伪代码新元素加入队尾;队尾指针 + 1; 出队操作: 伪代码访问队头元素;队头指针 + 1; 任意时刻队列内包含的元素数量为: (m+rear−front)%m(m+rear-front)\%m(m+rear−front)%m #...
        1.1k words 1 mins.

        # 图的存储结构: AM: Adjacent Matrix 邻接矩阵 AL: Adjacent List 邻接表 RAL: Reverse Adjacent List 逆邻接表 TL: Triple List 三元组顺序表 (往往是行逻辑链接的,性能更好) CL: Cross List 十字链表 # 图按存储结构分为: AMGraph: Adjacent Matrix Graph ALGraph: Adjacent List Graph TLGraph: Triple List Graph RALGraph: Reverse Adjacent List Graph CLGraph:...
        \ No newline at end of file diff --git a/page/13/index.html b/page/13/index.html index 0c365738..f8bc50ed 100644 --- a/page/13/index.html +++ b/page/13/index.html @@ -1 +1 @@ -Balloon Party = 風船のパーティー = fuusen no party
        3.1k words 3 mins.

        # AOV: Activity On Vertex 略(不考) # AOE: Activity On Edge 与 AOV 网相对应的是 AOE (Activity On Edge) ,是边表示活动的有向无环图,如图所示。图中顶点表示事件 (Event),每个事件表示在其前的所有活动已经完成,其后的活动可以开始;弧表示活动,弧上的权值表示相应活动所需的时间或费用 与 AOE 有关的研究问题 完成整个工程至少需要多少时间? 哪些活动是影响工程进度 (费用) 的关键? 工程完成最短时间:从起点到终点的最长路径长度...
        12k words 11 mins.

        # Scala # import 在 Scala 中,import 语句用于引入其他类、对象或特质,以便在当前作用域中可以直接访问它们的成员。下面是一些关于 Scala 中 import 的常见用法: 基本用法 a// 导入单个类import scala.collection.mutable.ListBuffer// 导入整个包import scala.collection.mutable._// 重命名导入的类 / 对象 / 特质import java.util.&#123;ArrayList => JArrayList, HashMap =>...
        1k words 1 mins.

        # Multiway Tree 多叉树任意节点的所有子树也都是有顺序的 多叉树可以按固定的方法向二叉树转化: 对所有节点只保留与最左侧子节点的连接,删除其他连接 从左到右依次连接同一层的兄弟节点 (siblings) # B-Tree Banlanced Multiway Tree, 顾名思义,B - 树是一种平衡的多路查找树,是平衡二叉树的外延概念。 B-Tree 是为外部查找(如磁盘)设计的一种平衡查找树。 系统从磁盘读取数据到内存时是以存储块 (block) 为基本单位的,位于同一个磁盘块中的数据会被一次性读取出来,而不是需要什么取什么。B-Tree...
        9.5k words 9 mins.

        # 二叉树的一些规范 (考研) 节点的度数:节点出发指向下一层节点的边的数量 (出度) N0,N1,N2N_0,N_1,N_2N0​,N1​,N2​:分别表示度数为 1,2,3 的顶点数量 # 结论 任意二叉树满足: n=N0+N1+N2n=N_0+N_1+N_2n=N0​+N1​+N2​ n−1=0×N0+1×N1+2×N2=N1+2N2n-1=0\times N_0+1\times N_1+2\times...
        5.4k words 5 mins.

        # 哈希表 / 散列表 应用哈希函数,由记录的关键字确定记录在表中的地址,并将记录放入此地址,这样构成的表叫哈希表 # 哈希查找 / 散列查找 利用哈希函数进行查找的过程叫哈希查找 # 哈希映射函数 / 散列函数 在记录的关键字与记录的存储地址之间建立的一种对应关系叫哈希函数 # 原始 hash 函数 简单一点的 hash 算法就是直接取模 (除留余数法)。 高级一点的 hash 算法会模乘以一个大素数来将源数据的分布规律变得不那么显著,也就是说,不管输入的是一串分布很紧凑的数字序列还是很稀疏的数字序列,生成的结果都是非常分散的数字序列。 linux 内核代码里面的这个 hash...
        1.5k words 1 mins.

        基排序 / 桶排序 / 数字排序 # 原理 需要频繁进行分组重排,(类似于归并)适用于链表 可分为两种: MSD,最高位优先 (Most Significant Digit first) LSD,最低位优先 (Least Significant Digit first) # 代码 c++//Definition for singly-linked list.struct ListNode &#123; int val; ListNode *next; ListNode() : val(0), next(nullptr) &#123;&#125;...
        3.8k words 3 mins.

        区分:堆排序的堆不是堆栈的堆 # 原理 堆排序可以视为一种聪明的选择排序。 堆排序适合数组这样的数据结构。 选择排序需要不断在未排序部分找最大 / 小值然后放在已排序部分后面,堆排序只是将 “找最值” 的部分用 “调整堆” 代替,调整好的堆的堆顶就是我们想要的最值。 堆排序基于二叉树结构,堆是一种按一定规律存储数据的二叉树。利用堆最值位于堆顶的特点,不断将堆顶的最值和堆最后一个元素交换(堆最后一个元素后面都是已排序部分),然后重新调整未排序部分使其再变回一个堆。 # 最大堆 / 小根堆 / 小顶堆 最大值在根节点 / 堆顶 结点的键值都小于等于其父结点的键值 排降序建最大堆 # 最小堆...
        \ No newline at end of file +Balloon Party = 風船のパーティー = fuusen no party
        3.1k words 3 mins.

        # AOV: Activity On Vertex 略(不考) # AOE: Activity On Edge 与 AOV 网相对应的是 AOE (Activity On Edge) ,是边表示活动的有向无环图,如图所示。图中顶点表示事件 (Event),每个事件表示在其前的所有活动已经完成,其后的活动可以开始;弧表示活动,弧上的权值表示相应活动所需的时间或费用 与 AOE 有关的研究问题 完成整个工程至少需要多少时间? 哪些活动是影响工程进度 (费用) 的关键? 工程完成最短时间:从起点到终点的最长路径长度...
        12k words 11 mins.

        # Scala # import 在 Scala 中,import 语句用于引入其他类、对象或特质,以便在当前作用域中可以直接访问它们的成员。下面是一些关于 Scala 中 import 的常见用法: 基本用法 a// 导入单个类import scala.collection.mutable.ListBuffer// 导入整个包import scala.collection.mutable._// 重命名导入的类 / 对象 / 特质import java.util.&#123;ArrayList => JArrayList, HashMap =>...
        1k words 1 mins.

        # Multiway Tree 多叉树任意节点的所有子树也都是有顺序的 多叉树可以按固定的方法向二叉树转化: 对所有节点只保留与最左侧子节点的连接,删除其他连接 从左到右依次连接同一层的兄弟节点 (siblings) # B-Tree Banlanced Multiway Tree, 顾名思义,B - 树是一种平衡的多路查找树,是平衡二叉树的外延概念。 B-Tree 是为外部查找(如磁盘)设计的一种平衡查找树。 系统从磁盘读取数据到内存时是以存储块 (block) 为基本单位的,位于同一个磁盘块中的数据会被一次性读取出来,而不是需要什么取什么。B-Tree...
        9.5k words 9 mins.

        # 二叉树的一些规范 (考研) 节点的度数:节点出发指向下一层节点的边的数量 (出度) N0,N1,N2N_0,N_1,N_2N0​,N1​,N2​:分别表示度数为 1,2,3 的顶点数量 # 结论 任意二叉树满足: n=N0+N1+N2n=N_0+N_1+N_2n=N0​+N1​+N2​ n−1=0×N0+1×N1+2×N2=N1+2N2n-1=0\times N_0+1\times N_1+2\times...
        5.4k words 5 mins.

        # 哈希表 / 散列表 应用哈希函数,由记录的关键字确定记录在表中的地址,并将记录放入此地址,这样构成的表叫哈希表 # 哈希查找 / 散列查找 利用哈希函数进行查找的过程叫哈希查找 # 哈希映射函数 / 散列函数 在记录的关键字与记录的存储地址之间建立的一种对应关系叫哈希函数 # 原始 hash 函数 简单一点的 hash 算法就是直接取模 (除留余数法)。 高级一点的 hash 算法会模乘以一个大素数来将源数据的分布规律变得不那么显著,也就是说,不管输入的是一串分布很紧凑的数字序列还是很稀疏的数字序列,生成的结果都是非常分散的数字序列。 linux 内核代码里面的这个 hash...
        1.5k words 1 mins.

        基排序 / 桶排序 / 数字排序 # 原理 需要频繁进行分组重排,(类似于归并)适用于链表 可分为两种: MSD,最高位优先 (Most Significant Digit first) LSD,最低位优先 (Least Significant Digit first) # 代码 c++//Definition for singly-linked list.struct ListNode &#123; int val; ListNode *next; ListNode() : val(0), next(nullptr) &#123;&#125;...
        3.8k words 3 mins.

        区分:堆排序的堆不是堆栈的堆 # 原理 堆排序可以视为一种聪明的选择排序。 堆排序适合数组这样的数据结构。 选择排序需要不断在未排序部分找最大 / 小值然后放在已排序部分后面,堆排序只是将 “找最值” 的部分用 “调整堆” 代替,调整好的堆的堆顶就是我们想要的最值。 堆排序基于二叉树结构,堆是一种按一定规律存储数据的二叉树。利用堆最值位于堆顶的特点,不断将堆顶的最值和堆最后一个元素交换(堆最后一个元素后面都是已排序部分),然后重新调整未排序部分使其再变回一个堆。 # 最大堆 / 小根堆 / 小顶堆 最大值在根节点 / 堆顶 结点的键值都小于等于其父结点的键值 排降序建最大堆 # 最小堆...
        \ No newline at end of file diff --git a/page/2/index.html b/page/2/index.html index b9b91cb3..75bd8ae8 100644 --- a/page/2/index.html +++ b/page/2/index.html @@ -1 +1 @@ -Balloon Party = 風船のパーティー = fuusen no party
        2.5k words 2 mins.

        今天因为校青马班的活动又跑到南校开会、参观团一大。中涂我也不能用电脑,于是无聊之际,手机打开了一直感兴趣的《饿殍:明末千里行》的游戏录播,看了前面就上头了。于是一直到凌晨 3 点,打完了 2 个 TE ,非常有感触,于是一边听着 ED 写下本篇文章作记录。(即使第二天有早八我也不管了!) 这是我玩的第二篇国产的国产文字冒险游戏。(上一部是《OPUS: 龙脉长歌》,这两部观感都非常好)《饿殍》难怪这么火,是有些原因的: 首先作为一个全年龄向的游戏,受众确实更加广泛 人物立绘、CG...
        4.3k words 4 mins.

        # 匹配 # 匹配基础概念 匹配 MMM —— 如果 MMM 是图 GGG 的边子集 (不含环),且 MMM 中的任意两条边没有共同顶点,则称 MMM 是 GGG 的一个匹配,或对集,或边独立集。 如果 GGG 中顶点是 GGG 的匹配 MMM 中某条边的端点,称它为 MMM 饱和点,否则为 MMM 非饱和点。 最大匹配 MMM (无向图定义) —— 如果 MMM 是图 GGG 的包含边数最多的匹配,称 MMM 是 GGG 的一个最大匹配。 特别是,若最大匹配饱和了 GGG 的所有顶点,称它为 GGG 的一个完美匹配。 MMM 交错路 —— 如果 MMM 是图 GGG 的匹配, GGG...
        2.1k words 2 mins.

        两类主要的半导体存储器是 RAM 和 ROM。 RAM (随机访问存储器) 类型的存储器在等量时间内可以访问所有的地址,并且可以按照任何顺序来选择这些地址以执行读或者写操作。所有的 RAM 都具有读和写的能力。因为 RAM 在电源关闭时会丢失所存储的数据,所以它是易失性内存。 RAM 系列 RAM 的两种类别是静 RAM (SRAM) 和动态 RAM (DRAM) 。当直流电源移走后, SRAM 和 DRAM 都会丢失存储的数据,因此被归类为易失性存储器。 静态 RAM 通常使用锁存器作为存储元件,因而只要加上直流电源,就可以永远地存储数据。 异步 SRAM (ASRAM) 同步...
        6.7k words 6 mins.

        冯・诺伊曼计算机体系决定了操作系统需要内存管理 # 虚拟内存物理实现 连续分配 单一连续分配 分区 固定分区分配 动态分区分配 非连续分配 段式 页式 段页式 ⭐️内部碎片和外部碎片 内部碎片 是已经被分配给某个进程的,该进程没有利用的 固定分区分配产生这种碎片 外部碎片 是无法被分配给任何进程的,不属于任何进程的 动态分区分配产生这种碎片 可以通过紧凑技术克服,需要动态重定位寄存器硬件支持 # 分区 (连续分配) #...
        9.7k words 9 mins.

        # 环的相关定义 设 RRR 是非空集合,如果在 RRR 上定义了两个二元运算 “+++”(称为加法)和 “ ⋅\cdot⋅ ”(称为乘法),且满足: (1) RRR 关于加法 +++ 构成交换群(即加群) (2) RRR 关于乘法 ⋅\cdot⋅ 构成半群,即乘法满足结合律:∀a,b,c∈R,  a⋅(b⋅c)=(a⋅b)⋅c\forall a,b,c\in R,\ \ a\cdot(b\cdot c)=(a\cdot b)\cdot c∀a,b,c∈R,  a⋅(b⋅c)=(a⋅b)⋅c; (3) 乘法...
        4.2k words 4 mins.

        # 欧拉图 # 定义 欧拉图存在欧拉闭迹,半欧拉图只存在欧拉迹 若能遍历完所有的边但是没法回到起始点,称为非欧拉图但是有欧拉迹 # 欧拉图 假定 GGG 是一个连通图,且是欧拉图,则下列命题等价: GGG 是欧拉图 GGG 的每个点的度数是偶数 GGG 的边集能划分为边不重的圈的并 (GGG 存在欧拉闭迹) # 半欧拉图 假定 GGG 是一个连通图,且是半欧拉图,则下列命题等价: GGG 是半欧拉图 GGG 有且仅有 2 个点的度数是奇数 (GGG 存在欧拉迹) # 小结论💡 当 nnn 满足什么条件时,完全图 KnK_nKn​ 是欧拉图? nnn 为奇数 当 nnn...
        304 words 1 mins.

        # 串行输入 / 串行输出 # 串行输入 / 并行输出 # 并行输入 / 串行输出 # 并行输入 / 并行输出 最简单 # 双向移位寄存器 # 74HC194 通用 4 位双向移位寄存器 # 移位寄存器计数器 # 约翰逊计数器 一般来说,约翰逊计 数器将会产生模 2n2n2n ,其中 nnn 是计数器中级的个数 # 环形计数器 # 移位寄存器应用 # 精确延时控制 串行输入 / 串行输出移位寄存器可以用来提供从输入到输出的时间延迟,此时间延迟是寄存器中数 nnn...
        2.9k words 3 mins.

        PVE 版本:8.1.4 ,显卡:NVIDIA GTX 745 # 宿主机配置 # IOMMU shellnano /etc/default/grub将 GRUB_CMDLINE_LINUX_DEFAULT=&quot;...&quot; 中的内容修改为如下: shellGRUB_CMDLINE_LINUX_DEFAULT="quiet intel_iommu=on iommu=pt video=efifb:off"准确来说是在 &quot;...&quot; 原来的内容后面增加这些内容,其中: quiet...
        \ No newline at end of file +Balloon Party = 風船のパーティー = fuusen no party
        2.5k words 2 mins.

        今天因为校青马班的活动又跑到南校开会、参观团一大。中涂我也不能用电脑,于是无聊之际,手机打开了一直感兴趣的《饿殍:明末千里行》的游戏录播,看了前面就上头了。于是一直到凌晨 3 点,打完了 2 个 TE ,非常有感触,于是一边听着 ED 写下本篇文章作记录。(即使第二天有早八我也不管了!) 这是我玩的第二篇国产的国产文字冒险游戏。(上一部是《OPUS: 龙脉长歌》,这两部观感都非常好)《饿殍》难怪这么火,是有些原因的: 首先作为一个全年龄向的游戏,受众确实更加广泛 人物立绘、CG...
        4.3k words 4 mins.

        # 匹配 # 匹配基础概念 匹配 MMM —— 如果 MMM 是图 GGG 的边子集 (不含环),且 MMM 中的任意两条边没有共同顶点,则称 MMM 是 GGG 的一个匹配,或对集,或边独立集。 如果 GGG 中顶点是 GGG 的匹配 MMM 中某条边的端点,称它为 MMM 饱和点,否则为 MMM 非饱和点。 最大匹配 MMM (无向图定义) —— 如果 MMM 是图 GGG 的包含边数最多的匹配,称 MMM 是 GGG 的一个最大匹配。 特别是,若最大匹配饱和了 GGG 的所有顶点,称它为 GGG 的一个完美匹配。 MMM 交错路 —— 如果 MMM 是图 GGG 的匹配, GGG...
        2.1k words 2 mins.

        两类主要的半导体存储器是 RAM 和 ROM。 RAM (随机访问存储器) 类型的存储器在等量时间内可以访问所有的地址,并且可以按照任何顺序来选择这些地址以执行读或者写操作。所有的 RAM 都具有读和写的能力。因为 RAM 在电源关闭时会丢失所存储的数据,所以它是易失性内存。 RAM 系列 RAM 的两种类别是静 RAM (SRAM) 和动态 RAM (DRAM) 。当直流电源移走后, SRAM 和 DRAM 都会丢失存储的数据,因此被归类为易失性存储器。 静态 RAM 通常使用锁存器作为存储元件,因而只要加上直流电源,就可以永远地存储数据。 异步 SRAM (ASRAM) 同步...
        6.7k words 6 mins.

        冯・诺伊曼计算机体系决定了操作系统需要内存管理 # 虚拟内存物理实现 连续分配 单一连续分配 分区 固定分区分配 动态分区分配 非连续分配 段式 页式 段页式 ⭐️内部碎片和外部碎片 内部碎片 是已经被分配给某个进程的,该进程没有利用的 固定分区分配产生这种碎片 外部碎片 是无法被分配给任何进程的,不属于任何进程的 动态分区分配产生这种碎片 可以通过紧凑技术克服,需要动态重定位寄存器硬件支持 # 分区 (连续分配) #...
        9.7k words 9 mins.

        # 环的相关定义 设 RRR 是非空集合,如果在 RRR 上定义了两个二元运算 “+++”(称为加法)和 “ ⋅\cdot⋅ ”(称为乘法),且满足: (1) RRR 关于加法 +++ 构成交换群(即加群) (2) RRR 关于乘法 ⋅\cdot⋅ 构成半群,即乘法满足结合律:∀a,b,c∈R,  a⋅(b⋅c)=(a⋅b)⋅c\forall a,b,c\in R,\ \ a\cdot(b\cdot c)=(a\cdot b)\cdot c∀a,b,c∈R,  a⋅(b⋅c)=(a⋅b)⋅c; (3) 乘法...
        4.2k words 4 mins.

        # 欧拉图 # 定义 欧拉图存在欧拉闭迹,半欧拉图只存在欧拉迹 若能遍历完所有的边但是没法回到起始点,称为非欧拉图但是有欧拉迹 # 欧拉图 假定 GGG 是一个连通图,且是欧拉图,则下列命题等价: GGG 是欧拉图 GGG 的每个点的度数是偶数 GGG 的边集能划分为边不重的圈的并 (GGG 存在欧拉闭迹) # 半欧拉图 假定 GGG 是一个连通图,且是半欧拉图,则下列命题等价: GGG 是半欧拉图 GGG 有且仅有 2 个点的度数是奇数 (GGG 存在欧拉迹) # 小结论💡 当 nnn 满足什么条件时,完全图 KnK_nKn​ 是欧拉图? nnn 为奇数 当 nnn...
        304 words 1 mins.

        # 串行输入 / 串行输出 # 串行输入 / 并行输出 # 并行输入 / 串行输出 # 并行输入 / 并行输出 最简单 # 双向移位寄存器 # 74HC194 通用 4 位双向移位寄存器 # 移位寄存器计数器 # 约翰逊计数器 一般来说,约翰逊计 数器将会产生模 2n2n2n ,其中 nnn 是计数器中级的个数 # 环形计数器 # 移位寄存器应用 # 精确延时控制 串行输入 / 串行输出移位寄存器可以用来提供从输入到输出的时间延迟,此时间延迟是寄存器中数 nnn...
        2.9k words 3 mins.

        PVE 版本:8.1.4 ,显卡:NVIDIA GTX 745 # 宿主机配置 # IOMMU shellnano /etc/default/grub将 GRUB_CMDLINE_LINUX_DEFAULT=&quot;...&quot; 中的内容修改为如下: shellGRUB_CMDLINE_LINUX_DEFAULT="quiet intel_iommu=on iommu=pt video=efifb:off"准确来说是在 &quot;...&quot; 原来的内容后面增加这些内容,其中: quiet...
        \ No newline at end of file diff --git a/page/3/index.html b/page/3/index.html index 34619def..6eada0eb 100644 --- a/page/3/index.html +++ b/page/3/index.html @@ -1 +1 @@ -Balloon Party = 風船のパーティー = fuusen no party
        3.3k words 3 mins.

        # Intro 死锁的必要条件 互斥:一次只有一个进程使用一个资源,其他进程不能访问分配给其他进程的资源 占有等待 (持有且等待):当一个进程等待其他进程时,继续占有已分配的资源 非抢占:不能强行抢占进程已占有的资源 循环等待:存在一个闭合的进程链,每个进程至少占有此链中下一个进程所需的一个资源前三个只会导致死锁的可能性,而最后一个循环等待是导致死锁的 “最终极条件” 处理死锁的方法: 确保系统永远不会进入死锁状态 锁预防,在 OS 设计阶段就设计成不会出现死锁 死锁避免,在 OS...
        46k words 42 mins.

        年轻人的第一场 CTF ,学到了很多,比较走运拿了第五。最遗憾的是密码学板块了,认真学了不少抽象代数但还是没有做出几题。 队友们:(我起了个整活队名,其实是抄袭另一个叫做 “憧憬成为 CTFer” 的 ) # AI # Network Reverse 网络结构长这样: Sequential( (0): Conv2d(3, 4, kernel_size=(2, 2), stride=(1, 1)) (1): GELU(approximate='none') (2): Conv2d(4, 8, kernel_size=(5, 5), stride=(1, 1)) (3):...
        15k words 14 mins.

        # 公式表 # 基本变换对 # 性质表 # 双边拉普拉斯变换 # 定义 X(s)=∫−∞+∞x(t)e−stdtX(s)=\int_{-\infin}^{+\infin}x(t)e^{-st}dt X(s)=∫−∞+∞​x(t)e−stdt 称为 x(t)x(t)x(t) 的拉普拉斯变换,后面简记为 LT,其中 s=σ+jωs=\sigma+j\omegas=σ+jω 为复数。 若 σ=0\sigma=0σ=0 ,s=jωs=j\omegas=jω 为纯虚数则为傅里叶变换,也就是说连续时间傅里叶变换是双边拉普拉斯变换在 𝜎=0...
        264 words 1 mins.

        # MP 正向传播 反向传播 https://www.cnblogs.com/charlotte77/p/5629865.html 感知机训练 BP 算法 # CNN 最大池化 max pooling 可以是取最大值、平均值。保留区域的特征,降维降低计算复杂度 # RNN 在 CNN 的基础上,增加了记忆功能 # LSTM 长短期记忆网络 (Long Short-term Memory) # BPTT (Backpropagation through time) # Transformer 通过最小化交叉熵 (minimize cross entropy) 训练 # Encoder #...
        6.2k words 6 mins.

        # 割边 # 严格定义 如果 ω(G−e)&gt;ω(G)\omega(G-e)\gt\omega(G)ω(G−e)&gt;ω(G) ,则 eee 为 GGG 的一条割边或桥。 # 充要条件 eee 是 GGG 的割边当且仅当 eee 不在 GGG 的任何圈中。 必要性: 假设 eee 是 GGG 的割边且 eee 在 GGG 的某圈中 充分性: 假设 eee 不在 GGG 的任何圈中且 eee 不是 GGG 的割边,则 G−eG-eG−e 连通。 于是 G−eG-eG−e 中存在一条 (u,v)(u,v)(u,v) 路 lll ,显然...
        1.6k words 1 mins.

        SageMath 🔧 一个非常好用的工具 # 扩展维纳攻击 当私钥过小时,能够对 NNN 进行分解。维纳证明了: d&lt;13N14(q&lt;p&lt;2q)d\lt\frac{1}{3}N^{1\over 4}\\ (q\lt p\lt 2q) d&lt;31​N41​(q&lt;p&lt;2q) 的时候,一定能够对 NNN 进行分解。 # 格规约 # LLL Lenstra–Lenstra–Lovász 基于格的约简算法,它将输入矩阵转换为其最简形式。 使用 LLL...
        8.8k words 8 mins.

        # 概述 # 机器学习范式 监督学习(归纳学习) 给定训练数据 training data 和期望的输出(标签)desired outputs (with labels) 半监督学习 给定训练数据 training data 和部分期望的输出 some labels 无监督学习 仅给定训练数据 training data without labels 强化学习 Reinforce Learning (RL) 从序列决策中获取奖励 observations and periodic rewards as the agent takes sequential actions; #...
        2.5k words 2 mins.

        部署了一个 overleaf 玩,目的需求是做一个共享文档,作为腾讯文档、知识库的替代品 注册管理员账号的时候,出现了报错: alertSession error. Please check you have cookies enabled. If the problem persists, try clearing your cache and cookies.查阅这个 issue 后,得到了解决方案,将 config/variables.env 这里注释掉: variables.env # SHARELATEX_BEHIND_PROXY=true#...
        4.6k words 4 mins.

        # 群的同态 # 定义 设 GGG 与 G′G&#x27;G′ 是两个群,函数 φ:G→G′φ:G→G&#x27;φ:G→G′ 如果满足: ∀a,b∈G,  φ(ab)=φ(a)φ(b)∀a,b\in G,\ \ φ(ab)= φ(a)φ(b) ∀a,b∈G,  φ(ab)=φ(a)φ(b) 则 φφφ 是群 GGG 到 G′G&#x27;G′ 的同态 (homomorphism)(一个函数) φ(a)φ(b)φ(a)φ(b)φ(a)φ(b) 是 G′G&#x27;G′ 中的运算,而 ababab 是 GGG 中的运算 考察同态...
        \ No newline at end of file +Balloon Party = 風船のパーティー = fuusen no party
        3.3k words 3 mins.

        # Intro 死锁的必要条件 互斥:一次只有一个进程使用一个资源,其他进程不能访问分配给其他进程的资源 占有等待 (持有且等待):当一个进程等待其他进程时,继续占有已分配的资源 非抢占:不能强行抢占进程已占有的资源 循环等待:存在一个闭合的进程链,每个进程至少占有此链中下一个进程所需的一个资源前三个只会导致死锁的可能性,而最后一个循环等待是导致死锁的 “最终极条件” 处理死锁的方法: 确保系统永远不会进入死锁状态 锁预防,在 OS 设计阶段就设计成不会出现死锁 死锁避免,在 OS...
        46k words 42 mins.

        年轻人的第一场 CTF ,学到了很多,比较走运拿了第五。最遗憾的是密码学板块了,认真学了不少抽象代数但还是没有做出几题。 队友们:(我起了个整活队名,其实是抄袭另一个叫做 “憧憬成为 CTFer” 的 ) # AI # Network Reverse 网络结构长这样: Sequential( (0): Conv2d(3, 4, kernel_size=(2, 2), stride=(1, 1)) (1): GELU(approximate='none') (2): Conv2d(4, 8, kernel_size=(5, 5), stride=(1, 1)) (3):...
        15k words 14 mins.

        # 公式表 # 基本变换对 # 性质表 # 双边拉普拉斯变换 # 定义 X(s)=∫−∞+∞x(t)e−stdtX(s)=\int_{-\infin}^{+\infin}x(t)e^{-st}dt X(s)=∫−∞+∞​x(t)e−stdt 称为 x(t)x(t)x(t) 的拉普拉斯变换,后面简记为 LT,其中 s=σ+jωs=\sigma+j\omegas=σ+jω 为复数。 若 σ=0\sigma=0σ=0 ,s=jωs=j\omegas=jω 为纯虚数则为傅里叶变换,也就是说连续时间傅里叶变换是双边拉普拉斯变换在 𝜎=0...
        264 words 1 mins.

        # MP 正向传播 反向传播 https://www.cnblogs.com/charlotte77/p/5629865.html 感知机训练 BP 算法 # CNN 最大池化 max pooling 可以是取最大值、平均值。保留区域的特征,降维降低计算复杂度 # RNN 在 CNN 的基础上,增加了记忆功能 # LSTM 长短期记忆网络 (Long Short-term Memory) # BPTT (Backpropagation through time) # Transformer 通过最小化交叉熵 (minimize cross entropy) 训练 # Encoder #...
        6.2k words 6 mins.

        # 割边 # 严格定义 如果 ω(G−e)&gt;ω(G)\omega(G-e)\gt\omega(G)ω(G−e)&gt;ω(G) ,则 eee 为 GGG 的一条割边或桥。 # 充要条件 eee 是 GGG 的割边当且仅当 eee 不在 GGG 的任何圈中。 必要性: 假设 eee 是 GGG 的割边且 eee 在 GGG 的某圈中 充分性: 假设 eee 不在 GGG 的任何圈中且 eee 不是 GGG 的割边,则 G−eG-eG−e 连通。 于是 G−eG-eG−e 中存在一条 (u,v)(u,v)(u,v) 路 lll ,显然...
        1.6k words 1 mins.

        SageMath 🔧 一个非常好用的工具 # 扩展维纳攻击 当私钥过小时,能够对 NNN 进行分解。维纳证明了: d&lt;13N14(q&lt;p&lt;2q)d\lt\frac{1}{3}N^{1\over 4}\\ (q\lt p\lt 2q) d&lt;31​N41​(q&lt;p&lt;2q) 的时候,一定能够对 NNN 进行分解。 # 格规约 # LLL Lenstra–Lenstra–Lovász 基于格的约简算法,它将输入矩阵转换为其最简形式。 使用 LLL...
        8.8k words 8 mins.

        # 概述 # 机器学习范式 监督学习(归纳学习) 给定训练数据 training data 和期望的输出(标签)desired outputs (with labels) 半监督学习 给定训练数据 training data 和部分期望的输出 some labels 无监督学习 仅给定训练数据 training data without labels 强化学习 Reinforce Learning (RL) 从序列决策中获取奖励 observations and periodic rewards as the agent takes sequential actions; #...
        2.5k words 2 mins.

        部署了一个 overleaf 玩,目的需求是做一个共享文档,作为腾讯文档、知识库的替代品 注册管理员账号的时候,出现了报错: alertSession error. Please check you have cookies enabled. If the problem persists, try clearing your cache and cookies.查阅这个 issue 后,得到了解决方案,将 config/variables.env 这里注释掉: variables.env # SHARELATEX_BEHIND_PROXY=true#...
        4.6k words 4 mins.

        # 群的同态 # 定义 设 GGG 与 G′G&#x27;G′ 是两个群,函数 φ:G→G′φ:G→G&#x27;φ:G→G′ 如果满足: ∀a,b∈G,  φ(ab)=φ(a)φ(b)∀a,b\in G,\ \ φ(ab)= φ(a)φ(b) ∀a,b∈G,  φ(ab)=φ(a)φ(b) 则 φφφ 是群 GGG 到 G′G&#x27;G′ 的同态 (homomorphism)(一个函数) φ(a)φ(b)φ(a)φ(b)φ(a)φ(b) 是 G′G&#x27;G′ 中的运算,而 ababab 是 GGG 中的运算 考察同态...
        \ No newline at end of file diff --git a/page/4/index.html b/page/4/index.html index 0c8e1101..f0518bd3 100644 --- a/page/4/index.html +++ b/page/4/index.html @@ -1 +1 @@ -Balloon Party = 風船のパーティー = fuusen no party
        7.3k words 7 mins.

        # 树的基本性质 # 树的叶子数 设 TTT 为 (n,m)(n,m)(n,m) 树,TTT 中有 nin_ini​ 个度为 i,(1≤i≤k)i,(1\leq i\leq k)i,(1≤i≤k) 的点,且有 ∑ni=n\sum n_i=n∑ni​=n ,则有:n1=2+n3+2n4+⋯+(k−2)nkn_1=2+n_3+2n_4+\cdots+(k-2)n_kn1​=2+n3​+2n4​+⋯+(k−2)nk​。 证明:由 m=n−1m=n-1m=n−1...
        6k words 5 mins.

        # 证书相关文件格式 证书 (Certificate) - .cer 、 .crt 私钥 (Private Key) - .key 证书签名请求 (Certificate signing request) - .csr 证书吊销列表 (Certificate Revocation List) - .crl 以上除了 .crl 均可以使用这两种编码方式,从而有这样的后缀: base64 编码 - .pem 二进制编码 (少见) - .der 为了给我们的网页应用使用 https 协议,我们需要得到有效的第三方认证的证书以及对应的私钥。 CA,certificate...
        3.6k words 3 mins.

        # 智能体 # 定义 Agent 是指驻留在某一环境下能够自主(Autonomous)、灵活(Flexible)地执行动作以满足设计目标的行为实体。 上述定义具有如下二个特点。 (1) 定义方式 : Agent 概念定义是基于 Agent 的外部可观察行为特征,而不是其内部的结构 (2) 抽象层次 : Agent 概念更加贴近于人们对现实世界(而不是计算机世界)中行为实体的理解 # 特点 # 驻留性 # 自主性 智能体能在没有人类或其他智能体的干涉和指导的情况下运行,并能根据其内部状态和感知到的环境信息来决定该实施什么样的动作,进而控制自身的行为 #...
        14k words 13 mins.

        # 公式表 # 基本傅里叶变换对 # 性质 # 非周期信号 # FT 公式推导 定义 x~[n]\widetilde{x}[n]x[n] 为 x[n]x[n]x[n] 的周期延拓,以 NNN 为周期,x[n]x[n]x[n] 是 x~[n]\widetilde{x}[n]x[n] 的一个周期。 当 T→∞T\to\infinT→∞ 时,在 (−∞,+∞)(-\infin,+\infin)(−∞,+∞) 内 x~[n]=x[n]\widetilde{x}[n]=x[n]x[n]=x[n] ;在该极限情况下有...
        2.3k words 2 mins.

        # 归纳总结 缩写: CTFS—— 连续时间傅里叶级数 DTFS—— 离散时间傅里叶级数 CTFT—— 连续时间傅里叶变换 DTFT—— 离散时间傅里叶变换 一图归纳: 信号在时域的特性和在频域的特性之间存在以下对应关系: 时域的周期性 —— 频域的离散性 时域的非周期性 —— 频域的连续性 时域的离散性 —— 频域的周期性 时域的连续性 —— 频域的非周期性 # CTFT 的对偶 若 x(t)⟷CTFTX(jω)x(t)\stackrel{CTFT}{\longleftrightarrow}X(j\omega)x(t)⟷CTFT​X(jω)...
        8.4k words 8 mins.

        # 调度相关概念 为什么需要调度? 释放被阻塞、挂起的进程占用的 CPU 资源 实现 I/O 操作和 CPU 并行执行( I/O 密集型和 CPU 密集型) 调度的层次、类型 处理器 / CPU 调度⭐️(进程调度的主要内容) 作业调度 /long-term scheduling / 长程调度 / 高级调度 The decision to add to the pool of processes to be executed 内存调度 /medium-term scheduling / 中程调度 / 中级调度 The decision to add to the number of...
        8.8k words 8 mins.

        # 正规子群 # 定义 设 HHH 是群 GGG 的子群 (⚠️得先是子群),如果 ∀a∈G∀a\in G∀a∈G 都有 Ha=aHHa=aHHa=aH ,则称 HHH 是 GGG 的正规子群 (normal subgroup) 或不变子群 (invariant subgroup),记作: H⊴GH\unlhd G H⊴G 条件 Ha=aHHa=aHHa=aH 仅表示两个集合 HaHaHa 与 aHaHaH 相等。不能错误地认为,由 Ha=aHHa=aHHa=aH 可推出对 HHH 的任意元素 hhh 有 ha=ahha=ahha=ah。 正确的理解是:对任意...
        4.9k words 4 mins.

        用邻接矩阵或关联矩阵表示图,称为图的代数表示。用矩阵表示图,主要有两个优点: (1) 能够把图输入到计算机中 (2) 可以用代数方法研究图论 Spectral Graph Theory 是研究图和代数的一个领域 # 邻接矩阵 Adjaceny matrix is n×nn\times nn×n symmetric matrix # 定义 设 GGG 为 nnn 阶图,V={v1,z2,⋯ ,vn}V=\{v_1,z_2,\cdots,v_n\}V={v1​,z2​,⋯,vn​} ,邻接矩阵 A(G)=(aij)A(G)=...
        1.4k words 1 mins.

        This article isn't finished yet... Graph Partitioning Bi-partitioning task # Minmal-cut # Normalized-cut Criterion: Normalized-cut [Shi-Malik, '97] Define: Connectivity between groups relative to the density of each...
        \ No newline at end of file +Balloon Party = 風船のパーティー = fuusen no party
        7.3k words 7 mins.

        # 树的基本性质 # 树的叶子数 设 TTT 为 (n,m)(n,m)(n,m) 树,TTT 中有 nin_ini​ 个度为 i,(1≤i≤k)i,(1\leq i\leq k)i,(1≤i≤k) 的点,且有 ∑ni=n\sum n_i=n∑ni​=n ,则有:n1=2+n3+2n4+⋯+(k−2)nkn_1=2+n_3+2n_4+\cdots+(k-2)n_kn1​=2+n3​+2n4​+⋯+(k−2)nk​。 证明:由 m=n−1m=n-1m=n−1...
        6k words 5 mins.

        # 证书相关文件格式 证书 (Certificate) - .cer 、 .crt 私钥 (Private Key) - .key 证书签名请求 (Certificate signing request) - .csr 证书吊销列表 (Certificate Revocation List) - .crl 以上除了 .crl 均可以使用这两种编码方式,从而有这样的后缀: base64 编码 - .pem 二进制编码 (少见) - .der 为了给我们的网页应用使用 https 协议,我们需要得到有效的第三方认证的证书以及对应的私钥。 CA,certificate...
        3.6k words 3 mins.

        # 智能体 # 定义 Agent 是指驻留在某一环境下能够自主(Autonomous)、灵活(Flexible)地执行动作以满足设计目标的行为实体。 上述定义具有如下二个特点。 (1) 定义方式 : Agent 概念定义是基于 Agent 的外部可观察行为特征,而不是其内部的结构 (2) 抽象层次 : Agent 概念更加贴近于人们对现实世界(而不是计算机世界)中行为实体的理解 # 特点 # 驻留性 # 自主性 智能体能在没有人类或其他智能体的干涉和指导的情况下运行,并能根据其内部状态和感知到的环境信息来决定该实施什么样的动作,进而控制自身的行为 #...
        14k words 13 mins.

        # 公式表 # 基本傅里叶变换对 # 性质 # 非周期信号 # FT 公式推导 定义 x~[n]\widetilde{x}[n]x[n] 为 x[n]x[n]x[n] 的周期延拓,以 NNN 为周期,x[n]x[n]x[n] 是 x~[n]\widetilde{x}[n]x[n] 的一个周期。 当 T→∞T\to\infinT→∞ 时,在 (−∞,+∞)(-\infin,+\infin)(−∞,+∞) 内 x~[n]=x[n]\widetilde{x}[n]=x[n]x[n]=x[n] ;在该极限情况下有...
        2.3k words 2 mins.

        # 归纳总结 缩写: CTFS—— 连续时间傅里叶级数 DTFS—— 离散时间傅里叶级数 CTFT—— 连续时间傅里叶变换 DTFT—— 离散时间傅里叶变换 一图归纳: 信号在时域的特性和在频域的特性之间存在以下对应关系: 时域的周期性 —— 频域的离散性 时域的非周期性 —— 频域的连续性 时域的离散性 —— 频域的周期性 时域的连续性 —— 频域的非周期性 # CTFT 的对偶 若 x(t)⟷CTFTX(jω)x(t)\stackrel{CTFT}{\longleftrightarrow}X(j\omega)x(t)⟷CTFT​X(jω)...
        8.4k words 8 mins.

        # 调度相关概念 为什么需要调度? 释放被阻塞、挂起的进程占用的 CPU 资源 实现 I/O 操作和 CPU 并行执行( I/O 密集型和 CPU 密集型) 调度的层次、类型 处理器 / CPU 调度⭐️(进程调度的主要内容) 作业调度 /long-term scheduling / 长程调度 / 高级调度 The decision to add to the pool of processes to be executed 内存调度 /medium-term scheduling / 中程调度 / 中级调度 The decision to add to the number of...
        8.8k words 8 mins.

        # 正规子群 # 定义 设 HHH 是群 GGG 的子群 (⚠️得先是子群),如果 ∀a∈G∀a\in G∀a∈G 都有 Ha=aHHa=aHHa=aH ,则称 HHH 是 GGG 的正规子群 (normal subgroup) 或不变子群 (invariant subgroup),记作: H⊴GH\unlhd G H⊴G 条件 Ha=aHHa=aHHa=aH 仅表示两个集合 HaHaHa 与 aHaHaH 相等。不能错误地认为,由 Ha=aHHa=aHHa=aH 可推出对 HHH 的任意元素 hhh 有 ha=ahha=ahha=ah。 正确的理解是:对任意...
        4.9k words 4 mins.

        用邻接矩阵或关联矩阵表示图,称为图的代数表示。用矩阵表示图,主要有两个优点: (1) 能够把图输入到计算机中 (2) 可以用代数方法研究图论 Spectral Graph Theory 是研究图和代数的一个领域 # 邻接矩阵 Adjaceny matrix is n×nn\times nn×n symmetric matrix # 定义 设 GGG 为 nnn 阶图,V={v1,z2,⋯ ,vn}V=\{v_1,z_2,\cdots,v_n\}V={v1​,z2​,⋯,vn​} ,邻接矩阵 A(G)=(aij)A(G)=...
        1.4k words 1 mins.

        This article isn't finished yet... Graph Partitioning Bi-partitioning task # Minmal-cut # Normalized-cut Criterion: Normalized-cut [Shi-Malik, '97] Define: Connectivity between groups relative to the density of each...
        \ No newline at end of file diff --git a/page/5/index.html b/page/5/index.html index f999cc17..91cdb394 100644 --- a/page/5/index.html +++ b/page/5/index.html @@ -1 +1 @@ -Balloon Party = 風船のパーティー = fuusen no party
        5.8k words 5 mins.

        按照所用知识的确定性,可以分为确定性和不确定性推理: 确定性推理是建立在经典逻辑基础上的,经典逻辑的基础之一就是集合论。这在很多实际情况中是很难做到的,如高、矮、胖、瘦就很难精确地分开; 不确定性推理就是从不确定性初始证据出发,通过运用不确定性的知识,最终推出具有一定程度的不确定性但却是合理或者近乎合理的结论的思维过程。 常识(common...
        11k words 10 mins.

        # 群子集的乘积运算 # 定义 设 AAA 和 BBB 是群 GGG 的两个非空子集,称集合: AB={ab∣a∈A,b∈B}AB = \{ab|a\in A, b\in B \} AB={ab∣a∈A,b∈B} 为群的子集 AAA 与 BBB 的乘积 (product) 如果 g∈Gg\in Gg∈G 为群 GGG 的一个元素,A={g}A=\{g\}A={g} ,则: ABABAB 简记为 gB={gb∣b∈B}gB = \{gb|b\in B\}gB={gb∣b∈B}; BABABA 简记为...
        5.6k words 5 mins.

        基本框架 数字电路 组合逻辑电路 以门电路为基本单元 时序逻辑电路 以触发器为基本单元 # 组合逻辑电路基础 Programmable Logic Device : PLD —— 可编程逻辑器件 # 特点 功能上,任意时刻的输出仅仅取决于该时刻的输入,与电路原来的状态无关; 结构上,组合逻辑电路不含有存储单元(记忆单元),只有从输入到输出的通路,没有反馈环路 # 分析方法 给定一个组合逻辑电路,分析其电路逻辑功能: 根据电路图,从输入到输出逐级分析,得到输出与输入的逻辑函数式; 对得到的逻辑函数式进行化简、形式变换或列出真值表,最后分析其逻辑功能; # 设计方法 Step...
        12k words 11 mins.

        This article isn't finished yet... # 严格备选 (Strict Alternation) # PetersonSolution # 互斥锁 互斥锁是一种外部强加的控制方案,或者说是全局控制; 获得锁: 伪代码acquire &#123; while (!available) ; /* busy wait */ available = false;&#125;释放锁: 伪代码release &#123; available = true;&#125;利用互斥锁的临界区问题解决方案: 伪代码while (true)...
        23k words 21 mins.

        # 公式表 # 傅里叶变换性质 # 基本傅里叶变换对 连续时间非周期信号傅里叶变换的推导基本思想 非周期函数可以通过周期性延拓得到周期函数 周期函数可以令基波周期趋于无穷大得到连续的非周期函数 对于周期性矩形脉冲信号: ak=2T1TSa(kπ2T1T)Tak=2T1Sa(ωT1)\begin{aligned} a_k &amp;= \frac{2T_1}{T}\text{Sa}(k\pi\frac{2T_1}{T})\\ Ta_k &amp;= 2T_1\text{Sa}(\omega...
        3k words 3 mins.

        # 置换群的定义 设 S={1,2,⋯ ,n}S=\{1,2,\cdots,n\}S={1,2,⋯,n} ; S 上的任意双函数 σ:S→S\sigma:S\to Sσ:S→S 称为 S 上的一个 n 元置换 (permutation),记为: σ=(12⋯nσ(1)σ(2)⋯σ(n))\sigma= \begin{pmatrix} 1 &amp; 2 &amp; \cdots &amp; n \\ \sigma(1) &amp; \sigma(2)...
        3.9k words 4 mins.

        This article ism't finished yet... # 顶点标号法 Dantzig 算法,1959 年,旦捷希 (Dantzig) 发现了在赋权图中求由点 a 到点 b 的最短路好算法,称为顶点标号法。 和弗洛伊德算法非常非常像... ,在这里补充一下 Floyid 算法对邻接矩阵求最短距离矩阵过程,这个过程和 Dantzig 几乎一致(完全一致好吧,考试可能会考?): 按顺序一列一列看,看第 iii 列时: 按顺序一行一行看,看第 jjj 行时,若 aji≠∞a_{ji}\neq\infinaji​=∞ 则: 看第 jjj 列,看...
        1.3k words 1 mins.

        这是老师布置的一个小作业,让我们在 linux 内核源码里增加自定义的系统调用程序,用这篇文章记录一下完整的操作过程; 这样扩展 Linux 内核的传统方式需要每次都重新编译内核,更现代的方式是使用 eBPF 使用清华镜像源下载 linux 内核 5.10.19 版本,解压然后进入解压目录: shellwget https://mirrors.tuna.tsinghua.edu.cn/kernel/v5.x/linux-5.10.19.tar.gz......tar -zxvf linux-5.10.19.tar.gz......cd linux-5.10.19在 kernel/sys.c...
        \ No newline at end of file +Balloon Party = 風船のパーティー = fuusen no party
        5.8k words 5 mins.

        按照所用知识的确定性,可以分为确定性和不确定性推理: 确定性推理是建立在经典逻辑基础上的,经典逻辑的基础之一就是集合论。这在很多实际情况中是很难做到的,如高、矮、胖、瘦就很难精确地分开; 不确定性推理就是从不确定性初始证据出发,通过运用不确定性的知识,最终推出具有一定程度的不确定性但却是合理或者近乎合理的结论的思维过程。 常识(common...
        11k words 10 mins.

        # 群子集的乘积运算 # 定义 设 AAA 和 BBB 是群 GGG 的两个非空子集,称集合: AB={ab∣a∈A,b∈B}AB = \{ab|a\in A, b\in B \} AB={ab∣a∈A,b∈B} 为群的子集 AAA 与 BBB 的乘积 (product) 如果 g∈Gg\in Gg∈G 为群 GGG 的一个元素,A={g}A=\{g\}A={g} ,则: ABABAB 简记为 gB={gb∣b∈B}gB = \{gb|b\in B\}gB={gb∣b∈B}; BABABA 简记为...
        5.6k words 5 mins.

        基本框架 数字电路 组合逻辑电路 以门电路为基本单元 时序逻辑电路 以触发器为基本单元 # 组合逻辑电路基础 Programmable Logic Device : PLD —— 可编程逻辑器件 # 特点 功能上,任意时刻的输出仅仅取决于该时刻的输入,与电路原来的状态无关; 结构上,组合逻辑电路不含有存储单元(记忆单元),只有从输入到输出的通路,没有反馈环路 # 分析方法 给定一个组合逻辑电路,分析其电路逻辑功能: 根据电路图,从输入到输出逐级分析,得到输出与输入的逻辑函数式; 对得到的逻辑函数式进行化简、形式变换或列出真值表,最后分析其逻辑功能; # 设计方法 Step...
        12k words 11 mins.

        This article isn't finished yet... # 严格备选 (Strict Alternation) # PetersonSolution # 互斥锁 互斥锁是一种外部强加的控制方案,或者说是全局控制; 获得锁: 伪代码acquire &#123; while (!available) ; /* busy wait */ available = false;&#125;释放锁: 伪代码release &#123; available = true;&#125;利用互斥锁的临界区问题解决方案: 伪代码while (true)...
        23k words 21 mins.

        # 公式表 # 傅里叶变换性质 # 基本傅里叶变换对 连续时间非周期信号傅里叶变换的推导基本思想 非周期函数可以通过周期性延拓得到周期函数 周期函数可以令基波周期趋于无穷大得到连续的非周期函数 对于周期性矩形脉冲信号: ak=2T1TSa(kπ2T1T)Tak=2T1Sa(ωT1)\begin{aligned} a_k &amp;= \frac{2T_1}{T}\text{Sa}(k\pi\frac{2T_1}{T})\\ Ta_k &amp;= 2T_1\text{Sa}(\omega...
        3k words 3 mins.

        # 置换群的定义 设 S={1,2,⋯ ,n}S=\{1,2,\cdots,n\}S={1,2,⋯,n} ; S 上的任意双函数 σ:S→S\sigma:S\to Sσ:S→S 称为 S 上的一个 n 元置换 (permutation),记为: σ=(12⋯nσ(1)σ(2)⋯σ(n))\sigma= \begin{pmatrix} 1 &amp; 2 &amp; \cdots &amp; n \\ \sigma(1) &amp; \sigma(2)...
        3.9k words 4 mins.

        This article ism't finished yet... # 顶点标号法 Dantzig 算法,1959 年,旦捷希 (Dantzig) 发现了在赋权图中求由点 a 到点 b 的最短路好算法,称为顶点标号法。 和弗洛伊德算法非常非常像... ,在这里补充一下 Floyid 算法对邻接矩阵求最短距离矩阵过程,这个过程和 Dantzig 几乎一致(完全一致好吧,考试可能会考?): 按顺序一列一列看,看第 iii 列时: 按顺序一行一行看,看第 jjj 行时,若 aji≠∞a_{ji}\neq\infinaji​=∞ 则: 看第 jjj 列,看...
        1.3k words 1 mins.

        这是老师布置的一个小作业,让我们在 linux 内核源码里增加自定义的系统调用程序,用这篇文章记录一下完整的操作过程; 这样扩展 Linux 内核的传统方式需要每次都重新编译内核,更现代的方式是使用 eBPF 使用清华镜像源下载 linux 内核 5.10.19 版本,解压然后进入解压目录: shellwget https://mirrors.tuna.tsinghua.edu.cn/kernel/v5.x/linux-5.10.19.tar.gz......tar -zxvf linux-5.10.19.tar.gz......cd linux-5.10.19在 kernel/sys.c...
        \ No newline at end of file diff --git a/page/6/index.html b/page/6/index.html index 123eafa0..f8c54e29 100644 --- a/page/6/index.html +++ b/page/6/index.html @@ -1 +1 @@ -Balloon Party = 風船のパーティー = fuusen no party
        1.5k words 1 mins.

        # 基本概念 !!! 可能会在选择填空题中考察 上拉 —— 使输出钳位至高电平 下拉 —— 使输出钳位至低电平; 灌电流 —— 输出为低电平时,负载电流从外部电路注入输出端口; 拉电流 —— 输出为高电平时,负载电流从输出端口流出到外部电路; 当门电路没有外部的有源输入信号,而是输入端口与地之间接入一个电阻时: 关门电阻 —— 能够维持门电路输出高电平的电阻的最大值; 开门电阻 —— 能够维持门电路输出低电平的电阻的最小值; (联想 TTL 反相器的输入端负载特性曲线) 扇入系数 —— 门电路允许的输入端的个数; 扇出系数 —— 门电路允许驱动的同类型负载门的个数; 高阻态...
        13k words 12 mins.

        # 搜索相关概念 对于给定的问题,智能系统的行为一般是找到能够达到所希望目标的动作序列,并使其所付出的代价最小、性能最好。 基于给定的问题,问题求解的第一步是目标的表示。 搜索就是找到智能系统的动作序列的过程。 搜索算法的: 输入是给定的问题 搜索什么(目标) 在哪里搜索(搜索空间) 和通常的搜索空间不同,人工智能中大多数问题的状态空间在问题求解之前不是全部知道的 输出是表示为动作序列的方案。 一旦有了方案,就可以执行该方案所给出的动作了。(执行阶段) 求解问题包括: 目标表示 搜索 执行 #...
        15k words 13 mins.

        # 连续时间周期信号的傅里叶级数表示 # 连续时间傅里叶级数 成谐波关系的复指数信号集:ejkω0t,   k=0,±1,±2,⋯e^{jk\omega_0 t},\ \ \ k=0,\pm 1,\pm 2,\cdotsejkω0​t,   k=0,±1,±2,⋯...
        3k words 3 mins.

        # 宿主与虚拟机之间的网络问题 我们的两台宿主居然都 ping 不到虚拟机,导致后续 haproxy 代理转发后出现所有虚拟机 server 都是 down 状态 原因应该是宿主只有一个校园网网段的 ip,但是虚拟机只有实验室局域网网段的 ip 这个问题需要配置宿主节点的网络,可以在 PVE 控制面板修改 网络 / Network: 实际上就是编辑该节点的: /etc/network/interfaces 配置文件 应该需要配置虚拟网卡才能解决,但是我们的一台服务器有两张网卡(图中 eno1 eno2) 所以我们最后的解决方案给服务器插了多一条网线,配置该网卡 ip 为实验室局域网的...
        953 words 1 mins.

        # VMware 或 VirtualBox 导出虚拟机 导出之前可以参考以下几个因素,避免后续麻烦: 虚拟机由 VMware 或 VirtualBox 导出,磁盘镜像有可能是 vmdk 格式,如果从 KVM 管理器导出,可能是 qcow2 格式。最流行的虚拟机导出格式是 OVF 标准,但实际上由于 OVF 标准本身不完善,以及虚拟机管理器导出的众多非标准扩展信息,跨管理器使用 OVF 往往受很多限制。 除了格式不兼容之外,如果虚拟机管理器之间的虚拟硬件设备差别太大,也可能导致虚拟机镜像导入失败。特别是 Windows 虚拟机,对于硬件变化特别敏感。为解决这一问题,可以在导出 Windows...
        3k words 3 mins.

        # 前言 在 Matrix 实验室用电子垃圾搭了一个小集群。 起因是 jygg 拿了一台被淘汰的存储型服务器给 cjl 玩,(据说是以前买来发现性能太差了,于是就把里面的 64T 的硬盘全部拿走了装到别的服务器上了,这台就废弃了),cjl 整了个 500g 的机械硬盘给服务器装上了,同时把不知为啥留在这里的一个台式 PC 给我玩,然后索性就装上了 PVE,在 jygg 的帮助下非常顺利搭了一个两台物理机组成的集群。 # 安装 PVE 到官网下载 PVE,这是一个只有 1G 大小左右的基于 debian 的装好了 proxmox 相关软件、服务的镜像,然后用 U 盘给物理机安装。 安装过程可以用...
        9.6k words 9 mins.

        # 群 (Group) 的定义 如果集合 G 及其二元运算 ⋅\cdot⋅ 满足下面的性质,则称 (G,⋅)(G,\cdot )(G,⋅) 是群: 二元运算 ⋅\cdot⋅ 满足结合律:即 ∀a,b,c∈G∀a,b,c\in G∀a,b,c∈G 有 (a⋅b)⋅c=a⋅(b⋅c)(a\cdot b)\cdot c=a\cdot(b\cdot c)(a⋅b)⋅c=a⋅(b⋅c); 有单位元 eee,即存在 e∈Ge\in Ge∈G,使得 ∀a∈G∀a\in G∀a∈G 有 e⋅a=a⋅e=ae\cdot...
        460 words 1 mins.

        想在这里特别地感谢你!想在这里特别地感谢你! 想在这里特别地感谢你! 每一句“生日快乐”每一句“生日快乐” 每一句“生日快乐” 就算只是一句“生日快乐”就算只是一句“生日快乐” 就算只是一句“生日快乐” 都让我非常幸福!都让我非常幸福! 都让我非常幸福! 特别感谢:特别感谢: 特别感谢: Luv thank you LDY thank you kazegairoman thank you CJL thank...
        17k words 16 mins.

        Linear Time Invariant 线性时不变 基本信号应具有以下特点: 本身尽可能简单,并且用它们的线性组合能够表示尽可能广泛的信号 LTI 系统对这种信号的响应易于计算且有利于对系统的深入理解 两类有用的基本信号: 复指数信号 单位冲激信号 本章主要考虑以单位冲激信号为基本信号进行信号与系统分析 # 卷积和 # 定义 对于任意一个离散时间系统: 设离散时间系统为 LTI 的,定义单位脉冲响应 h[n]h[n]h[n] 为以单位冲激信号 δ[n]\delta[n]δ[n] 为输入时的输出: δ[n]→h[n]\delta[n]\to...
        \ No newline at end of file +Balloon Party = 風船のパーティー = fuusen no party
        1.5k words 1 mins.

        # 基本概念 !!! 可能会在选择填空题中考察 上拉 —— 使输出钳位至高电平 下拉 —— 使输出钳位至低电平; 灌电流 —— 输出为低电平时,负载电流从外部电路注入输出端口; 拉电流 —— 输出为高电平时,负载电流从输出端口流出到外部电路; 当门电路没有外部的有源输入信号,而是输入端口与地之间接入一个电阻时: 关门电阻 —— 能够维持门电路输出高电平的电阻的最大值; 开门电阻 —— 能够维持门电路输出低电平的电阻的最小值; (联想 TTL 反相器的输入端负载特性曲线) 扇入系数 —— 门电路允许的输入端的个数; 扇出系数 —— 门电路允许驱动的同类型负载门的个数; 高阻态...
        14k words 13 mins.

        # 搜索相关概念 对于给定的问题,智能系统的行为一般是找到能够达到所希望目标的动作序列,并使其所付出的代价最小、性能最好。 基于给定的问题,问题求解的第一步是目标的表示。 搜索就是找到智能系统的动作序列的过程。 搜索算法的: 输入是给定的问题 搜索什么(目标) 在哪里搜索(搜索空间) 和通常的搜索空间不同,人工智能中大多数问题的状态空间在问题求解之前不是全部知道的 输出是表示为动作序列的方案。 一旦有了方案,就可以执行该方案所给出的动作了。(执行阶段) 求解问题包括: 目标表示 搜索 执行 #...
        15k words 13 mins.

        # 连续时间周期信号的傅里叶级数表示 # 连续时间傅里叶级数 成谐波关系的复指数信号集:ejkω0t,   k=0,±1,±2,⋯e^{jk\omega_0 t},\ \ \ k=0,\pm 1,\pm 2,\cdotsejkω0​t,   k=0,±1,±2,⋯...
        3k words 3 mins.

        # 宿主与虚拟机之间的网络问题 我们的两台宿主居然都 ping 不到虚拟机,导致后续 haproxy 代理转发后出现所有虚拟机 server 都是 down 状态 原因应该是宿主只有一个校园网网段的 ip,但是虚拟机只有实验室局域网网段的 ip 这个问题需要配置宿主节点的网络,可以在 PVE 控制面板修改 网络 / Network: 实际上就是编辑该节点的: /etc/network/interfaces 配置文件 应该需要配置虚拟网卡才能解决,但是我们的一台服务器有两张网卡(图中 eno1 eno2) 所以我们最后的解决方案给服务器插了多一条网线,配置该网卡 ip 为实验室局域网的...
        953 words 1 mins.

        # VMware 或 VirtualBox 导出虚拟机 导出之前可以参考以下几个因素,避免后续麻烦: 虚拟机由 VMware 或 VirtualBox 导出,磁盘镜像有可能是 vmdk 格式,如果从 KVM 管理器导出,可能是 qcow2 格式。最流行的虚拟机导出格式是 OVF 标准,但实际上由于 OVF 标准本身不完善,以及虚拟机管理器导出的众多非标准扩展信息,跨管理器使用 OVF 往往受很多限制。 除了格式不兼容之外,如果虚拟机管理器之间的虚拟硬件设备差别太大,也可能导致虚拟机镜像导入失败。特别是 Windows 虚拟机,对于硬件变化特别敏感。为解决这一问题,可以在导出 Windows...
        3k words 3 mins.

        # 前言 在 Matrix 实验室用电子垃圾搭了一个小集群。 起因是 jygg 拿了一台被淘汰的存储型服务器给 cjl 玩,(据说是以前买来发现性能太差了,于是就把里面的 64T 的硬盘全部拿走了装到别的服务器上了,这台就废弃了),cjl 整了个 500g 的机械硬盘给服务器装上了,同时把不知为啥留在这里的一个台式 PC 给我玩,然后索性就装上了 PVE,在 jygg 的帮助下非常顺利搭了一个两台物理机组成的集群。 # 安装 PVE 到官网下载 PVE,这是一个只有 1G 大小左右的基于 debian 的装好了 proxmox 相关软件、服务的镜像,然后用 U 盘给物理机安装。 安装过程可以用...
        9.6k words 9 mins.

        # 群 (Group) 的定义 如果集合 G 及其二元运算 ⋅\cdot⋅ 满足下面的性质,则称 (G,⋅)(G,\cdot )(G,⋅) 是群: 二元运算 ⋅\cdot⋅ 满足结合律:即 ∀a,b,c∈G∀a,b,c\in G∀a,b,c∈G 有 (a⋅b)⋅c=a⋅(b⋅c)(a\cdot b)\cdot c=a\cdot(b\cdot c)(a⋅b)⋅c=a⋅(b⋅c); 有单位元 eee,即存在 e∈Ge\in Ge∈G,使得 ∀a∈G∀a\in G∀a∈G 有 e⋅a=a⋅e=ae\cdot...
        460 words 1 mins.

        想在这里特别地感谢你!想在这里特别地感谢你! 想在这里特别地感谢你! 每一句“生日快乐”每一句“生日快乐” 每一句“生日快乐” 就算只是一句“生日快乐”就算只是一句“生日快乐” 就算只是一句“生日快乐” 都让我非常幸福!都让我非常幸福! 都让我非常幸福! 特别感谢:特别感谢: 特别感谢: Luv thank you LDY thank you kazegairoman thank you CJL thank...
        17k words 16 mins.

        Linear Time Invariant 线性时不变 基本信号应具有以下特点: 本身尽可能简单,并且用它们的线性组合能够表示尽可能广泛的信号 LTI 系统对这种信号的响应易于计算且有利于对系统的深入理解 两类有用的基本信号: 复指数信号 单位冲激信号 本章主要考虑以单位冲激信号为基本信号进行信号与系统分析 # 卷积和 # 定义 对于任意一个离散时间系统: 设离散时间系统为 LTI 的,定义单位脉冲响应 h[n]h[n]h[n] 为以单位冲激信号 δ[n]\delta[n]δ[n] 为输入时的输出: δ[n]→h[n]\delta[n]\to...
        \ No newline at end of file diff --git a/page/7/index.html b/page/7/index.html index 77cdac90..344b75b4 100644 --- a/page/7/index.html +++ b/page/7/index.html @@ -1 +1 @@ -Balloon Party = 風船のパーティー = fuusen no party
        5.2k words 5 mins.

        # 线程的基本概念 # 为什么需要线程 有的进程可能需要 “同时” 做很多事,而传统的进程只能串行地执行一系列程序。为此,引入了 “线程”,来增加并发度。 大多数现代应用程序都是多线程的,线程在应用程序中运行,应用程序的多个任务可以由单独的线程实现 如 Web 浏览器: 更新显示 获取数据 拼写检查 响应网络请求 进程创建很重载,而线程创建很轻量; 可以简化代码,提高效率; 内核 (作为内核程序进程) 通常是多线程的,多个线程在内核中运行,每个线程执行一个特定的任务; 现代服务器 —— 多线程服务器,可以同时处理多个并发请求 #...
        21k words 19 mins.

        # 进程的基本概念 # 进程的定义 典型的定义: 进程是进程实体的运行过程,是系统进行资源分配和调度的一个独立单位 == 一个具有独立功能的程序在一个数据集合上运行的过程 # 区分进程、程序、进程实体 程序(Program):是静态的,就是个存放在磁盘里的可执行文件,就是一系列的指令集合 静态连接 动态链接 进程(Process):是动态的,是程序的一次执行过程 进程实体(进程映像):是静态的,反应了进程在某一时刻的状态 # 进程实体的组成 一个 == 进程实体(进程映像)== 由三个部分组成: PCB 进程描述信息 进程控制和管理信息 资源分配清单 处理机状态信息(CPU...
        719 words 1 mins.

        # 环境变量文件的加载顺序 在登录 Linux 系统并启动一个 bash shell 时,默认情况下 bash 会在若干个文件中查找环境变量的设置。这些文件统称为系统环境文件。bash 检查的环境变量文件的顺序情况取决于系统运行的 shell 方式。系统运行 shell 方式一般为 4 种: 是否交互式 是否登陆式 哪些情况 是 是 通过系统用户登录默认运行的 shell 是 否 非登录交互式运行 shell 否 是 执行脚本运行非交互式 shell 否 否 无 # bash 命令启动 shell 命令 bash 启动 shell 时,可以通过选项改变其行为 shellbash...
        976 words 1 mins.

        # 环境 / 情景 MacOS 14.2 M1 用 VSCode 远程连接 Ubuntu 22.04 # 问题 在虚拟机上安装了 Enviroment Modules ,并按照官方配置文档配置了 bash 启动时自动加载 module 指令,ssh 远程登录虚拟机 'module' 指令可用,但是 VSCode 远程连接虚拟机发现 module 指令使用不了。 # 原因 在 VSCode 的 bash 中检查发现当前 shell 不是登录式的: shellshopt login_shell# 结果为 off 不是登陆式login_shell off分析发现由于 VSCode...
        12k words 11 mins.

        强烈推荐:https://hrl.boyuai.com/ # 基本概念 RL, in a nutshell, is to “learn to make good sequences of decisions through trail-and-errors” # 机器学习范式 监督 给定训练数据 training data 和期望的输出(标签)desired outputs (with labels) 半监督 给定训练数据 training data 和部分期望的输出 some labels 无监督 仅给定训练数据 training data without...
        2.1k words 2 mins.

        # 题面 设 f:A→Bf:A\to Bf:A→B 是函数,定义 A 上的关系 R,∀a,b∈A\forall a,b \in A∀a,b∈A,a R b 当且仅当 f (a) = f (b) 。 证明 R 是等价关系,并给出它的等价类和商集,求这样的等价关系有多少个? # 证明 R 是等价关系 显然 R 是等价关系,因为对∀a, b, c\in A,f (a)=f (a), f (a)=f (b) 蕴涵 f (b) = f (a) , f (a) = f (b) 且 f (b) = f (c) 蕴涵 f (a) = f (c) ,即 R...
        2.6k words 2 mins.

        # 基本概念 # 不严格定义的概念 # 集合 作为整体研究的一堆东西,用大写字母 A, B, C, ⋯表示 # 元素 集合这一堆东西中的每一个,用小写字母 a, b, c, ⋯表示 # 属于 元素与集合间的关系 元素 a 属于集合 A ,记为 a∈Aa\in Aa∈A a 不属于 A,记为 a∉Aa\notin Aa∈/A 元素与集合间的属于关系也称为成员关系,元素是集合的成员 # 全集 研究范围内的所有东西,记为 U # 用逻辑语言严格定义的概念 # 子集关系 A⊆B⇔∀x(x∈A→x∈B)A \subseteq B \lrArr \forall...
        12k words 11 mins.

        # 图的基础知识 # 图的定义 一 个 图 GGG 定 义 为 一 个 有 序 对 (V , E), 记 为 G = ( V , E ), 其 中: V 是一个非空集合,称为顶点集或边集,其元素称为项点或点 E 是由 V 中的点组成的无序点对构成的集合,称为边集,其元素称为边,且同一 点对在 E 中可出现多次。 图 GGG 的顶点集也记为 V (G),边集也记为 E (G)。 顶点集和边集都有限的图称为有限图。 只有一个项点而无边的图称为平凡图。其他所有的图都称为非平凡图。 做题往往需要先排除平凡图 边集为空的图称为空图。 图 GGG 的顶点数 (或阶数) 和边数可分别用符号 n(G)...
        5.3k words 5 mins.

        # 逻辑代数中的基本运算 # 三种基本运算 # 真值表 表征逻辑事件输入(条件)和输出(结果)之间全部可能状态的表格 与运算: A B Y 0 0 0 0 1 0 1 0 0 1 1 1 或运算: A B Y 0 0 0 0 1 1 1 0 1 1 1 1 非运算: A Y 0 1 1 0 # 其他复合逻辑运算 # 与非 “先与再非” # 或非 “先或再非” # 与或非 “先与再或再非” # 异或 XOR 不同时输出 1,相同时输出 0 A⊕B=A‾B+AB‾A\oplus...
        \ No newline at end of file +Balloon Party = 風船のパーティー = fuusen no party
        5.2k words 5 mins.

        # 线程的基本概念 # 为什么需要线程 有的进程可能需要 “同时” 做很多事,而传统的进程只能串行地执行一系列程序。为此,引入了 “线程”,来增加并发度。 大多数现代应用程序都是多线程的,线程在应用程序中运行,应用程序的多个任务可以由单独的线程实现 如 Web 浏览器: 更新显示 获取数据 拼写检查 响应网络请求 进程创建很重载,而线程创建很轻量; 可以简化代码,提高效率; 内核 (作为内核程序进程) 通常是多线程的,多个线程在内核中运行,每个线程执行一个特定的任务; 现代服务器 —— 多线程服务器,可以同时处理多个并发请求 #...
        21k words 19 mins.

        # 进程的基本概念 # 进程的定义 典型的定义: 进程是进程实体的运行过程,是系统进行资源分配和调度的一个独立单位 == 一个具有独立功能的程序在一个数据集合上运行的过程 # 区分进程、程序、进程实体 程序(Program):是静态的,就是个存放在磁盘里的可执行文件,就是一系列的指令集合 静态连接 动态链接 进程(Process):是动态的,是程序的一次执行过程 进程实体(进程映像):是静态的,反应了进程在某一时刻的状态 # 进程实体的组成 一个 == 进程实体(进程映像)== 由三个部分组成: PCB 进程描述信息 进程控制和管理信息 资源分配清单 处理机状态信息(CPU...
        719 words 1 mins.

        # 环境变量文件的加载顺序 在登录 Linux 系统并启动一个 bash shell 时,默认情况下 bash 会在若干个文件中查找环境变量的设置。这些文件统称为系统环境文件。bash 检查的环境变量文件的顺序情况取决于系统运行的 shell 方式。系统运行 shell 方式一般为 4 种: 是否交互式 是否登陆式 哪些情况 是 是 通过系统用户登录默认运行的 shell 是 否 非登录交互式运行 shell 否 是 执行脚本运行非交互式 shell 否 否 无 # bash 命令启动 shell 命令 bash 启动 shell 时,可以通过选项改变其行为 shellbash...
        976 words 1 mins.

        # 环境 / 情景 MacOS 14.2 M1 用 VSCode 远程连接 Ubuntu 22.04 # 问题 在虚拟机上安装了 Enviroment Modules ,并按照官方配置文档配置了 bash 启动时自动加载 module 指令,ssh 远程登录虚拟机 'module' 指令可用,但是 VSCode 远程连接虚拟机发现 module 指令使用不了。 # 原因 在 VSCode 的 bash 中检查发现当前 shell 不是登录式的: shellshopt login_shell# 结果为 off 不是登陆式login_shell off分析发现由于 VSCode...
        12k words 11 mins.

        强烈推荐:https://hrl.boyuai.com/ # 基本概念 RL, in a nutshell, is to “learn to make good sequences of decisions through trail-and-errors” # 机器学习范式 监督 给定训练数据 training data 和期望的输出(标签)desired outputs (with labels) 半监督 给定训练数据 training data 和部分期望的输出 some labels 无监督 仅给定训练数据 training data without...
        2.6k words 2 mins.

        # 基本概念 # 不严格定义的概念 # 集合 作为整体研究的一堆东西,用大写字母 A, B, C, ⋯表示 # 元素 集合这一堆东西中的每一个,用小写字母 a, b, c, ⋯表示 # 属于 元素与集合间的关系 元素 a 属于集合 A ,记为 a∈Aa\in Aa∈A a 不属于 A,记为 a∉Aa\notin Aa∈/A 元素与集合间的属于关系也称为成员关系,元素是集合的成员 # 全集 研究范围内的所有东西,记为 U # 用逻辑语言严格定义的概念 # 子集关系 A⊆B⇔∀x(x∈A→x∈B)A \subseteq B \lrArr \forall...
        2.1k words 2 mins.

        # 题面 设 f:A→Bf:A\to Bf:A→B 是函数,定义 A 上的关系 R,∀a,b∈A\forall a,b \in A∀a,b∈A,a R b 当且仅当 f (a) = f (b) 。 证明 R 是等价关系,并给出它的等价类和商集,求这样的等价关系有多少个? # 证明 R 是等价关系 显然 R 是等价关系,因为对∀a, b, c\in A,f (a)=f (a), f (a)=f (b) 蕴涵 f (b) = f (a) , f (a) = f (b) 且 f (b) = f (c) 蕴涵 f (a) = f (c) ,即 R...
        12k words 11 mins.

        # 图的基础知识 # 图的定义 一 个 图 GGG 定 义 为 一 个 有 序 对 (V , E), 记 为 G = ( V , E ), 其 中: V 是一个非空集合,称为顶点集或边集,其元素称为项点或点 E 是由 V 中的点组成的无序点对构成的集合,称为边集,其元素称为边,且同一 点对在 E 中可出现多次。 图 GGG 的顶点集也记为 V (G),边集也记为 E (G)。 顶点集和边集都有限的图称为有限图。 只有一个项点而无边的图称为平凡图。其他所有的图都称为非平凡图。 做题往往需要先排除平凡图 边集为空的图称为空图。 图 GGG 的顶点数 (或阶数) 和边数可分别用符号 n(G)...
        5.3k words 5 mins.

        # 逻辑代数中的基本运算 # 三种基本运算 # 真值表 表征逻辑事件输入(条件)和输出(结果)之间全部可能状态的表格 与运算: A B Y 0 0 0 0 1 0 1 0 0 1 1 1 或运算: A B Y 0 0 0 0 1 1 1 0 1 1 1 1 非运算: A Y 0 1 1 0 # 其他复合逻辑运算 # 与非 “先与再非” # 或非 “先或再非” # 与或非 “先与再或再非” # 异或 XOR 不同时输出 1,相同时输出 0 A⊕B=A‾B+AB‾A\oplus...
        \ No newline at end of file diff --git a/page/8/index.html b/page/8/index.html index f7a07870..db0cd37a 100644 --- a/page/8/index.html +++ b/page/8/index.html @@ -1 +1 @@ -Balloon Party = 風船のパーティー = fuusen no party
        2.1k words 2 mins.

        # 编码的概念 广义的编码指的就是将信息从一种形式转换为另一种规范的形式; 对于数字电路,就是将现实世界中的各种物理信息转换为离散的电压序列,来表示、处理和分析信息; 事实上,进位计数制也是一种编码,用不同进制数来表示 “量”; 将数制与码制分开介绍,首先单独讲解数制是因为数制的一个重要的特点是其编码后能够参与算术运算;而一般来说码制是对非数值信息的表示; # 二进制编码 # 无符号 只能表示 0 和正数 # 有符号 (有符号)二进制数:在前面增加一个符号位,0 表示正数,1 表示负数;引入符号位后,一个二进制数的最高位是符号位,符号位的 0 / 1...
        3.7k words 3 mins.

        # 系统的描述 # 连续 / 离散 时间系统 # 连续时间系统 输入信号与输出响应都是连续时间信号的系统。 eg: 机械系统、电路系统 # 离散时间系统 输入信号与输出响应都是离散时间信号的系统。 eg: 人口问题 # 两种描述方法 # 输入输出描述法 (微分方程、差分⽅程) 着眼于系统输入与输出的关系,适⽤用于单输入 — 单输出系统。 # 状态变量描述法 (状态⽅程、输出方程) 即可描述输入与输出的关系,还可以描述系统的内部状态 既可⽤用于单输入 — 单输出系统,⼜可⽤于多输⼊ — 多输出系统。 #...
        754 words 1 mins.

        # 数制 # 数制的概念 数制 (numerical system),又称 “计数制”,指用一组固定的符号和统一的规则来表示数值的方法; 数制关注的内容:每一位的构成(使用什么符号即数码)和进位的规则; 基本概念: 基数 —— 使用的数码的个数,即每一位最多能用多少个符号来表示;基数是 N 则称为 N 进制, 即 “逢 N 进一” 权 —— 每一位的 1 代表的大小; # 常用数制 # 不同进制的转换 # 其他进制 -&gt; 十进制 只需要按照 D=∑kiNiD=\sum k_i N^iD=∑ki​Ni 展开即可; kik_iki​ 为第 i 位的系数 NiN^iNi...
        1.9k words 2 mins.

        # 人工智能的概念 人工智能(Artificial Intelligence - AI):研究如何通过计算机技术实现一个智能体,让智能体完成类似于人类智能的行为。 # 智能体 Agent 完成智能行为的主体,是物理或抽象实体,可以感知世界并对世界施加作用。 大模型智能体 LLM Agent 在近几年随 ChatGPT 兴起的新概念 # 智能体的两大目标 能否使智能体像人一样思考、理性地思考? 能否使智能体像人一样行动、理性地行动? # 智能体可否像人一样思考 首先,需要测量人如何思考的方法:内省、心理实验、 脑成像。 #...
        9.4k words 9 mins.

        # 能量 / 功率 信号 # 能量信号 能量为一有界值,即 0 &lt; E &lt; + ∞,此时 P = 0 有时求 P 不好求,但是 E 好求,可以证明 E 为有界的,从而得到 P = 0 # 功率信号 功率为一有界值,即 0 &lt; P &lt; + ∞ ,此时 E = ∞ 有时求 E 不好求,但是 P 好求,可以证明 P 为有界的,从而得到 E = ∞ # 信号的分解 #...
        12k words 11 mins.

        # 操作系统的定义 实际上操作系统没有普遍接受的定义 操作系统 (Operating System,OS)是指控制和管理整个计算机系统的硬件和软件资源,并合理地组织调度计算机的工作和资源的分配;以提供给用户和其他软件方便的接口和环境;它是计算机系统中最基本的系统软件。 # 操作系统的功能和目标 # 作为系统资源的管理者 # 提供的功能 处理机管理 存储器管理 文件管理 设备管理 # 目标 安全、高效 # 向上层提供方便易用的服务 # GUI (Graphical User Interface) 图形化用户接口 # CLI (Command Line Interface)...
        27k words 25 mins.

        # 参考资料 Rust 语言圣经 (自称) 锈书 官网 # 环境配置 更新 Rust bashrustup update卸载 Rust bashrustup self uninstall查看版本 h bashrustc --version## 显示格式rustc.x.y.z([commit hash] [commit date])安装 Rust 同时还会安装文档,查看本地文档 bashrustup doc# Rust 工具 # 编译器 # rustc 只适合单文件编译 shellrustc xxx.rs &lt;output># cargo Cargo 是 Rust...
        9.4k words 9 mins.

        # pthread 库 # 简介 pthread 是 POSIX 线程库,它是一种跨平台的线程库,可以在不同的操作系统上使用。它是 C 语言的标准库,提供了一组函数来创建、管理和同步线程。&quot;pthread&quot; 库相对于 &quot;thread&quot; 库来说,更加底层和与操作系统紧密相关。 thread 是 C11 标准提供的线程库,它是 C 的一部分。它提供了一种面向对象的方式来创建、管理和同步线程。&quot;thread&quot; 库提供了 std::thread...
        197 words 1 mins.

        # DBMS 数据库管理系统 DBMS 的分类: 關聯式資料庫(SQL) 非關聯式資料庫 - (nosQL /not just SQL) • MySQL • MogoDB • Oracle • Redis • PostgreSQL • DynamoDB • SQL Server • Elaticsearch 關聯式資料庫管理系統 (RDBMS) 非關聯式資料庫管理系統 (NRDBMS) SQL - Structured Query Language - 结构性询问语言
        1.4k words 1 mins.

        # require require 是 Node.js 中用于加载模块的函数。它的主要作用是将外部模块引入到当前的 JavaScript 文件中,以便在代码中使用被引入的模块的功能、变量或对象。 require 函数的基本语法是: require(moduleName);其中, moduleName 是要引入的模块的名称或路径。 require 函数的工作原理如下: 首先,Node.js 会检查 moduleName 是否是一个核心模块(Node.js 内置的模块)。如果是核心模块,Node.js 会直接加载并返回该模块的导出内容。 如果 moduleName 不是核心模块,Node.js...
        \ No newline at end of file +Balloon Party = 風船のパーティー = fuusen no party
        2.1k words 2 mins.

        # 编码的概念 广义的编码指的就是将信息从一种形式转换为另一种规范的形式; 对于数字电路,就是将现实世界中的各种物理信息转换为离散的电压序列,来表示、处理和分析信息; 事实上,进位计数制也是一种编码,用不同进制数来表示 “量”; 将数制与码制分开介绍,首先单独讲解数制是因为数制的一个重要的特点是其编码后能够参与算术运算;而一般来说码制是对非数值信息的表示; # 二进制编码 # 无符号 只能表示 0 和正数 # 有符号 (有符号)二进制数:在前面增加一个符号位,0 表示正数,1 表示负数;引入符号位后,一个二进制数的最高位是符号位,符号位的 0 / 1...
        3.7k words 3 mins.

        # 系统的描述 # 连续 / 离散 时间系统 # 连续时间系统 输入信号与输出响应都是连续时间信号的系统。 eg: 机械系统、电路系统 # 离散时间系统 输入信号与输出响应都是离散时间信号的系统。 eg: 人口问题 # 两种描述方法 # 输入输出描述法 (微分方程、差分⽅程) 着眼于系统输入与输出的关系,适⽤用于单输入 — 单输出系统。 # 状态变量描述法 (状态⽅程、输出方程) 即可描述输入与输出的关系,还可以描述系统的内部状态 既可⽤用于单输入 — 单输出系统,⼜可⽤于多输⼊ — 多输出系统。 #...
        754 words 1 mins.

        # 数制 # 数制的概念 数制 (numerical system),又称 “计数制”,指用一组固定的符号和统一的规则来表示数值的方法; 数制关注的内容:每一位的构成(使用什么符号即数码)和进位的规则; 基本概念: 基数 —— 使用的数码的个数,即每一位最多能用多少个符号来表示;基数是 N 则称为 N 进制, 即 “逢 N 进一” 权 —— 每一位的 1 代表的大小; # 常用数制 # 不同进制的转换 # 其他进制 -&gt; 十进制 只需要按照 D=∑kiNiD=\sum k_i N^iD=∑ki​Ni 展开即可; kik_iki​ 为第 i 位的系数 NiN^iNi...
        1.9k words 2 mins.

        # 人工智能的概念 人工智能(Artificial Intelligence - AI):研究如何通过计算机技术实现一个智能体,让智能体完成类似于人类智能的行为。 # 智能体 Agent 完成智能行为的主体,是物理或抽象实体,可以感知世界并对世界施加作用。 大模型智能体 LLM Agent 在近几年随 ChatGPT 兴起的新概念 # 智能体的两大目标 能否使智能体像人一样思考、理性地思考? 能否使智能体像人一样行动、理性地行动? # 智能体可否像人一样思考 首先,需要测量人如何思考的方法:内省、心理实验、 脑成像。 #...
        9.4k words 9 mins.

        # 能量 / 功率 信号 # 能量信号 能量为一有界值,即 0 &lt; E &lt; + ∞,此时 P = 0 有时求 P 不好求,但是 E 好求,可以证明 E 为有界的,从而得到 P = 0 # 功率信号 功率为一有界值,即 0 &lt; P &lt; + ∞ ,此时 E = ∞ 有时求 E 不好求,但是 P 好求,可以证明 P 为有界的,从而得到 E = ∞ # 信号的分解 #...
        27k words 25 mins.

        # 参考资料 Rust 语言圣经 (自称) 锈书 官网 # 环境配置 更新 Rust bashrustup update卸载 Rust bashrustup self uninstall查看版本 h bashrustc --version## 显示格式rustc.x.y.z([commit hash] [commit date])安装 Rust 同时还会安装文档,查看本地文档 bashrustup doc# Rust 工具 # 编译器 # rustc 只适合单文件编译 shellrustc xxx.rs &lt;output># cargo Cargo 是 Rust...
        12k words 11 mins.

        # 操作系统的定义 实际上操作系统没有普遍接受的定义 操作系统 (Operating System,OS)是指控制和管理整个计算机系统的硬件和软件资源,并合理地组织调度计算机的工作和资源的分配;以提供给用户和其他软件方便的接口和环境;它是计算机系统中最基本的系统软件。 # 操作系统的功能和目标 # 作为系统资源的管理者 # 提供的功能 处理机管理 存储器管理 文件管理 设备管理 # 目标 安全、高效 # 向上层提供方便易用的服务 # GUI (Graphical User Interface) 图形化用户接口 # CLI (Command Line Interface)...
        9.4k words 9 mins.

        # pthread 库 # 简介 pthread 是 POSIX 线程库,它是一种跨平台的线程库,可以在不同的操作系统上使用。它是 C 语言的标准库,提供了一组函数来创建、管理和同步线程。&quot;pthread&quot; 库相对于 &quot;thread&quot; 库来说,更加底层和与操作系统紧密相关。 thread 是 C11 标准提供的线程库,它是 C 的一部分。它提供了一种面向对象的方式来创建、管理和同步线程。&quot;thread&quot; 库提供了 std::thread...
        197 words 1 mins.

        # DBMS 数据库管理系统 DBMS 的分类: 關聯式資料庫(SQL) 非關聯式資料庫 - (nosQL /not just SQL) • MySQL • MogoDB • Oracle • Redis • PostgreSQL • DynamoDB • SQL Server • Elaticsearch 關聯式資料庫管理系統 (RDBMS) 非關聯式資料庫管理系統 (NRDBMS) SQL - Structured Query Language - 结构性询问语言
        7.4k words 7 mins.

        # JSON.stringify() JSON.stringify() 是 JavaScript 中的一个方法,用于将 JavaScript 对象或值转换为 JSON 字符串。 具体作用如下: 序列化: JSON.stringify() 方法将 JavaScript 对象或值序列化为一个 JSON 字符串。序列化是指将对象的结构和数据转换为字符串的过程,使其可以在网络传输或存储中进行传递和持久化。 JSON 格式:JSON(JavaScript Object Notation)是一种常用的数据交换格式,具有简洁、易读和跨语言的特性。 JSON.stringify() 将 JavaScript...
        \ No newline at end of file diff --git a/page/9/index.html b/page/9/index.html index 3f0b1a30..a1b385d7 100644 --- a/page/9/index.html +++ b/page/9/index.html @@ -1 +1 @@ -Balloon Party = 風船のパーティー = fuusen no party
        7.4k words 7 mins.

        # JSON.stringify() JSON.stringify() 是 JavaScript 中的一个方法,用于将 JavaScript 对象或值转换为 JSON 字符串。 具体作用如下: 序列化: JSON.stringify() 方法将 JavaScript 对象或值序列化为一个 JSON 字符串。序列化是指将对象的结构和数据转换为字符串的过程,使其可以在网络传输或存储中进行传递和持久化。 JSON 格式:JSON(JavaScript Object Notation)是一种常用的数据交换格式,具有简洁、易读和跨语言的特性。 JSON.stringify() 将 JavaScript...
        2.5k words 2 mins.

        # CMake 简介 CMake 官网:https://cmake.org WIKI:https://en.wikipedia.org/wiki/CMake CMake 自身也是一种编程语言,可以用它实现基本的程序逻辑,此如循环、条件分支等等,其中最典型的应用是根据工程的配置来选择性编译部分源代码,比如有些功能只有在测试版本中才会开启。 和其它的构建工具一样,CMake 的本质依然是定义各个目标之间的关联,比如您的项目由哪些可执行文件、动态库、资源文件组成,然后它们又是通过哪些源文件编译,用到了哪些外部关联等等,本身并没有编译和链接的功能。 除了基本的构建功能之外,CMake...
        239 words 1 mins.

        树莓派配置静态 ip h bashsudo nano /etc/dhcpcd.conf文件末尾添加: h bashinterface eth0static ip_address=192.168.134.32/24static routers=192.168.134.1static domain_name_servers=192.168.134.1 8.8.8.8其中 192.168.134.32 可以直接是路由器给树莓派分配的 ip 地址,作为静态 ip 地址 ctrl+s 保存,ctrl+x 退出,重启树莓派即可
        234 words 1 mins.

        这里是我的博客,整理收纳了一些笔记、思维导图、资料 您可以叫我 fuuzen 或者「風船」,这是我随便起的一个用的比较久的网名,没有什么特别含义。fuusen 即「ふうせん」,但这个 ID 已经在 github 被注册了,所以只好叫做 fuuzen 了 这个博客网站的名字是 Balloon Party 或者「風船のパーティー」,也是我随便起的一个名字,没有什么特别含义 如果这里所整理收纳的资料对您的学习有帮助的话,欢迎您向您的朋友们推荐本博客!也欢迎您向我指出资料中的任何错误或提出宝贵的意见!
        388 words 1 mins.

        # frp # Tailscale # Ngrok # 绑定自己的域名(要充钱) Ngrok uses the Host header sent by the browser to determine how to redirect your traffic you will need to &quot;reserve&quot; the domain you want to use in the Ngrok dashboard. Note that doing this requires a paid Ngrok plan. After reserving your...
        667 words 1 mins.

        查看 niginx 进程: ps -ef|grep nginx进程: master worker 查看 nginx 的:版本、安装目录 (--prefix)、编译参数 nginx -V重新加载配置文件: shellnginx -s reload查看配置文件 nginx.config 所在目录,验证默认配置文件,常用于检查配置文件有没有写对: nginx -t常见目录有: /etc/nginx/conf/usr/local/etc/nginx/opt/homebrew/etc/nginx停止 nginx 服务: nginx -s quit //优雅停止nginx -s...
        379 words 1 mins.

        Hexo 是一个基于 Node 框架的博客框架 pnpm install hexo-cli -g //安装hexo init blog //初始化cd blog; npm install //安装依赖blog 目录下生成静态文件(到 public 目录下,用于发布) hexo g //缩略hexo generate //一样blog 目录下清除已生成的静态文件和 cache( 到 public 目录下) hexo cleanblog 目录下本地运行 hexo 服务 hexo s //缩略hexo server //一样在 blog 目录下将...
        458 words 1 mins.

        # YatCPU 实验报告 2023 年秋季学期的计算机组成原理实验项目 YatCPU 的实验报告 # YatCPU 相关资料 YatCPU 的教学文档和仓库 2023 年秋季学期由 Tokisakix 整理的文档和仓库,新纳入了 docker 环境搭建脚本、zybo-z710 开发板移植及烧板脚本 # 实验报告 YatCPU - - - lab1 - - - lab2 - - - lab3 - - - lab4 # 关于挑战实验 阅读了 egos-2000 操作系统的文档和仓库,按照其教程实现了操作系统映像文件的编译和写入 microSD 卡的过程 但是 egos-2000...
        14k words 13 mins.

        YatCPU - - - lab1 - - - lab2 - - - lab3 - - - lab4 # 测试用例及其波形图分析 # SayGoodbyeTest src/test/scala/riscv/BoardTest.scala 其中 Top 模块来自: # 测试用例的功能 用于烧板验证,但由于 CPU 还未能实现中断,该程序只能通过 LED 灯闪烁判断 CPU 能否正常工作 在 lab1 中该测试用例可以用于检查 CPU 及其内部各个模块、模块内的端口信号是否有完整地实现、初始化,即 &quot;Fully...
        6.6k words 6 mins.

        YatCPU - - - lab1 - - - lab2 - - - lab3 - - - lab4 # 测试用例及其波形图分析 # BoardSayGoodbyeTest src/test/scala/riscv/BoardTest.scala 其中 Top 模块来自: 测试执行程序来自 csrc/say_goodbye.c : # 测试用例的功能 测试能否正常执行打印 &quot;Never gonna give you up~ Never gonna let you down~\nNever gonna run around and~ desert...
        \ No newline at end of file +Balloon Party = 風船のパーティー = fuusen no party
        1.4k words 1 mins.

        # require require 是 Node.js 中用于加载模块的函数。它的主要作用是将外部模块引入到当前的 JavaScript 文件中,以便在代码中使用被引入的模块的功能、变量或对象。 require 函数的基本语法是: require(moduleName);其中, moduleName 是要引入的模块的名称或路径。 require 函数的工作原理如下: 首先,Node.js 会检查 moduleName 是否是一个核心模块(Node.js 内置的模块)。如果是核心模块,Node.js 会直接加载并返回该模块的导出内容。 如果 moduleName 不是核心模块,Node.js...
        2.5k words 2 mins.

        # CMake 简介 CMake 官网:https://cmake.org WIKI:https://en.wikipedia.org/wiki/CMake CMake 自身也是一种编程语言,可以用它实现基本的程序逻辑,此如循环、条件分支等等,其中最典型的应用是根据工程的配置来选择性编译部分源代码,比如有些功能只有在测试版本中才会开启。 和其它的构建工具一样,CMake 的本质依然是定义各个目标之间的关联,比如您的项目由哪些可执行文件、动态库、资源文件组成,然后它们又是通过哪些源文件编译,用到了哪些外部关联等等,本身并没有编译和链接的功能。 除了基本的构建功能之外,CMake...
        239 words 1 mins.

        树莓派配置静态 ip h bashsudo nano /etc/dhcpcd.conf文件末尾添加: h bashinterface eth0static ip_address=192.168.134.32/24static routers=192.168.134.1static domain_name_servers=192.168.134.1 8.8.8.8其中 192.168.134.32 可以直接是路由器给树莓派分配的 ip 地址,作为静态 ip 地址 ctrl+s 保存,ctrl+x 退出,重启树莓派即可
        234 words 1 mins.

        这里是我的博客,整理收纳了一些笔记、思维导图、资料 您可以叫我 fuuzen 或者「風船」,这是我随便起的一个用的比较久的网名,没有什么特别含义。fuusen 即「ふうせん」,但这个 ID 已经在 github 被注册了,所以只好叫做 fuuzen 了 这个博客网站的名字是 Balloon Party 或者「風船のパーティー」,也是我随便起的一个名字,没有什么特别含义 如果这里所整理收纳的资料对您的学习有帮助的话,欢迎您向您的朋友们推荐本博客!也欢迎您向我指出资料中的任何错误或提出宝贵的意见!
        388 words 1 mins.

        # frp # Tailscale # Ngrok # 绑定自己的域名(要充钱) Ngrok uses the Host header sent by the browser to determine how to redirect your traffic you will need to &quot;reserve&quot; the domain you want to use in the Ngrok dashboard. Note that doing this requires a paid Ngrok plan. After reserving your...
        667 words 1 mins.

        查看 niginx 进程: ps -ef|grep nginx进程: master worker 查看 nginx 的:版本、安装目录 (--prefix)、编译参数 nginx -V重新加载配置文件: shellnginx -s reload查看配置文件 nginx.config 所在目录,验证默认配置文件,常用于检查配置文件有没有写对: nginx -t常见目录有: /etc/nginx/conf/usr/local/etc/nginx/opt/homebrew/etc/nginx停止 nginx 服务: nginx -s quit //优雅停止nginx -s...
        379 words 1 mins.

        Hexo 是一个基于 Node 框架的博客框架 pnpm install hexo-cli -g //安装hexo init blog //初始化cd blog; npm install //安装依赖blog 目录下生成静态文件(到 public 目录下,用于发布) hexo g //缩略hexo generate //一样blog 目录下清除已生成的静态文件和 cache( 到 public 目录下) hexo cleanblog 目录下本地运行 hexo 服务 hexo s //缩略hexo server //一样在 blog 目录下将...
        458 words 1 mins.

        # YatCPU 实验报告 2023 年秋季学期的计算机组成原理实验项目 YatCPU 的实验报告 # YatCPU 相关资料 YatCPU 的教学文档和仓库 2023 年秋季学期由 Tokisakix 整理的文档和仓库,新纳入了 docker 环境搭建脚本、zybo-z710 开发板移植及烧板脚本 # 实验报告 YatCPU - - - lab1 - - - lab2 - - - lab3 - - - lab4 # 关于挑战实验 阅读了 egos-2000 操作系统的文档和仓库,按照其教程实现了操作系统映像文件的编译和写入 microSD 卡的过程 但是 egos-2000...
        6.6k words 6 mins.

        YatCPU - - - lab1 - - - lab2 - - - lab3 - - - lab4 # 测试用例及其波形图分析 # BoardSayGoodbyeTest src/test/scala/riscv/BoardTest.scala 其中 Top 模块来自: 测试执行程序来自 csrc/say_goodbye.c : # 测试用例的功能 测试能否正常执行打印 &quot;Never gonna give you up~ Never gonna let you down~\nNever gonna run around and~ desert...
        4.9k words 4 mins.

        YatCPU - - - lab1 - - - lab2 - - - lab3 - - - lab4 # 测试用例及其波形图分析 # SayGoodbyeTest src/test/scala/riscv/BoardTest.scala 其中 Top 模块来自: 测试执行程序来自 csrc/say_goodbye.c : # 测试用例的功能 测试能否正常执行打印 &quot;Never gonna give you up~ Never gonna let you down~\nNever gonna run around and~ desert...
        \ No newline at end of file diff --git a/rss.xml b/rss.xml index 6c2178b1..ea2033a6 100644 --- a/rss.xml +++ b/rss.xml @@ -2167,6 +2167,30 @@ A\gamma(z)&amp;=By[-1]+Cy[-2]+D\chi(z)\\ </tbody> </table> <script type="text&#x2F;javascript" src="https://unpkg.com/kity@2.0.4/dist/kity.min.js"></script><script type="text&#x2F;javascript" src="https://unpkg.com/kityminder-core@1.4.50/dist/kityminder.core.min.js"></script><script defer="true" type="text&#x2F;javascript" src="https://unpkg.com/hexo-simple-mindmap@0.8.0/dist/mindmap.min.js"></script><link rel="stylesheet" type="text&#x2F;css" href="https://unpkg.com/hexo-simple-mindmap@0.8.0/dist/mindmap.min.css"> ]]> + + + https://fuuzen.github.io/cs/os/ysosv2/lab5/ + YSOSv2: lab5 实验报告 + https://fuuzen.github.io/cs/os/ysosv2/lab5/ + + + + + + + Fri, 24 May 2024 00:00:00 +0800 + https://fuuzen.github.io/math/discrete-math/graph/match/ @@ -2342,30 +2366,6 @@ l(v),&amp;\text{others}\\ <p>该算法把上面原来的匈牙利算法用于其中,主要是用来判定和求完美匹配。</p> </blockquote> <script type="text&#x2F;javascript" src="https://unpkg.com/kity@2.0.4/dist/kity.min.js"></script><script type="text&#x2F;javascript" src="https://unpkg.com/kityminder-core@1.4.50/dist/kityminder.core.min.js"></script><script defer="true" type="text&#x2F;javascript" src="https://unpkg.com/hexo-simple-mindmap@0.8.0/dist/mindmap.min.js"></script><link rel="stylesheet" type="text&#x2F;css" href="https://unpkg.com/hexo-simple-mindmap@0.8.0/dist/mindmap.min.css"> ]]> - - - https://fuuzen.github.io/cs/os/ysosv2/lab5/ - YSOSv2: lab5 实验报告 - https://fuuzen.github.io/cs/os/ysosv2/lab5/ - - - - - - - Fri, 24 May 2024 00:00:00 +0800 - https://fuuzen.github.io/ie/de/memory-storage/ diff --git a/si-zheng/ma-yuan/ke-hou-xi-ti/index.html b/si-zheng/ma-yuan/ke-hou-xi-ti/index.html index ee25b3e4..d81608ba 100644 --- a/si-zheng/ma-yuan/ke-hou-xi-ti/index.html +++ b/si-zheng/ma-yuan/ke-hou-xi-ti/index.html @@ -1 +1 @@ -马原课后习题 - 马克思主义基本原理 - 思政 | Balloon Party = 風船のパーティー = fuusen no party

        # 思维导图

        # 绪论部分

        绪论部分

        # 第一章 世界的物质性及发展规律

        第一章 世界的物质性及发展规律

        # 第二章 实践与认识及其发展规律

        第二章 实践与认识及其发展规律

        # 第三章 人类社会及其发展规律

        第三章 人类社会及其发展规律

        # 第四章 资本主义的本质及规律

        第四章 资本主义的本质及规律

        # 第五章 资本主义的发展及其趋势

        第五章 资本主义的发展及其趋势

        # 第六章社会主义的发展及其规律(本章仅供学习参考,不作考试要求)

        第六章 社会主义的发展及其规律(本章仅供学习参考,不作考试要求)

        # 第七章 共产主义是人类最崇高的社会理想

        第七章 共产主义是人类最崇高的社会理想

        # 大纲

        # 绪论部分

        # 什么是马克思主义?

        • a) 从创造者、继承者的认识成果讲,马克思主义是由马克思、恩格斯创立的,由其后继者不断丰富和发展的观点和学说的科学理论体系。

        • b) 从阶级属性讲,马克思主义是关于无产阶级解放、全人类解放和每个人自由而全面发展的学说,是指引人民创造美好生活的行动指南,具有鲜明的阶级性。

        • c) 从研究对象和主要内容讲,马克思主义是无产阶级的科学世界观和方法论,是关于自然、社会和人类思维发展的一般规律的学说,是关于社会主义必然代替资本主义、最终实现共产主义的学说。

        • d) 马克思主义哲学、马克思主义政治经济学和科学社会主义的有机统一并共同构成了马克思主义理论的主体内容。

        # 马克思主义的直接的理论来源?

        • a) 资本主义生产方式造成社会化大生产迅猛发展的同时也带来了严重的社会灾难(社会两极分化;周期性经济危机频发)

        • b) 无产阶级在反抗资产阶级剥削和压迫的斗争中,逐步走向自觉,并迫切渴望科学的理论指导

        • c) 19C 西欧三大先进思潮为马克思主义的创立提供直接的理论来源(德国古典哲学、英国古典政治经济学和英法两国的空想社会主义)

        • d) 19C 三大科学发现也为马克思主义的产生提供了自然科学前提(细胞学说、能量守恒和转化定律)

        • e) 还有主观条件,结合马恩二人的生平谈就行

        # 马克思主义的鲜明特征?

        • a) 科学性

          • a.马克思主义是对自然、社会和人类思维发展本质和规律的正确反映。它是在社会实践和科学发展的基础上产生的,并在自身发展过程中不断总结实践经验,吸取自然科学和社会科学发展的最新成就。

          • b.马克思主义具有科学的世界观和方法论基础,即辩证唯物主义和历史唯物主义,这是马克思主义的一个突出特征和理论优势,也是马克思主义科学性的重要体现。

          • c.马克思主义理论是一个逻辑严密的有机整体,它以事实为依据、以规律为对象,并以实践为检验标准。马克思主义的发展具有科学探索性,是一个不断探索和掌握客观规律的过程。

        • b) 人民性

          • a.人民至上是马克思主义的政治立场。马克思主义政党把人民放在心中最高位置,一切奋斗都致力于实现最广大人民的根本利益,这是马克思主义最鲜明的政治立场。

          • b.人民群众是历史的创造者,是社会主义事业的依靠力量,马克思主义的人民性是以阶级性为深刻基础的,是无产阶级先进性的体现。

          • c.马克思主义是关于无产阶级解放的学说。无产阶级解放和全人类解放是完全一致的。

        • c) 实践性

          • a.马克思主义具有突出的实践精神,它始终强调理论与实践的统一,始终坚持与社会主义实际运动紧密结合。马克思主义是从实践中来,到实践中去,在实践中接受检验,并随实践的发展而不断发展的学说。

          • b.从马克思主义的使命和作用来说,它不是一种纯粹解释世界的学说,而是直接服务于无产阶级和人民群众改造世界的实践活动的科学理论。

          • c.从马克思主义的内容来看,实践观点是马克思主义首要的和基本的观点,这一基本观点体现在马克思主义全部思想内容之中。

        • d) 发展性

          • a.马克思主义是不断发展的学说,具有与时俱进的理论品质。马克思主义是时代的产物,并随着时代、实践和科学的发展而不断发展。

          • b.马克思主义理论体系是开放的,它不断吸取人类最新的文明成果来充实和发展自己。

          • c.马克思主义在指导人们认识世界和改造世界的过程中,在指导社会主义事业发展的过程中,不断与时代特征和各国具体实际相结合,得到丰富和发展,并形成新的理论成果。

        # 结合当代世界所面临的课题和当代青年所肩负的使命,谈谈马克思主义的当代价值和指导意义?

        • (1)当代世界所面临的新课题和当代青年所肩负的新使命

          • a) 在经济全球化的背景下,资本主义内部矛盾越来越突出,经济政治发展的不平衡加剧了资本主义世界的两极分化,贫富差距日益扩大。20 世纪 90 年代末的东南亚金融危机以及由华盛顿开始波及全球的 2008 年全球金融危机,都暴露了资本主义的缺陷。当代世界的新课题是和平与发展,各国都希望在一个相对稳定的环境中谋求国内经济发展,提高人们的生活水平。因此,马克思主义思想对现代社会仍有指导意义。

          • b) 不同的时代有不同的历史责任与使命,当代青年是在中国改革开放新时期成长起来的,处在一个继往开来的历史时期,肩负着祖国人民更多的期望,也应肩负起民族复兴的重任。当代青年所处的时代正是社会主义现代化建设的关键时刻,中华民族的伟大复兴对当代青年提出了新的挑战,我们有责任也有义务来维护祖国的统一,保卫祖国,建设祖国。而大学生作为当代青年的主体和中坚力量,更应该发挥引领带头作用。

          • c) 总而言之,马克思主义作为具有科学性、真理性的学说,对当代青年的人生发展有重要的指引作用,有助于青年们树立正确的世界观、人生观和价值观,提高其应对和处理问题的能力。

        • (2)马克思主义的当代价值

          • ① 当代世界变化的认识工具

            • a.马克思主义给予我们观察当代世界的宏大视野。

            • b.马克思主义给予我们透视时代风云的锐利目光。

            • c.马克思主义给予我们展望未来世界的长远眼光和战略定力指引。

          • ② 当代中国发展的行动指南

            • a.马克思主义是指引当代中国发展的精神旗帜。

            • b.马克思主义是推动当代中国发展的精神动力。

            • c.马克思主义是引领当代中国实践的行动指南。

          • ③ 引领人类社会进步的科学真理

            • a.从人类历史发展的大视野来看,世界仍然处于马克思主义所指明的从资本主义走向社会主义的大时代。马克思主义所揭示的资本主义基本矛盾仍然存在,而且在近年来西方的金融危机和社会危机中呈现出某种激化的趋势。

            • b.人类的未来仍然需要马克思主义的启迪和指引。社会是在矛盾中进步的,每一个时代的社会进步总是伴随着相应的社会问题,面对当今社会的重大问题,还是需要从马克思主义中寻找智慧。

        • (3)对当代青年的指导意义

          • 青年是祖国的未来、事业的希望。青年强则国家强,青年一代有理想、有本领、有担当,国家就有前途,民族就有希望。学习马克思主义理论可以帮助青年提高分析和解决问题的能力,有助于其树立正确的世界观、人生观和价值观。当代青年在学习马克思主义的过程中,要有正确的态度和科学的方法:

            • ① 努力学习和掌握马克思主义的基本立场、观点、方法。
            • ② 坚持理论联系实际的马克思主义学风。
            • ③ 自觉将马克思主义内化于心、外化于行。

        # 阅读感受:

        • 《资本论》内容宏大,理论深奥,创造了一个崭新的思想体系。其研究世界的方法源于德国古典哲学、英法两国的空想社会主义和英国古典政治经济学。马克思吸收了黑格尔的辩证法思想,相信能够用一个辩证法公式概括人类的进化历程。他认为,所有哲学家所做的一切都在致力于解释世界,但他同时认为,问题的关键在于如何改造世界。马克思在黑格尔辩证法的基础之上,颠覆了传统的 “形而上学”,建立了一个现实中得以实践的最大的思想体系。《资本论》以唯物史观的基本思想为指导,通过深刻分析资本主义生产方式,揭示了资本主义社会发展的规律,同时也使唯物史观得到了科学的验证和进一步的丰富和发展。它运用唯物史观的观点和方法,将社会关系归结为生产关系,将生产关系归结于生产力水平,从而证明了社会形态的发展是一个不以人的意志为转移的自然历史过程。

        # 第一章 世界的物质性及发展规律

        # 马克思主义的物质范畴的理论意义

        • (1)马克思主义的物质观
          马克思主义认为,世界是物质的世界,物质是第一性,物质是不依赖于人类的意识而存在,并能为人类的意识所反映的客观存在。“物质” 范畴是唯物主义哲学关于世界本原和统一性的最高抽象,是唯物主义世界观的基石。马克思主义批判继承了以往唯物主义的传统,在总结现代自然科学重大成就的基础上,创立了辩证唯物主义的物质观。20 世纪初,列宁对物质概念做出了科学的概括:“物质是标志客观实在的哲学范畴,这种客观实在是人通过感觉感知的,它不依赖于我们的感觉而存在,为我们的感觉所复写、摄影、反映。”

        • (2)马克思主义的物质观具有深刻的理论意义

          • ① 坚持唯物主义一元论,同唯心主义一元论和二元论划清了界限。马克思主义指明了物质对于意识的独立性、根源性,以及意识对于物质的依赖性、派生性。因为意识不过是物质的反映,不能离开物质而独立存在,所以意识不可能成为世界的另一种本原。

          • ② 坚持了能动的反映论和可知论,批判了不可知论。目前世界上还有很多事物未被人类认识,但这并不意味着它们不可认识。人们已经认识到未知世界与已知世界都是客观存在的。世界上只有尚未认识之物,没有不可认识之物。

          • ③ 体现了唯物论和辩证法的统一,克服了形而上学唯物主义的缺陷。马克思主义的物质观认为,物质的唯一特性是客观实在性,既肯定了哲学物质范畴同自然科学物质结构理论的联系,又将它们相区别。从个性中看到共性,从暂时中发现永恒,从相对中找到绝对,这是马克思主义物质观体现的唯物辩证法。

          • ④ 体现了唯物主义自然观与唯物主义历史观的统一,为彻底的唯物主义奠定了理论基础。马克思主义的物质观建立了统一说明自然历史过程的唯物主义原则,揭示了自然和社会的物质性,实现了唯物主义自然观和历史观的辩证统一。

        # 物质与意识的辩证关系

        物质决定意识,意识对物质具有反作用,这种反作用就是意识的能动作用。

        • (1)物质决定意识
          意识是人脑的机能和属性,是客观世界的主观映像。物质对意识的决定作用表现在意识的起源、本质和作用上。

          • ① 从意识的起源来看,其形成和发展经历了三个阶段,即:由一切物质所具有的反应特性到低等生物的刺激感应性,再到高等动物的感觉和心理,最终发展成人类的意识。意识不仅是自然界长期发展的产物,而且是社会历史发展的产物。劳动为意识的产生和发展提供了客观需要和可能;人们在劳动和交往中形成的语言促进了意识的发展。

          • ② 从意识的本质来看,意识是客观世界的主观映像。马克思指出:“观念的东西不外是移入人的头脑并在人的头脑中改造过的物质的东西而已。”

        • (2)意识对物质具有反作用
          物质决定意识,意识的能动作用是指意识对物质具有反作用。意识的能动作用主要表现在:

          • ① 意识活动具有目的性和计划性。马克思认为:“人在劳动过程结束时得到的结果,在这个过程开始时就已经在劳动者的表象中存在着,即已经观念地存在着”。人的整个实践过程,就是围绕意识活动所构建的目标和蓝图来进行的。

          • ② 意识活动具有创造性。人的意识可以对客观世界中的感性材料加工和建构,在思维中构造一个现实中没有的理想世界。

          • ③ 意识具有指导实践改造客观世界的作用。在实践中形成观念,以这些观念为指导,通过实践使之变成客观现实。④ 意识具有调控人的行为和生理活动的作用。现代科学证明,意识、心理因素对人的行为选择和健康状况能够产生重要影响。

        • (3)主观能动性和客观规律性的统一
          正确认识和把握物质与意识的辩证关系,需要处理好主观能动性与客观规律性的关系,原因在于:

          • ① 尊重客观规律是正确发挥主观能动性的前提。人们正确地认识世界,有效地改造世界的前提是认识和掌握客观规律。

          • ② 只有充分发挥主观能动性,才能正确认识和利用客观规律。承认规律的客观性,并不是说人在规律面前无能为力、无所作为。人类能够按照客观规律改造世界,以达到自己的目的。

        # 运用矛盾的普遍性和特殊性辩证关系原理,说明把马克思主义基本真理与中国具体实际相结合的重要性。答:

        • (1)矛盾普遍性与特殊性的辩证关系

          • ① 矛盾的普遍性即矛盾的共性,矛盾的特殊性即矛盾的个性。矛盾的共性是无条件的、绝对的;矛盾的个性是有条件的、相对的。

          • ② 任何现实存在的事物的矛盾都是共性和个性的有机统一,共性寓于个性之中,没有离开个性的共性,也没有离开共性的个性。矛盾的共性和个性相统一的关系,既是客观事物固有的辩证法,也是科学的认识方法。

        • (2)把马克思主义基本真理与中国具体实际相结合的重要性
          矛盾的普遍性和特殊性辩证关系的原理,是把马克思主义普遍真理同各国具体实际相结合的理论基础,也是建设中国特色社会主义的理论根据。

          • ① 矛盾的普遍性和特殊性辩证统一的原理,是把马克思主义普遍真理与中国的具体实际相结合,走建设有中国特色社会主义道路的重要理论依据。矛盾的普遍性和特殊性的关系就是共性与个性、绝对与相对的关系。我国走中国特色社会主义道路,是中国共产党把马克思主义普遍真理同中国的具体实际相结合的过程中实现的第二次飞跃。在这里,“中国特色” 是中国不同于其他社会主义国家的个性、特殊性。“社会主义” 则是中国和其他社会主义国家的共性、普遍性。

          • ② 根据共性与个性相统一的原理,我们坚持中国特色社会主义道路。

            • a.不管中国的情况如何特殊,都必须坚持马克思主义的普遍真理,坚持社会主义根本制度和基本原则,这是共性。

            • b.社会主义的一般只能在各国特色的个别中存在,我们又必须建设中国特色的社会主义,从中国的国情出发,注意把握中国的特点。只有坚持党的基本路线不动摇,走自己的路,我们才能在建设中国特色社会主义的道路上不断取得胜利,这是个性。

        # 不断增强思维能力:辩证思维能力、历史思维能力、战略思维能力、底线思维能力、创新思维能力。

        答:创新思维能力是对常规思维的突破,是破除迷信、超越陈规,善于因时制宜、知难而进、开拓创新的能力。

        • (1)培养和提高创新思维能力的重要性

          • ① 坚持创新思维,才能推动发展、取得进步。“发展的观点” 是唯物辩证法的基本观点,马克思主义关于人类社会不断进步的发展观,不仅揭示了人类社会历史是不断向前发展的,而且说明了正是人类的创新思维和创造智慧推动了社会更好地发展。

          • ② 创新是一个民族进步的灵魂,是一个国家兴旺发达的不竭动力,也是中华民族最深沉的民族禀赋。这些年中华民族进步和贡献的取得,正是中华民族勇于创新和善于创新的结果。失去了创新思维,就会失去前进的动力,导致封闭和落后。

          改革开放以来,中国共产党高度重视思想解放和理论创新,开辟了中国特色社会主义道路,形成了中国特色社会主义理论体系,确立了中国特色社会主义制度,发展了中国特色社会主义文化,实现了中国人民从站起来到富起来、强起来的伟大飞跃。从某种意义上说,这些成就正是我们党坚持创新思维、大力推进理论创新和实践创新的结果。

        • (2)提高自己的创新思维能力的做法

          • ① 提高创新思维能力,必须坚决反对各种形而上学的思维方式,做到不唯书、不唯上,只唯实。提高创新思维能力,需要坚持和运用唯物辩证法,坚决反对形而上学。

            • a.要破除对 “本本” 的迷信;

            • b.要破除对 “经验” 的迷信;

            • c.要破除对 “权威” 的迷信。

          • ② 问题意识是提高创新思维能力的起点和动力源。

          增强问题意识、坚持问题导向,就是承认矛盾的普遍性客观性,就是善于把认识和化解矛盾作为打开工作局面的突破口,善于具体问题具体分析,透过现象看本质,抓主要矛盾和矛盾的主要方面,科学分析问题,深入研究问题。

          • ③ 善于继承才能善于创新。

            • a.创新思维不是凭空产生的,善于继承,才能将过去、现在和未来联系起来;

            • b.善于创新,才会有现在超越过去、未来超越现在的前进运动。

            • c.中国特色社会主义是科学社会主义理论逻辑和中国社会发展历史逻辑的辩证统一,是中国共产党人运用创新思维,在继承和发展中形成的创新成果。

          • ④ 在解放思想中锻造创新思维。

            • a.提高创新思维能力,就是要有敢为人先的锐气,打破惯性思维,以思想认识的新飞跃打开工作的新局面;

            • b.提高创新思维的过程,就是思想解放的过程。没有思想的大解放,就不可能有创新思维的大突破。

        # 第二章 实践与认识及其发展规律

        # 实践与认识的辩证关系及其规律

        • (1)毛泽东《实践论》的主要观点:
          毛泽东强调:“一个正确的认识,往往需要经过由物质到精神,由精神到物质,即由实践到认识,由认识到实践这样多次的反复,才能够完成。” 如此 “实践、认识、再实践、再认识,循环往复以至无穷,而实践和认识之每一循环的内容,都比较地进到了高一级的程度”。这就是认识辩证运动发展的基本过程,也是认识运动的总规律。认识是一个反复循环和无限发展的过程,是一个波浪式前进和螺旋式上升的过程。认识在实践基础上不断深化发展,并推动实践在认识指导下不断深入推进。这个过程既不是封闭式的循环,也不是直线式的发展,而是波浪式前进和螺旋式上升的。

        • (2)实践与认识的辩证关系

          • ① 实践是认识的基础,实践在认识活动中起决定性作用。主要表现在以下四个方面:

            • a.实践是认识的来源。认识的内容是在实践活动的基础上产生和发展的。只有在实践中实际改造和变革对象,人们才能形成对认识对象属性、本质、规律更为准确的认识,并以这种认识指导实践活动。

            • b.实践是认识发展的动力。认识的产生和发展源于实践的需要,也正是这种需要推动人类的科学发现和技术发明,推动人类的思想进步和理论创新。实践的需要是推动认识在深度和广度上不断发展之根本。此外,实践是认识发展的动力,还表现为实践为认识的发展提供了如经验资料、实验仪器和工具等的手段和条件。另外,实践改造了人的主观世界,锻炼和提高了人的认识能力。人们正是在实践的推动下,不断突破旧认识、旧思想,引起认识上的新飞跃,从而不断有新发现、有所前进。

            • c.实践是认识的目的。通过实践获得认识,其最终目的是为实践服务,指导实践,以满足生活和生产的需要。

            • d.实践是检验认识真理性的唯一标准。真理不是自封的。判断认识的真理性,不是依主观感觉,而是依客观上社会实践的结果。

          • ② 认识反作用于实践。

            • a.人的实践活动是受人的意识支配的,实践的这种本质特性决定着它不能离开认识的指导。

            • b.认识活动及其成果有相对独立性,遵循其特有的活动方式和发展规律,它一经形成,便能反作用于实践。

            • c.认识对实践的指导作用表现在:正确的认识能够推动实践的发展,而错误的认识则会对实践起阻碍作用。

        # 真理、真理的绝对性和相对性及其辩证关系、真理与谬误(自学);

        • (1)真理是绝对的又是相对的原因
          真理是一个过程。就真理的发展过程以及人们对它的认识和掌握程度来说,真理既具有绝对性,又具有相对性,它们是同一客观真理的两
          种属性,这是真理问题上的辩证法。任何真理都是绝对性和相对性的统一,二者相互联系、不可分割。两者关系具体如下:

          • ①真理的绝对性

            真理的绝对性是指真理主客观统一的确定性和发展的无限性。它有两个方面的含义:

            • a.任何真理都标志着主观与客观之间的符合,都包含着不依赖于人和人的意识的客观内容,都同谬误有原则的界限。这一点是绝对的、无条件的。在这个意义上,承认了真理的客观性也就是承认了真理的绝对性。

            • b.人类认识按其本性来说,能够正确认识无限发展着的物质世界,认识每前进一步,都是对无限发展着的物质世界的接近,这一点也是绝对的、无条件的。在这个意义上,承认了世界的可知性,承认人能够获得关于无限发展着的物质世界的正确认识,也就是承认了真理的绝对性。

          • ②真理的相对性

            真理的相对性是指人们在一定条件下对客观事物及其本质和发展规律的正确认识总是有限度的、不完善的。它具有两个方面的含义:

            • a.从客观世界的整体来看,任何真理都只是对客观世界的某一阶段、某一部分的正确认识,人类已经达到的认识的广度总是有限度的,因而,认识有待扩展。

            • b.就特定事物而言,任何真理都只是对客观对象一定方面、一定层次和一定程度的正确认识,认识反映事物的深度是有限度的,或是近似性的。因而,认识有待深化。任何真理都只能是主观对客观事物近似正确即相对正确的反映。

          • ③真理的绝对性和相对性的关系

            真理的绝对性和相对性是辩证统一的,具体如下:

            • a.二者相互依存。即人们对于客观事物及其本质和规律的每一个正确认识,都是在一定范围内、一定程度上、一定条件下的认识,因而是相对的和有局限的;但是,在这一定范围内、一定程度上、一定条件下,它也是对客观对象的正确反映,因而它也是无条件的、绝对的。

            • b.二者相互包含。一是真理的绝对性寓于真理的相对性之中。任何真理所包含的客观内容都只能是人们在特定历史条件下所把握到的,都只是对客观世界及其事物的一定范围、一定程度的正确反映。二是真理的相对性包含并表现着真理的绝对性。任何真理都与谬误有本质的区别,标志着人们在一定范围内和一定层次上达到对于无限发展着的物质世界的正确认识,包含着确定的客观内容。毛泽东把真理的绝对性与相对性的关系比喻为长河与水滴的关系,“马克思主义者承认,在绝对的总的宇宙发展过程中,各个具体过程的发展都是相对的,因而在绝对真理的长河中,人们对于在各个一定发展阶段上的具体过程的认识只具有相对的真理性。无数相对的真理之总和,就是绝对的真理”。

            • c.相对性向绝对性转化。真理永远处在由相对向绝对的转化和发展中,是从真理的相对性走向绝对性、接近绝对性的过程。任何真理性的认识都是由真理的相对性向绝对性转化过程中的一个环节,这是真理发展的规律。真理的绝对性与相对性根源于人认识世界的能力的无限性与有限性、绝对性与相对性的矛盾。人类的思维按其本性、可能和历史的终极目的来说,是能够认识无限发展着的物质世界的,思维是无限的和绝对的。但是,具体到每一个人乃至每一代人,由于受到客观事物及其本质的显露程度、社会历史的实践水平、主观的条件以及生命的有限性等各方面的限制,其思维也是有限的和相对的。人的认识能力、思维能力是无限性与有限性、绝对性与相对性的对立统一,作为人的正确认识成果的真理,也必然是绝对性和相对性的对立统一。

        • (2)真理的绝对性和相对性相统一对马克思主义的重要意义

          • ① 割裂真理绝对性与相对性之间的辩证关系,就会走向形而上学的真理观,即绝对主义和相对主义。绝对主义在实际工作中表现为教条主义、思想僵化,把马克思主义当作一成不变的公式。相对主义否认真理内容的客观性,把真理的相对性歪曲成了主观随意性,由此陷入主观真理论,导致不可知论和诡辩论。

          • ② 马克思主义是客观真理,是绝对性和相对性的统一

            马克思主义正确地反映了人类社会发展的规律,是经过长期革命、建设和改革的实践所证明了的真理,因而具有绝对性的一面。但是,马克思主义并没有穷尽对一切事物及其规律的认识,仍然需要随着社会实践的发展而发展,因而又具有相对性的一面。正因为马克思主义具有绝对性,所以我们必须坚持以马克思主义为指导思想;也因为它具有相对性,所以我们必须在实践中丰富和发展马克思主义。既坚持又发展,才是对待马克思主义的正确态度。

        # 真理和价值的辩证关系。

        • (1)真理尺度和价值尺度的含义

          • ① 人们的实践活动总是受着真理尺度和价值尺度的制约。

          • ② 实践的真理尺度是指在实践中人们必须遵循正确反映客观事物本质和规律的真理。只有按照真理办事,才能在实践中取得成功。

          • ③ 实践的价值尺度是指在实践中人们都是按照自己的尺度和需要去认识世界和改造世界。这一尺度体现了人的活动的目的性。

        • (2)真理尺度与价值尺度的关系

          • ① 任何实践活动都要受到这两种尺度的共同制约,任何活动都是合规律性和合目的性的统一。

          • ② 真理与价值或真理尺度与价值尺度之间是紧密联系、不可分割的。一方面,价值尺度必须以真理为前提。脱离了真理尺度,价值尺度就偏离了正确的轨道。另一方面,人类满足自身需要的内在尺度,推动新的真理不断被发现。脱离了价值尺度,真理就缺失了主体意义。

        • (3)帮助和启示
          因为实践具有具体性和历史性,真理尺度与价值尺度的统一也是具体的和历史的,二者的统一会随着实践的发展而不断发展到更高级的程度,真理由相对走向绝对,人的需要和利益也日益呈现出多元化特征。真理尺度与价值尺度是否达到了具体的、历史的统一,必须通过实践来验证。新时代中国特色社会主义的实践充分体现了真理尺度与价值尺度的辩证统一。正如习近平所指出的:“中国共产党人的理想信念,建立在马克思主义科学真理的基础之上,建立在马克思主义揭示的人类社会发展规律的基础之上,建立在为最广大人民谋利益的崇高价值的基础之上。我们坚定,是因为我们追求的是真理。我们坚定,是因为我们遵循的是规律。我们坚定,是因为我们代表的是最广大人民的根本利益。”

        # 第三章 人类社会及其发展规律

        # 为什么说物质资料的生产方式是社会发展的决定性力量?

        答:物质资料的生产方式是社会发展的决定性力量,主要是基于以下原因:

        • (1)物质资料的生产方式是人类其他一切活动的首要前提,是人类社会赖以存在和发展的物质基础。人们为了能够创造历史,必须能够生活,为了生活,首先就需要衣、食、住及其他东西。因此,人类历史的第一个历史活动就是生产满足这些需要的物质资料本身。这是一个简单的事实和起码的真理。发现并承认这一真理,是历史观的一个伟大革命。

        • (2)物质资料的生产方式决定着社会的结构、性质和面貌。有什么样的生产方式便有什么样的社会形态。正如马克思所说:“手推磨产生的是封建主为首的社会,蒸汽磨产生的是工业资本家为首的社会。”

        • (3)生产方式的发展变化决定着社会历史的发展变化和形态更替。在物质资料的生产方式中,生产力是最活跃、最能动的因素,它总是要向前发展的,而生产关系则是相对稳定的。随着生产力的发展,原有的生产关系便由生产力发展的形式变成生产力发展的桎梏,由适合生产力的发展变成阻碍生产力的发展。只有变革生产关系,才能解放和发展生产力。随着生产关系(即经济基础)的变革,全部的、庞大的上层建筑也会或迟或早地发展变革,从而引起社会形态的更替。

        • 总之,人类社会的历史就是物质资料生产的历史,是劳动发展史。马克思正是从劳动发展史中找到了理解人类社会发展史的 “钥匙”。

        # 运用社会基本矛盾运动的原理分析深化改革的客观依据与重要意义。答:

        • 与深化改革最相近的是生产力与生产关系的矛盾运动。

          • 社会基本矛盾运动包括

            • 生产力与生产关系之间的矛盾

            • 经济基础与上层建筑之间的矛盾

          • 它们是贯穿社会发展过程始终,规定社会发展过程的基本性质和基本趋势,并对社会历史发展起根本推动作用的矛盾。

          • 因此,社会基本矛盾的解决对历史发展具有重要作用。

          • 而要解决社会基本矛盾,改革是一种重要的方式和途径。

        • (1)客观依据
          社会基本矛盾,特别是生产力与生产关系的矛盾在历史发展中起根本的推动作用,为了推动社会发展,必须解决好社会基本矛盾,改革就是其中一种重要方式。深化改革的客观依据就在于社会基本矛盾在历史发展过程中的重要作用,主要表现在:

          • ① 生产力是社会基本矛盾运动中最基本的动力因素,是人类社会发展和进步的最终决定力量。

          • ② 社会基本矛盾特别是生产力和生产关系的矛盾,是 “一切历史冲突的根源”,决定着社会中其他矛盾的存在和发展。

          • ③ 社会基本矛盾有不同的表现形式和解决方式,并从根本上影响和促进社会形态的变化和发展。

          • 因此,当两者不能够及时适应时,就需要改革来调整两者的关系,因而改革也是必不可少的。

        • (2)重要意义
          改革的成功,是对社会基本矛盾的某一方面或某种程度的解决,从而促进社会发展。社会生活的基础是物质生产,推动社会进步的最活跃、最革命的要素是生产力,社会主义的根本任务是解放和发展社会生产力。全面深化改革中,要把坚持发展作为解决我国所有问题的关键,推动我国社会生产力不断发展前进,进而推动人的全面发展和社会的全面进步。同时,还应通过改革,使生产关系适应生产力的发展、上层建筑适应经济基础的发展,从而顺利解决社会基本矛盾,促进社会的进一步发展。

        # 结合我国科学技术的重大成就,如高铁、大飞机以及 “天宫”“蛟龙”“天眼”“悟空”“墨子” 等,谈谈对科学技术在社会发展中的作用的认识。答:

        • (1)科学技术是先进生产力的重要标志,对于推动社会发展有非常重要的作用。科技革命是推动经济和社会发展的强大杠杆,近代以来,科技革命极大地推动了社会历史的进步,其主要是通过促进人们的生产方式、生活方式和思维方式的深刻变化来推动社会发展:

          • ① 对生产方式产生了深刻影响:

            • a.改变了社会生产力的构成要素。科技发展使生产自动化程度提高,大大地改变了体力劳动与脑力劳动的比例,使劳动力结构向着智能化趋势发展。

            • b.改变了人们的劳动形式。科技发展使人们的劳动方式经历了由机械自动化走向智能自动化、由局部自动化走向大系统管理和控制自动化的根本性变革。

            • c.改变了社会经济结构,特别是导致产业结构发生变革。科技的发展使第三产业在国民经济中所占的比重日益提高。

          • ② 对生活方式产生了巨大影响:

            • a.现代科技革命把人们带入了信息时代,要求人们不断更新和充实知识,以适应时代发展的需要。

            • b.现代信息技术为人们提供了处理、存储和传递信息的手段,给学习、工作带来了极大便利。

            • c.现代化的交通、通信等手段,为人们的交往提供了方便。

            • d.劳动生产率的提高,使人们自由支配的闲暇时间增多,为人的自由而全面的发展创造了更多条件。

          • ③促进了思维方式的变革。主要表现在新的科学理论和技术手段通过影响思维主体、思维客体和思维工具,引起了思维方式的变革正确把握科学技术的社会作用科学技术社会作用的两重性:

            • a.科学技术能够通过促进经济和社会发展造福于人类,科学技术的作用既受到一定客观条件的影响,也受到一定主观条件的影响。

            • b.科学技术的发展标志着人类改造自然能力的增强,意味着人们能够创造出更多的物质财富,对社会发展有巨大的推动作用。

        • (2)科学技术是社会发展的重要动力。近代中国落后于西方国家主要原因是科学技术水平低。习近平总书记提出 “科技兴则民族兴,科技强则国家强”。

        • (3)新时期我国的科学技术发展取得重大成就:2016 年 6 月 20 日,使用中国自主芯片制造的超级计算机 “神威 - 太湖之光” 登上全球超级计算机 500 强榜首。2016 年 8 月 16 日,长征二号运载火箭成功将世界首颗量子科学实验卫星 “墨子号” 发射升空。2016 年 8 月 25 日,我国研发的自主遥控水下机器人 “海斗” 号潜深达到 10767 米,首次进入万米时代。2016 年 9 月 15 日,“天宫二号” 顺利升空。2016 年 9 月 25 日,被誉为 “中国天眼” 的 FAST 射电望远镜落成启用。
          这些科学技术的发展不仅提高了我国的综合国力和国际地位,而且对我们的生产生活产生巨大影响。比如我国研制的全球超级计算机将对我们所进行的科学实验起到促进作用;“墨子号” 的发射升空将促进我国科研事业的发展,从而改善普通群众的生活。

        # 请结合人民群众是历史创造者的原理,谈谈对坚持以人民为中心重要性的认识。答:

        • (1)人民群众的含义
          人民群众是社会历史的主体,是历史的创造者,这是马克思主义最基本的观点之一。人民群众是一个历史范畴。从质上看,人民群众是指一切对社会历史发展起推动作用的人;从量上看,人民群众是指社会人口中的绝大多数。在不同的历史时期,人民群众有着不同的内容,包含着不同的阶级、阶层和集团,但其中最稳定的主体部分始终是从事物质资料生产的劳动群众。

        • (2)对坚持以人民为中心重要性的认识
          在实际生活中,我们必须坚持以人民为中心的思想,这是因为人民群众在创造历史过程中的决定作用。

          • ①人民群众是社会物质财富的创造者。人类社会赖以存在和发展的基础是物质资料的生产方式。人民群众创造的社会物质财富,是社会得以存在和发展的物质保障。人民群众的这一创造作用同生产力是社会发展的最终决定力量这一原理具有逻辑上的一致性,因为作为人民群众主体的劳动群众,乃是生产力的体现者。

          • ②人民群众是社会精神财富的创造者。首先,人民群众的社会实践活动是科学、文化、艺术的唯一源泉;其次,劳动群众为人们从事精神文化活动提供了一切物质手段和物质条件;再次,劳动知识分子在精神财富的创造过程中起着极其重要的作用。

          • ③人民群众是社会变革的决定力量。人民群众在创造社会财富的同时,也创造并改造着社会关系。人民群众既是社会革命的决定力量,又是社会改革的决定力量。社会变革、社会改革根源于社会基本矛盾,但生产关系一定要适合生产力发展状况的规律、上层建筑一定要适合经济基础发展状况的规律不可能自发地起作用,必须通过人民群众这一社会变革的主体才能实现其作用。

        # 习近平指出:“我们既要绿水青山,也要金山银山。宁要绿水青山,不要金山银山,而且绿水青山就是金山银山。” 请结合自然地理环境在社会生存和发展中的作用,谈谈应怎样认识和处理经济发展与环境保护的关系。答:

        • 自然地理环境是人类赖以存在和发展的必要前提,是社会存在的组成部分。没有自然界,没有感性的外部世界,工人什么也不能创造。没有地理环境,就不会有人和人类社会。可以看出,自然环境对人类的生存发展具有重要意义,因此,我们在发展经济的同时,不能忽略自然环境,不能以牺牲环境为代价来取得
          经济发展。我们必须坚持可持续的绿色发展理念,在发展经济的同时保护环境,“既要金山银山,也要绿水青山”。为此,我们必须做到以下四点:

        • (1)强化环境意识,树立生态理念要树立 “绿水青山就是金山银山” 的环境价值理念,实现以环境换取经济增长向以环境优化经济增长转变。绿水青山意味着优美的人居环境、清洁的水源和清新的空气,可以大大减少因环境污染和生态破坏造成的直接和间接损失,大大减轻因污染治理和生态恢复所需付出的巨大代价,大大缓解生态环境问题引发的各种社会矛盾,有利于维护社会稳定。

        • (2)把环境保护作为决策的重要环节,从源头落实环保基本国策
          环保从源头抓起,最重要的是各级政府、各级管理部门、各级领导要依法承担起改善环境质量和环境管理的责任,牢牢树立科学发展观念,转变把环境因素置于决策之外的决策模式,实行环境与发展综合决策。同时,必须树立正确的政绩观。

        • (3)把环境保护作为生产和消费过程中的重要环节,大力发展循环经济解决经济高速增长与生态环境日益恶化这一矛盾的根本出路是转变经济增长方式,用绿色核算体系来重新审视和把握经济发展的途径,走新型工业化道路,积极推动发展循环经济,实现经济与环境 “双赢”。

        • (4)把环境保护作为改善人居环境的重要环节,集中精力解决突出的环境问题
          改善人居环境不仅仅是改善市民的住房条件,要充分发挥环境保护的文明发展与协调稳定功能,以人为本,下决心解决老百姓广泛关注的影响环境质量和日常环境生活质量的突出问题。强化饮用水源的环保严管措施,保证饮用水安全。采取有效措施保护水源地,确保生态环境改善。大力改善区域环境,创建绿色环保文明社区,提高人民生活质量。

        • (5)正确认识和利用科学技术
          科学技术是把双刃剑,既能通过促进经济和社会发展造福于人类,也能在一定条件下对人类的生存和发展带来消极后果。

        # 第四章 资本主义的本质及规律

        # 为什么说 “资本来到世间,从头到脚,每个毛孔都滴着血和肮脏的东西”?答:

        • (1)资本主义的产生与发展
          资本主义于 14 世纪末 15 世纪初开始在地中海沿岸的一些城市出现,一是从小商品经济分化出来,二是从商人和高利贷者转化而来。资本主义生产关系产生之后,其成长是一个缓慢的过程。新兴资产阶级和资产阶级化的贵族用暴力剥夺小生产者从而加速形成资本主义生产方式。

        • (2)资本的原始积累
          资本原始积累,是生产者与生产资料相分离,货币资本迅速集中于少数人手中的历史过程。它主要是通过两个途径进行的:

          • ① 用暴力手段剥夺农民的土地,是资本原始积累过程的基础。资本家和封建贵族还通过 “掠夺教会地产,欺骗性地出让国有土地,盗窃公有地,用剥夺方法、用残暴的恐怖手段把封建财产和克兰财产转化为现代私有财产”,建立了资本主义的土地私有制,从而奠定了资本主义私有财产制度的基础。

          • ② 利用国家政权的力量进行残酷的殖民掠夺。新兴资产阶级在国外进行疯狂掠夺的同时,还通过国债制度、课税制度和保护关税制度等,加强对国内人民的剥削,在少数人手中积累起了大量的资本。这些极大地促进了资本主义的发展,缩短了封建生产方式转变为资本主义生产方式的历史过程。

        资本原始积累的事实表明,资产阶级的发家史就是一部罪恶的掠夺史。正如马克思所说:“资本来到世间,从头到脚,每个毛孔都滴着血和肮脏的东西。”

        # 如何理解商品二因素的矛盾来自劳动二重性的矛盾,归根结底来源于私人劳动和社会劳动之间的矛盾?答:

        • (1)商品经济存在着一系列的矛盾,其中私人劳动和社会劳动的矛盾是商品经济的基本矛盾,它决定着劳动二重性的矛盾和商品二因素的矛盾。
          商品经济存在的一系列的矛盾包括:

          • 商品的使用价值和价值的矛盾(即商品二因素的矛盾)

          • 具体劳动和抽象劳动的矛盾(即劳动二重性的矛盾)

          • 私人劳动和社会劳动的矛盾

        • (2)在私有制商品经济条件下,商品二因素的矛盾来源于生产商品的劳动二重性的矛盾。在商品交换过程中,生产商品使用价值的具体劳动不能直接进行量的比较,只有将具体劳动还原为抽象劳动,即从具体劳动中抽象出无差别的人类劳动,才可以进行量的比较。

        • (3)具体劳动能否还原为抽象劳动,取决于私人劳动和社会劳动能否实现一。一切商品,只有换成货币,商品生产者的私人劳动才会转化为社会劳动,具体劳动才会还原为抽象劳动,商品的使用价值和价值的矛盾才能得到解决。商品生产者的私人劳动生产的产品如果与社会的需求不相适应,该私人劳动就不被承认为社会劳动,其作为具体劳动的有用性质也就不被社会所承认,从而不能还原为抽象劳动,这意味着商品的价值不能实现,商品的使用价值和价值之间的矛盾没有解决。反过来,如果私人劳动生产的产品为社会所接受,则该私人劳动就被承认并转化为社会劳动,其作为具体劳动的有用性质就被社会所承认,从而可以还原为抽象劳动,这意味着商品的价值得到了实现,商品的使用价值和价值之间的矛盾得到了解决。

        # 如何理解 “资本是带来剩余价值的价值”?

        答:资本是带来剩余价值的价值,可以作以下理解:

        • (1)劳动力商品在使用价值上有一个很大的特点,其使用价值是价值的源泉,消费过程中能够创造新的价值,并且这个新的价值超过了劳动力本身的价值。由于该特点,货币所有者购买到劳动力以后,在消费过程中,不仅能够收回他在购买这种商品时支付的价值,还能得到增值的价值即剩余价值。

        • (2)资本是自行增殖的价值,是能够带来剩余价值的价值。在现实生活中,资本总是表现为一定的物,例如货币、机器设备、商品等,但这些物本身并不是资本,只有在一定的社会关系下,这些物被用来从事以获得剩余价值为目的的生产活动,也就是成为带来剩余价值的手段时,它才成为资本。所以马克思强调指出,资本的本质不是物,而是在物的外壳掩盖下的一种社会生产关系,即资本主义剥削关系。

        • (3)资本主义条件下,资本家剥削剩余价值具有隐蔽性。资本在运动过程中使用了例如货币、生产资料、商品等物质形态,资本家购买劳动力依照等价交换原则进行,于是在人们的观念上容易形成一种错觉,好像这些物天然就是资本,资本变成一种神秘的东西,似乎劳动的一切社会生产力并非劳动本身所有,而为资本所有,资本本身具有一种能使价值增殖的魔力,这就是资本拜物教。马克思指出:“资本家用货币购买的是雇佣工人的劳动力商品,劳动力商品的使用价值是劳动,劳动本身并不是商品,劳动力商品的使用价值具有成为价值源泉的特殊属性,它的实际使用本身就是劳动的物化,从而创造了剩余价值,使货币转化为资本。” 马克思的分析揭露了资本拜物教产生的实质,为科学认识资本主义经济制度的本质提供了理论武器。

        # 运用历史和现实的事实说明经济危机是资本主义基本矛盾的集中体现。答:

        • 资本主义发展到一定阶段,就会出现生产的相对过剩,这是经济危机的基本特征。经济危机发生时,大量商品积压,企业减产甚至停工,伴随着许多金融机构倒闭,整个社会经济生活一片混乱。对该论述理解如下:

        • (1)在资本主义经济的发展过程中,经济危机是周期性地重演的,危机与危机之间的间隔表现了一定的规律性。第二次世界大战后,各主要资本主义国家发生了次数不等的经济危机。在战后各国的历次危机中,有的是属于部分国家同期发生的,有的是普遍性的资本主义世界经济危机。属于世界性经济危机的有三次,即 1957—1958 年、1973—1975 年和 1980—1982 年的经济危机,因为这三次危机表现了明显的国际同期性。
          第二次世界大战结束以来,随着国家垄断资本主义的形成和发展,虽然市场机制依然在资源配置过程中发挥着基础性调节作用,但它已不是唯一的经济调节机制了。调节机制发生了一些明显的变化,其主要表现是资产阶级国家对经济的干预不断加强。国家已经承担起了提供财产保护、增强国家竞争力、实现经济增长和充分就业、保持经济稳定、提高社会福利水平以及维护竞争秩序等重要职能。它与市场机制相辅相成,共同推动着资本主义经济的运行和发展。

        • (2)生产过剩不是生产的商品绝对超过了人们的物质生活需要,而是一种相对过剩,即生产的商品相对于人民群众有支付能力的需求来说是过剩了。生产相对过剩是资本主义经济危机的实质。经济危机的抽象的一般的可能性,是由货币作为流通手段和支付手段引起的。以货币为媒介的商品买卖在时间上分为两个相互独立的行为。如果一些商品生产者在出卖自己的商品后不接着购买他人生产的商品,就会有另一些商品生产者的商品卖不出去。同时,在商品买卖有更多的部分采取赊购赊销的情况下,若某些债务人在债务到期时不能支付,就会使整个信用关系体系遭到破坏。但是,这仅仅是危机在形式上的可能性。

        • (3)资本主义经济危机是资本主义基本矛盾存在、激化和集中暴露的证明。这一基本矛盾具体表现在两个方面:①生产无限扩大的趋势与劳动人民现有支付能力和需求相对缩小的矛盾;②个别企业内部生产的有组织性和整个社会生产的无政府状态之间的矛盾。正如马克思所指出的:“一切现实的危机的最终原因始终是:群众贫穷和群众的消费受到限制,而与此相对立,资本主义生产却竭力发展生产力,好像只有社会的绝对的消费能力才是生产力发展的界限。”

        • (4)在经济调节机制变化的同时,经济危机形态也发生了变化,表现为:危机对社会经济运行的干扰减轻,破坏性减弱,生产下降的幅度减小,失业率有所降低,企业破产的数量减少;危机周期的长度缩短;经济危机的四个阶段(即危机阶段、萧条阶段、复苏阶段和高涨阶段)之间的差别有所减弱,各阶段的交替过程已不如过去那样明显;金融危机对整个经济危机的影响加强。随着调节机制的变化,资本主义经济危机的形态尽管发生了这样或那样的变化,但是由于资本主义的基本矛盾所决定,资产阶级不论采取什么样的调节措施和手段,经济危机是克服不了的。经济危机是资本主义基本矛盾的集中体现。

        # 有人认为,资本主义民主是囿于 “钱主” 的民主,迷于 “游戏” 的民主,止于 “选举” 的民主。你如何看待这种说法,为什么?

        • 答:资本主义民主存在着自身无法克服的 “软肋” 和 “硬伤”,这是因为资本主义政治制度为资产阶级专政服务,其本质上是资产阶级进行政治统治和社会管理的手段和方式,不可避免地有其阶级和历史的局限性。

        • (1)资本主义的民主是金钱操纵下的民主,是资产阶级精英统治下的民主。民主是具体的、相对的,而不是抽象的、绝对的。资本主义社会的主导逻辑是资本逻辑,不仅经济领域服从这一逻辑的统治,民主政治领域同样服从这一逻辑的主宰。西式民主的实质是 “钱主”,金钱是西式民主的 “母乳”,无论是政党还是政客都在母乳的滋养下成长,政客与财团之间结成生死与共的命运共同体,也常被金钱、媒体、黑势力、财团等影响和操纵。在实际的选举过程中,为得到选民的支持和认可,参选者往往需要支付巨额的竞选活动经费,这笔经费是普通公民根本没有能力支付的,只有依靠大财团才能支付得起。资本主义政治制度中的选举只能是有钱人的游戏,是资本玩弄民意的过程。号称民主典范的美国,其总统和参众两院议员的选举在现实中已经沦为烧钱的游戏,一人一票异化为 “一元一票”,筹集竞选经费的能力早已成为问鼎白宫的风向标,金钱是 “打开权力之门的金钥匙”。历史学家作过统计,从 1860 年以来的历次美国总统大选中,竞选经费占优的一方几乎都获得了胜利。“钱主” 的后果就是 “钱权联姻”。在西方,钱与权具有天然的近亲关系,“钱能生权,权又能生更多的钱”“政治献金” 与 “政治分赃” 总是如影随形。美国民主的实质就是 “1% 所有,1% 统治,1% 享用”。

        • (2)资本主义的民主是迷于 “游戏” 的民主。资本家和富人拥有特权,这对劳动者和穷人严重不平等,资产阶级法律将这种不平等合法化。民主的游戏化、娱乐化,这是西方的又一杰作。西方将民主变成了游戏,选民以娱乐的心态对待民主。西式民主在游戏中沉沦,选民在娱乐中迷茫。在这场游戏中,赢的永远是政客,输的始终是选民。

        • (3)资本主义的民主是止于 “选举” 的民主。民主不等于选举,民主和选举不能等量齐观、等同视之。真正的民主体制应该包括民主选举、民主决策、民主管理、民主监督等各个环节,覆盖起点、过程、结果等各个阶段。资本主义国家的多党制仍是资产阶级选择自己的国家管理者,以实现其内部利益平衡的政治机制。然而,现代资本主义民主却大都沉迷于 “选举” 这一初始环节,“人民主权” 被置换成 “人民的选举权”,民主被简化为选举,选举进一步简化为投票,而对于决策是否民主、管理是否民主、监督是否民主等方面却并不感兴趣。止于 “选
          举” 的民主只顾开头,不顾过程和结尾。这种民主即便有始却不一定有终,即便能善始却未必能善终。其实,民主是基于文化传统和现实国情的长期实践,采取何种民主形式必须与各个国家的历史文化传统、社会状况、人口结构、宗教信仰、民族构成、经济发展水平、法制意识、国民素质等因素相结合,否则很难有效运转。

        # 第五章 资本主义的发展及其趋势

        # 垄断是怎样产生的?为什么说垄断并没有消除竞争?答:

        • (1)垄断产生的原因

          • ① 当生产集中发展到相当高的程度,极少数企业就会联合起来,操纵和控制本部门的生产和销售,实行垄断,以获得高额利润。

          • ② 企业规模巨大,形成对竞争的限制,也会产生垄断。

          • ③ 激烈的竞争给竞争各方带来的损失越来越严重,为了避免两败俱伤,企业之间会达成妥协,联合起来,实行垄断垄断组织垄断是通过一定的垄断组织形式实现的。垄断组织是指在一个或几个经济部门中,占据垄断地位的大企业联合。垄断组织的形式多种多样,如初级的短期价格协定。但本质上都是通过联合达到独占和瓜分商品生产和销售市场,操纵垄断价格,以攫取高额垄断利润。

        • (2)垄断没有消除竞争的原因

          • ① 垄断没有消除产生竞争的经济根源。

          • ② 垄断必须通过竞争来维持。

          • ③ 社会生产是复杂多样的,任何垄断组织都不可能把包罗万象的社会生产都包下来。

        • 为什么说国家垄断资本主义体现了资本主义生产关系的部分质变?
          答:国家垄断资本主义是垄断资本和国家政权密切结合的垄断资本主义。国家垄断资本主义的产生,标志着资本主义发展进入了新的阶段。国家垄断资本主义体现了资本主义生产关系的部分质变的原因是:

          • (1)国家垄断资本主义是垄断资本主义的新发展,它对资本主义经济的发展产生了积极的作用。

            • ① 国家垄断资本主义的出现在一定程度上有利于社会生产力的发展,资本主义国家的政府通过公共投资和支出,开办大型新兴工业企业,部分克服了社会化大生产与私人垄断资本之间的矛盾。
            • ② 私人垄断统治扩大了资本主义生产与消费的矛盾,资产阶级国家采取一定手段干预国民经济,减轻经济危机的程度,缓和了社会各阶级的利益矛盾,有利于社会稳定,促进经济较为协调地发展。
            • ③ 宏观调控的发展,使国家对经济的干预和调节与市场机制日益有机结合起来,自由放任的市场经济逐渐演变成为由国家干预的现代市场经济,一定程度上弥补了自由市场经济的缺陷,对于国家的经济运行和社会发展产生了积极作用。
            • ④ 资产阶级国家通过税收和收入再分配手段,提高了劳动人民的生活水平。
          • (2)国家垄断资本主义的出现并没有根本改变垄断资本主义的性质。

            • ①国家垄断资本主义仍然是以资本主义私有制为基础,维护资产阶级利益和资本主义制度。
            • ②国家垄断资本主义没有从根本上消除资本主义的基本矛盾,它代表的是资产阶级的总利益。

        # 如何认识和把握经济全球化及其影响?

        • 答:经济全球化是指在生产不断发展、科技加速进步、社会分工和国际分工不断深化、生产的社会化和国际化程度不断提高的情况下,世界各国、各地区的经济活动越来越超出某一国家和地区的范围而相互联系、相互依赖的过程。

        • (1)经济全球化的表现

          • ① 国际分工进一步深化。

          • ② 贸易全球化。

          • ③ 金融全球化。

          • ④ 企业生产经营全球化。

        • (2)经济全球化的动因从本质上讲,经济全球化是生产力发展和社会化大生产的必然要求。导致经济全球化迅猛发展的因素主要有三点:

          • ① 科学技术的进步和生产力的发展提供了坚实的物质基础和根本的推动力。

          • ② 跨国公司的发展提供了适宜的企业组织形式。

          • ③ 各国经济体制的变革是经济全球化的体制保障。

        • (3)经济全球化的影响

          • ① 积极影响

            • a.社会分工得以在更大的范围内进行,资金、技术等生产要素可以在国际社会流动和优化配置,由此带来巨大的分工利益,推动世界生产力的发展。

            • b.体现了社会化生产的要求,不仅发达国家从中受益,一些发展中国家在参与经济全球化进程中也得到了快速发展。全球化为发展中国家提供先进技术和管理经验、就业机会,推动发展中国家国际贸易和跨国公司发展。

          • ② 消极影响

            • a.发达国家与发展中国家在经济全球化过程中的地位和收益不平等、不平衡。

            • b.加剧了发展中国家资源短缺和环境污染恶化。

            • c.一定程度上增加经济风险。

          • ③ 正确看待经济全球化

            • a.“把困扰世界的问题简单归咎于经济全球化,既不符合事实,也无助于问题解决。” 国际金融危机不是经济全球化发展的必然产物,而是金融资本过度逐利、金融监管严重缺失的结果。

            • b.经济全球化不是一部分国家的独角戏,而是世界各国、各民族共同实现发展的大舞台。

            • c.面对不同国家在生产方式、发展水平、文化背景等方面的差异,要以共同构建人类命运共同体的理念引领经济全球化。

        # 近年来资本主义社会发生了哪些新变化?试分析其原因及影响。答:

        • (1)资本主义社会的新变化

          • ① 生产资料所有制的变化:

            • a.国家资本所有制是指生产资料由国家占有并服务于垄断资本的所有制形式。国家资本所有制主要存在于基础设施和公共事业部门,所以对整个社会经济的发展有着重要的影响。但就其性质而言,仍然是资本主义形式,体现着总资本家剥削雇佣劳动者的关系。

            • b.法人资本所有制是资本主义生产资料所有制发展的新形式,是法人股东化的产物。法人资本所有制有两种形式:一种是企业法人资本所有制,另一种是机构法人资本所有制,但其在性质上是一种基于资本雇佣劳动的垄断资本。

          • ② 集体所有制劳资关系和分配关系的变化

            • a.职工参与决策。这一制度旨在协调劳资关系,缓和阶级矛盾。

            • b.终身雇佣。目的是增强工人对企业的归属意识,从而更加自觉地服从资本家的统治。

            • c.职工持股。旨在通过使职工持有一部分本公司的股份来调动工人的生产积极性,使工人产生归属感,在生产中努力提高劳动生产率,增加剩余价值生产。

            • d.普及化、全民化的社会福利制度。在一定程度上满足劳动者的安全和保障需求,保证劳动者维持最低生活水平,改善劳动者的社会状况。

          • ③ 社会阶层和阶级结构的变化

            • a.资本家的地位和作用已经发生很大变化。

            • b.高级职业经理成为大公司经营活动的实际控制者。

            • c.知识型和服务型劳动者的数量不断增加,劳动方式发生了新变化。

          • ④ 经济调节机制的变化 a.去工业化和产业空心化日趋严重,产业竞争力下降。

            • b.经济高度金融化,虚拟经济与实体经济严重脱节。

            • c.财政严重债务化,债务危机频繁爆发。

            • d.两极分化和社会对立加剧。

            • e.经济增长乏力,发展活力不足,周期性危机与结构性危机交织在一起。

            • f.金融危机频发,全球经济屡受打击。

          • ⑤ 政治制度的变化

            • a.政治制度出现多元化的趋势,公民权利有所扩大。

            • b.重视并加强法制建设。

            • c.改良主义政党在政治舞台上的影响日益扩大。

        • (2)资本主义社会的新变化的原因

          • ① 科学技术革命和生产力的发展,是当代资本主义发生新变化的根本推动力量。

          • ② 工人阶级争取自身权利和利益的斗争,是推动当代资本主义发生新变化的重要力量。

          • ③ 社会主义制度初步显示的优越性对当代资本主义产生了重要影响。

          • ④ 主张改良主义的政党对资本主义制度的改革,也对当代资本主义新变化发挥了重要作用。

        • (3)资本主义社会的新变化的影响

          • ① 当代资本主义发生的变化从根本上说是人类社会发展一般规律和资本主义经济规律作用的结果。

          • ② 当代资本主义发生的变化是在资本主义制度基本框架内的变化,并不意味着资本主义生产关系的根本性质发生了变化。

          • ③ 没有改变其为资产阶级利益服务,仍然是服从于资产阶级进行统治和压迫需要的政治工具的本质属性,没有改变马克思主义关于资本主义的基本论断的科学性。

        # 如何理解资本主义的历史地位及其为社会主义所代替的历史必然性?

        • 答:从人类社会发展的长河看,社会主义取代资本主义,即社会主义公有制取代资本主义私有制,这是历史发展的基本趋势。

        • (1)资本主义的内在矛盾决定了资本主义必然被社会主义所代替

          • ① 资本主义基本矛盾 “包含着现代的一切冲突的萌芽”。

            • a.资本主义基本矛盾表现在阶级关系上,是无产阶级和资产阶级的对立。

            • b.资本主义基本矛盾表现在经济关系上,是个别工厂中生产的组织性和整个社会中生产的无政府状态之间的对立。

            • c.资本主义经济危机的爆发正是这个基本矛盾发展的结果。

          • ② 资本积累推动资本主义基本矛盾不断激化并最终否定资本主义自身。

            • a.从资本主义积累过程来看,资本主义基本矛盾在资本积累过程中不断发展。

            • b.资本的不断积累必然提高生产的社会化程度,这在客观上势必导致生产的集中和资本的集中,使资本的社会化占有成为可能。

            • c.资本的不断积累使对生产过程的管理社会化,相应地派生出管理社会化大生产的管理人员和专业的管理机构,而这些都弱化甚至排斥资本家个人在管理中的地位和作用。

            当资本主义基本矛盾及其派生的各种矛盾在资本积累中不断发展、激化到资本主义制度自身无法使之释放时,社会主义取代资本主义就将成为不可避免的结果。这是资本主义积累过程所具有的客观历史趋势。

          • ③ 国家垄断资本主义中孕育着某些社会主义的因素,将成为社会主义的前奏。

          • ④ 资本主义社会中存在着资产阶级和无产阶级两大阶级之间的矛盾和斗争。无产阶级和资产阶级之间不可调和的矛盾、对立和斗争,必然导致资本主义政治危机。资产阶级的灭亡和无产阶级的胜利是不可避免的。通过无产阶级革命推翻资本主义制度,用社会主义公有制代替资本主义私有制,成为人类社会由低级向高级发展的客观规律的必然要求。

        • (2)社会主义代替资本主义是一个复杂的、长期的历史过程。其原因在于:

          • ① 社会主义自身的巩固、发展和完善是一个长期的历史过程。

            • a.经济上的强烈反差使社会主义国家面临严峻挑战。

            • b.西方敌对势力的颠覆、破坏与和平演变活动,加重了社会主义建设的曲折性、复杂性和长期性。

            • c.对社会主义建设规律的探索需要经过一个长期曲折的过程。

          • ② 社会主义在世界范围代替资本主义是一个长期的历史过程。

            • a.社会主义的本质决定了它在全世界取代资本主义的长期性、曲折性与艰巨性。

            • b.资本主义的暂时强大决定了它在全世界被社会主义所代替的长期性、曲折性与艰巨性。

            • c.20 世纪末世界社会主义运动遭受了重大挫折,导致社会主义顺利发展、再造辉煌需要较长的时间。

        # 第六章 社会主义的发展及其规律(本章仅供学习参考,不作考试要求)

        # 如何认识世界社会主义五百年的发展历程?

        • 答:世界社会主义五百年的发展过程实际上经历了四个阶段,分别是社会主义从空想到科学、社会主义从理论到实践、社会主义从一国到多国、在中国焕发强大生机活力。

        • (1)社会主义从空想到科学

          • ①空想社会主义

          空想社会主义产生于 16 世纪初期,到 19 世纪上半叶达到顶峰。这 300 多年正是欧洲从封建主义生产方式向资本主义生产方式转变的时期。其发展大致分为三个阶段:16 世纪和 17 世纪的早期空想社会主义,有理想社会制度的空想的描写,代表著作是英国人莫尔的《乌托邦》和意大利人康帕内拉的《太阳城》;18 世纪的空想平均共产主义;19 世纪初期出现了批判空想社会主义。主要代表是法国的圣西门、傅立叶和英国的欧文,以三大空想社会主义者为代表的空想社会主义学说,它对资本主义的揭露和批判,提供了启发工人觉悟的极为宝贵的材料。空想社会主义是早期无产阶级意识和利益的先声,反映了早期无产阶级迫切要求改造现存社会、建立理想的新社会的愿望。但其没有能够指出真正的出路。空想社会主义看不到无产阶级负有的历史使命,反对政治斗争,幻想依靠统治阶级的帮助,通过和平的方式来实现社会主义。

          • ② 科学社会主义

          19 世纪 40 年代,资本主义生产方式在西欧先进国家已占统治地位,随着资本主义的发展,资本主义内部矛盾日益尖锐,无产阶级反对资产阶级的斗争日益高涨。马克思和恩格斯参加了当时阶级斗争的实践。在此基础上,周密地研究了资本主义生产方式的矛盾,批判地继承了 19 世纪三大空想社会主义者法国的圣西门、傅立叶和英国的欧文的思想成果,创立了唯物史观和剩余价值论,为实现社会主义从空想到科学的飞跃奠定了坚实的理论基础。他们阐明了生产社会性和生产资料资本主义私人占有形式之间的矛盾的发展,必然导致社会主义取代资本主义,以生产资料的公有制取代生产资料私有制,科学地论述了资本主义必然灭亡、社会主义必然胜利的客观规律,从根本上超越了空想社会主义,实现了社会主义从空想到科学的伟大飞跃。

        • (2)社会主义从理论到现实

          • ① 第一国际与巴黎公社

          1848 年欧洲革命后,资本主义在各国飞速发展,无产阶级力量不断壮大。到 19 世纪 50 年代末 60 年代初,反压迫反剥削的斗争实践使各国无产阶级认识到,他们有着共同的利益和共同的敌人,而以往分散的斗争常常使他们遭到同样的失败,无产阶级必须在国际范围内联合起来。1864 年,国际工人协会(第一国际)应运而生。马克思是第一国际的真正领袖,在他指导下,第一国际把无产阶级先进分子团结在自己周围,率领各国工人群众向资产阶级和压迫者进行坚决斗争,促进了马克思主义的传播和与国际工人运动的结合,初步确立了马克思主义在工人运动中的指导地位。1871 年爆发的巴黎公社革命,深受第一国际精神的影响,是建立无产阶级专政的第一次伟大尝试。公社在经济方面采取了措施,对资本主义企业进行了一定程度的限制和剥夺;成立了管理经济工作的最高权力机构 —— 粮食委员会、财政委员会和劳动、就业及交换委员会;对工人群众的生活状况和劳动条件尽可能加以改善,提出了劳动者八小时工作制的原则。巴黎公社是世界上无产阶级武装暴力直接夺取城市政权的第一次尝试。它丰富和发展了马克思主义关于阶级斗争和社会主义的学说,在国际共产主义运动上写下了光辉、伟大而悲壮的一页。

          • ② 十月革命与第一个社会主义国家的建立

          1917 年 11 月 7 日(俄历 10 月 25 日)作为伟大的十月社会主义革命胜利日载入史册。俄国十月社会主义革命是人类历史上第一次获得胜利的社会主义革命,世界上第一个社会主义国家由此诞生。十月革命的胜利沉重打击了帝国主义的统治,推动了国际社会主义运动的发展,鼓舞了殖民地半殖民地人民的解放斗争。它改变了俄国历史的发展方向,用社会主义方式改造俄国的道路,对整个人类社会的发展都产生了巨大的影响。十月革命是二十世纪国际共产主义运动的序幕,触发了此后各国社会主义运动在全球范围的扩张,许多殖民地或半殖民地的解放运动也因此得到了更多支持,加速了世界范围内帝国主义殖民体系的整体瓦解,深刻改变了国际力量对比和世界格局。

        • (3)社会主义从一国到多国

          • ① 社会主义在苏联一国的实践

          苏维埃俄国是世界上第一个社会主义国家。年轻的苏维埃政权建立之后,既面临着巩固政权的任务,又面临着在经济文化相对落后的条件下建设社会主义的任务。列宁对苏维埃俄国社会主义的探索大体经历了巩固苏维埃政权时期、战时共产主义时期和新经济政策时期。

          • ② 社会主义发展到多个国家俄国十月革命的胜利后,第一个社会主义国家诞生,开辟了人类历史新纪元,极大地鼓舞了世界各国人民,促进了世界社会主义运动的发展。二战以后,社会主义在世界范围内获得大发展,在欧洲、亚洲、拉丁美洲,先后有 15 个国家建立社会主义制度,在国家数量上,占世界的 1/14,在领土上,占世界的 1/4,在人口上,占总人口的 1/3。社会主义国家相继开展了大规模的社会主义建设,推动了国家经济、政治、文化和社会各项事业的发展,世界形成了社会主义和资本主义两种制度共处的新格局。
        • (4)社会主义在中国焕发出强大生机活力
          俄国十月革命的胜利,给中国人民送来了马克思列宁主义。1921 年 7 月,中国共产党成立并成为中国社会主义运动的领导力量。中国共产党率领中国人民进行的社会主义事业经历了从新民主主义革命到社会主义革命、建设、改革的发展过程,中国共产党在近百年的奋斗中不断发展壮大,在 21 世纪焕发出强大的生机活力。习近平总书记在十九大报告中指出:“中国特色社会主义进入新时代,意味着近代以来久经磨难的中华民族迎来了从站起来、富起来到强起来的伟大飞跃,迎来了实现中华民族伟大复兴的光明前景;意味着科学社会主义在 21 世纪的中国焕发出强大生机活力,在世界上高高举起了中国特色社会主义伟大旗帜;意味着中国特色社会主义道路、理论、制度、文化不断发展,拓展了发展中国家走向现代化的途径,给世界上那些既希望加快发展又希望保持自身独立性的国家和民族提供了全新选择,为解决人类问题贡献了中国智慧和中国方案。” 总之,
          中国特色社会主义是科学社会主义 “中国化” 的伟大成果,是在当代中国的成功实践,充分表明了社会主义在世界上人口最多的国家成功开辟出了通向繁荣富强的正确道路,显著地展现了社会主义制度的优越性,标志着社会主义在世界范围内正在开拓新的历史征程。

        # 如何认识和把握科学社会主义一般原则?答:

        • (1)科学社会主义一般原则的主要内容
          科学社会主义一般原则是社会主义事业发展规律的集中体现,是马克思主义政党领导人民进行社会主义革命、建设、改革的基本遵循。

          • ① 人类社会发展规律和资本主义基本矛盾是 “资本主义必然灭亡、社会主义必然胜利” 的根本依据。

          • ② 无产阶级是最先进最革命的阶级,肩负着推翻资本主义旧世界、建立社会主义和共产主义新世界的历史使命。

          • ③ 无产阶级革命是无产阶级进行斗争的最高形式,以建立无产阶级专政的国家为目的。

          • ④ 社会主义社会要在生产资料公有制基础上组织生产,以满足全体社会成员的需要为生产的根本目的。

          • ⑤ 社会主义社会要对社会生产进行有计划的指导和调节,实行按劳分配原则。

          • ⑥ 社会主义社会要合乎自然规律地改造和利用自然,努力实现人与自然的和谐共生。

          • ⑦ 社会主义社会必须坚持科学的理论指导,大力发展社会主义先进文化。

          • ⑧ 无产阶级政党是无产阶级的先锋队,社会主义事业必须始终坚持无产阶级政党的领导。

          • ⑨ 社会主义社会要大力解放和发展生产力,逐步消灭剥削和消除两极分化,实现共同富裕和社会全面进步,并最终向共产主义社会过渡。

        • (2)正确对待科学社会主义一般原则的科学态度

          • ① 必须始终坚持社会主义一般原则,反对任何背离科学社会主义一般原则的错误倾向。

          • ② 要善于把科学社会主义一般原则与本国实际相结合,创造性地回答和解决革命、建设、改革中的重大问题。

          • ③ 紧跟时代和实践的发展,在不断总结新鲜经验中进一步丰富和发展科学社会主义一般原则。

        # 请结合科学社会主义一般原则,谈谈为什么说中国特色社会主义是科学社会主义,而不是其他什么主义。答:

        • (1)科学社会主义一般原则
          科学社会主义一般原则是社会主义事业发展规律的集中体现,是马克思主义政党领导人民进行社会主义革命、建设、改革的基本遵循。马克思、恩格斯在深刻揭示人类社会发展一般规律的基础上,深入阐发资本主义基本矛盾及其发展趋势,并在指导国际工人运动的过程中不断总结经验,逐步形成了科学社会主义一般原则。这些原则在后来的社会主义革命和建设中得到了证实、丰富和发展。这些原则具体如下:

          • ① 人类社会发展规律和资本主义基本矛盾是 “资本主义必然灭亡、社会主义必然胜利” 的根本依据。

          • ② 无产阶级是最先进最革命的阶级,肩负着推翻资本主义旧世界、建立社会主义和共产主义新世界的历史使命。

          • ③ 无产阶级革命是无产阶级进行斗争的最高形式,以建立无产阶级专政的国家为目的。

          • ④ 社会主义社会要在生产资料公有制基础上组织生产,以满足全体社会成员的需要为生产的根本目的。

          • ⑤ 社会主义社会要对社会生产进行有计划的指导和调节,实行按劳分配原则。

          • ⑥ 社会主义社会要合乎自然规律地改造和利用自然,努力实现人与自然的和谐共生。

          • ⑦ 社会主义社会必须坚持科学的理论指导,大力发展社会主义先进文化。

          • ⑧ 无产阶级政党是无产阶级的先锋队,社会主义事业必须始终坚持无产阶级政党的领导。

          • ⑨ 社会主义社会要大力解放和发展生产力,逐步消灭剥削和消除两极分化,实现共同富裕和社会全面进步,并最终向共产主义社会过渡。

        • (2)中国特色社会主义是科学的社会主义的原因
          中国特色社会主义,是科学社会主义理论逻辑和中国社会发展历史逻辑的辩证统一。中国特色社会主义既坚持了科学社会主义的基本原则,又根据时代条件赋予其鲜明的中国特色,是一百多年来科学社会主义理论与实践发展的结晶,是当代中国的科学社会主义。

          • ① 新中国成立后,中国共产党领导人民开始社会主义革命和建设的伟大实践。在只有苏联模式可学的情况下,在开始进行社会主义建设时我们走了照搬苏联经验的路子。但是,我们党很快就觉察到苏联模式的种种弊端,果断决定独立探索适合中国国情的社会主义建设道路。这一探索尽管曾出现过脱离实际的严重失误,却还是取得了巨大的成绩,形成了关于社会主义建设的许多独创性的理论成果和实践成果。在新的历史时期,我们党坚持解放思想、实事求是,深刻总结历史经验教训,正确判断时代主题和基本国情,经过 30 多年的努力,形成了中国特色社会主义道路、中国特色社会主义理论体系、中国特色社会主义制度三位一体的伟大成果,取得了中国特色社会主义建设的巨大成就。

          • ② 中国特色社会主义之所以是社会主义而不是别的什么主义,就在于它始终坚持以科学社会主义的基本原则为理论源泉和理论核心。科学社会主义创始人对社会主义的设想和他们提出的科学社会主义基本原则的最终实现,需要一个过程。我们党始终将其作为核心价值追求,始终将其作为进行改革开放和实践创新、理论创新、制度创新的 “魂”。这样一种追求,这样的 “魂”,鲜明地体现在改革开放新时期我们党所制定的路线、方针、政策中。

          • ③ 科学社会主义包含极其丰富的内容,集中到一点,就是科学地论证了社会主义必然代替资本主义的历史趋势,论证了无产阶级取代资本主义、建设社会主义和共产主义的伟大历史使命。科学社会主义理论创立以来的历史证明它的理论是正确的,社会主义具有强大的生命力。同时,它又不是一成不变的教条。它在指导实践的过程中,又要接受实践的检验,随着实践的发展而不断扩展、不断深化、不断丰富、不断完善。中国特色社会主义理论体系,就是科学社会主义在当代中国的新发展。

          • ④ 当前,资本主义现实和社会主义实践同马克思、恩格斯两位伟大思想家所处的历史环境相比,都发生了巨大的变化,但从世界社会主义发展的五百年大视野来看,目前依然处在马克思主义所指明的历史时代,处在资本主义向社会主义转变的历史进程之中,科学社会主义从没有过时,仍然有强大的生命力,必须始终坚持、不动摇。正如习近平指出:“中国特色社会主义是社会主义而不是其他什么主义,科学社会主义基本原则不能丢,丢了就不是社会主义。”

        # 请谈谈中国特色社会主义的成功实践对世界社会主义发展的贡献。答:

        • 党的十九大报告作出了 “中国特色社会主义进入新时代” 的重大政治论断。习近平总书记在报告中指出:“中国特色社会主义进入新时代,意味着近代以来久经磨难的中华民族迎来了从站起来、富起来到强起来的伟大飞跃,迎来了实现中华民族伟大复兴的光明前景;意味着科学社会主义在二十一世纪的中国焕发出强大生机活力,在世界上高高举起了中国特色社会主义伟大旗帜;意味着中国特色社会主义道路、理论、制度、文化不断发展,拓展了发展中国家走向现代化的途径,给世界上那些既希望加快发展又希望保持自身独立性的国家和民族提供了全新选择,为解决人类问题贡献了中国智慧和中国方案。” 这一重大政治论断给我们许多启示,增加了我们对中国特色社会主义的自信,也使我们充分认识到中国特色社会主义不仅在 21 世纪的中国充分显现出生机与活力,取得伟大成功,使科学社会主义理论在中国实践中获得巨大发展,也使得中国特色社会主义超越了国界,具有世界性的意义,为发展中国家走向现代化提供了新的路径选择,也为世界发展和人类问题的解决贡献了中国智慧和中国方案。

        • (1)中国特色社会主义为世界贡献了中国智慧
          中国走上中国特色社会主义道路,是几代中国共产党人带领全国人民在艰难曲折中艰辛探索、不断实践创新的结果,体现了中国共产党人的不屈不挠,也体现了中国共产党人的卓越智慧,其中最核心的就是党的实事求是的思想路线,一切从实际出发,把马克思主义基本原理与中国客观实际相结合,走适合中国国情的社会主义道路。改革开放以来,我们党坚持实事求是的思想路线,在实践中不断探索和回答了什么是社会主义、怎样建设社会主义,建设什么样的党、怎样建设党,实现什么样的发展、怎样发展,并继续探索和回答新时代坚持和发展什么样的中国特色社会主义、怎样坚持和发展中国特色社会主义这些基本问题。党的思想路线体现的是马克思主义的世界观、方法论。因此,中国特色社会主义的成功,是马克思主义世界观和方法论的成功,是马克思主义中国化、时代化的成功。中国特色社会主义贡献给世界的智慧,最为根本的是实事求是,国家建设一定要从自己的国情出发,探索适合自己国情的建设道路,任何脱离实际、不符合国情的道路、方法、模式都是不可能取得成功的。

        • (2)中国特色社会主义为世界贡献了中国方案
          中国特色社会主义不仅为世界贡献了指导国家建设的正确的思想方法,还为世界贡献了国家建设的具体方案和治国理政的基本方略。中国特色社会主义治国方案既包括总目标、总任务、总体布局、战略布局和发展方向、发展方式、发展动力、战略步骤、外部条件、政治保证等基本方面的总体设计,也包括经济、政治、科技、文化、教育、民生、民族、宗教、社会、生态、国家安全、国防和军队、国家统一、统一战线、外交、党的建设等各方面的基本规定。还有就是,坚持党对一切工作的领导,坚持以人民为中心,坚持全面深化改革,坚持新发展理念,坚持人民当家作主,坚持全面依法治国,坚持社会主义核心价值体系,坚持在发展中保障和改善民生,坚持人与自然和谐共生,坚持总体国家安全观,坚持党对人民军队的绝对领导,坚持 “一国两制” 和推进祖国统一,坚持推动构建人类命运共同体,坚持全面从严治党等基本方略。这些既是新时代坚持和发展中国特色社会主义的基本方略,也是中国特色社会主义为解决人类问题向世界贡献的中国方案,给世界上那些既希望加快发展又希望保持自身独立性的国家和民族提供了全新选择。

        # 如何正确认识和把握共产党执政规律?

        答:可以从以下几个方面正确认识和把握共产党执政规律:

        • (1)共产党执政规律,最根本的是人心向背定理。习近平总书记指出:“一个政党,一个政权,其前途和命运最终取决于人心向背。”“人心是最大的政治。” 要始终把人民放在心中最高位置。坚持一切为了人民、一切依靠人民。

        • (2)共产党执政首要的是坚持党中央权威和集中统一领导,党是最高政治领导力量,党的领导是全面领导。党的十九大报告指明 “坚持党中央权威和集中统一领导,是党的政治建设的首要任务”,揭示了保证党的团结和集中统一的根本性问题,阐明了进行伟大斗争、建设伟大工程、推进伟大事业、实现伟大梦想的根本保证,意义重大而深远。

        • (3)共产党在执政过程中,要把党的政治建设摆在首位。要坚持党要管党、全面从严治党,以应对各种考验,坚定不移地惩治和预防腐败,勇于自我革命。只有全面从严治党,才能顺利实现中华民族伟大复兴的中国梦。

        • (4)要始终抓住科学执政、民主执政、依法执政这个要律,为实现国家治理现代化指明方向路径。

        # 第七章 共产主义是人类最崇高的社会理想

        # 请谈谈在未来理想社会的认识上,马克思主义经典作家与空想社会主义者有何本质区别。

        • 答:在未来理想社会的认识上,马克思主义经典作家与空想社会主义者的本质区别是:是否坚持科学的立场、观点和方法来预见未来。预见未来社会的方法论原则主要包括:

        • (1)在揭示人类社会发展一般规律的基础上指明社会发展的方向。

        • (2)在剖析资本主义旧世界的过程中阐发未来新世界的特点。

        • (3)在社会主义社会发展中不断深化对未来共产主义社会的认识。

        • (4)立足于揭示未来社会的一般特征,而不作详尽的细节描绘。

        # 既然共产主义理想的实现是历史的必然,为什么又要人们去努力追求?既然共产主义的最终实现是一个漫长的过程,为什么又说 “共产主义渺茫论” 是错误的?请用马克思主义的辩证观点予以解答。答:

        • (1)实现共产主义的历史必然

          • ① 共产主义理想的实现是历史规律的必然要求

            • a.共产主义理想是能够实现的理想,它与一切空想和幻想有着本质区别。

            • b.共产主义理想一定会实现,是以人类社会发展规律以及资本主义社会的基本矛盾发展为依据的。

            • c.社会主义运动的实践,特别是社会主义国家的兴起和不断发展,已经且正在用事实证明共产主义理想实现的必然性。从一定意义上讲,社会主义革命的胜利本身就是对共产主义理想可以实现的证明。

          • ② 实现共产主义是人类最伟大的事业

            • a.实现共产主义理想是广大人民群众的共同愿望。

            • b.实现共产主义,必须找到现实的阶级力量,这就是现代工人阶级或无产阶级。

            • c.无产阶级的解放与全人类的解放是完全一致的。

        • (2)共产主义的实现是一个漫长的过程

          • ① 资本主义的灭亡和向社会主义的转变是一个长期的过程

            • a.资本主义作为一个社会形态,其走向灭亡是一个长期的历史过程。资本主义国家经历过严重的经济和社会危机,在与社会主义国家的竞争中也伴随着巨大的压力。但资本主义并没有退出历史舞台,而是仍然有一定发展空间。这些危机背后隐藏着资本主义衰败的必然趋势,但目前谁也无法准确判断资本主义何时走到尽头。

            • b.社会主义革命取得胜利后,从资本主义转变为社会主义也需要一段时期。这个重要的过渡期是不以人的意志为转移的,它不能省略,更不可随意缩短。

            • c.在完成转变以后,仍然需要经历一段社会主义的发展阶段,才能最终逐渐走向共产主义。在任何国家,实现共产主义都不能超越社会主义发展阶段。不管资本主义国家发达到何种程度,当发生根本性的制度变革时,也不可能直接达到共产主义社会,只能首先进入社会主义社会,也就是共产主义社会的低级阶段。

            • d 要真正迎来新社会,消除旧社会痕迹,也需要经过一个很长的时期。

          • ② 社会主义社会自身巩固、发展和完善并最终成为共产主义也是一个长期的过程。

            • a.想要在全世界范围内实现共产主义,必须取决于社会主义国家的巩固和发展,取决于这些国家所经历的社会主义社会的时间长短。因此,共产主义只有在社会主义社会充分发展和高度发达的基础上才能实现。

            • b.经济上的强烈反差使社会主义国家面临严峻挑战。社会主义革命没有像马克思和恩格斯所设想的那样,在发达的资本主义国家首先发生,而是在少数经济和文化比较落后的国家首先取得胜利。这些国家基础差、起点低,要在经济上赶上资本主义发达国家,提高社会化生产水平,建设和完善社会主义政治、经济、文化等各项制度,体现出社会主义优于资本主义,需要付出更大、更艰苦和更长期的努力,是一个漫长的历史过程。

            • c.对社会主义建设规律的探索需要经过一个长期曲折的过程。建设社会主义是前无古人的崭新事业,没有现成固定的模式可循,需要各个社会主义国家在实践中把马克思主义的基本原理与本国的具体实践相结合,探索适合本国发展的社会主义建设道路和模式。而探索的过程不会是一帆风顺的,难免会出现不同程度的失误,甚至发生曲折和暂时倒退现象。这种探索过程的曲折性,无疑会增加这些国家建设、巩固和发展社会主义的艰巨性与长期性。

            • d.社会主义是共产主义的低级阶段,也是实现共产主义的必由之路。作为同一社会形态发展进程中的两个阶段,社会主义是共产主义的必要准备,共产主义是社会主义发展的必然趋势。随着社会主义生产力和其他因素的不断发展,属于社会主义的一些因素将逐步消失,而共产主义的所有因素将逐步增长。社会主义社会不断发展和完善的过程,就是共产主义因素不断扩大的过程。一旦条件完全成熟,社会主义社会就自然过渡到共产主义社会。

        • (3)实现共产主义是历史发展规律的必然要求,“共产主义渺茫论” 是错误的

          • ① 共产主义渺茫论者不了解共产主义的含义。

          共产主义有两方面的含义:一方面指的是在共产主义思想体系指导下的共产主义运动;另一方面指的是共产主义的社会制度。当然,共产主义社会制度彻底实现,是需要经过若干代人的不懈努力的,是一个远大的奋斗目标。但是,作为在共产主义思想体系指导下的共产主义运动,作为一种亿万人民在共产党领导下改造社会的革命实践,则是一点也不 “渺茫”,而是实实在在地、每日每时地存在于我们的现实生活中。一个多世纪以来,各国无产阶级及其政党为了实现共产主义制度这一伟大理想,进行了艰苦卓绝的斗争,诸如巴黎公社、十月社会主义革命以及我国的社会主义革命和社会主义建设,每向前发展一步都意味着向共产主义靠拢一步。今天,我们的一切革命工作都是伟大的共产主义运动的具体体现和组成部分。

          • ② 共产主义渺茫论者割裂了共产主义与社会主义之间的联系。

          共产主义渺茫论者忽视了一个很重要的事实,即共产主义运动不仅在全世界,而且在中国也早已存在,而且已经取得了显著成果,这就是共产主义社会的过渡阶段 —— 社会主义社会,已经在中国和世界一些国家建立起来并取得重大发展,从而使共产主义运动发生了质的飞跃。我国社会主义制度的建立,不仅使人民从被压迫和被剥削的旧制度下解放出来,而且,随着社会主义改革和建设的不断深入,人们生活的不断改善,社会主义制度的不断完善以及社会道德水准和文化教育科学事业的向前推进,共产主义理想逐渐深入人心。

          • ③共产主义渺茫论者不能正确对待社会主义事业中的曲折和困难。

          马克思主义辩证法告诉我们,事物的发展不是直线上升,而是在曲折中前进的。社会主义事业也是如此,它有成功,也有失误,有前进,也有暂时的困难。中国的改革和经济建设取得了举世瞩目的伟大成就,但也存在不少问题。只有坚持中国特色社会主义道路,集中精力把经济建设搞上去,人民生活得到了不断改善,人们才能看到社会主义的前途和希望。只要把社会主义建设好,才能为将来的共产主义打下了坚实的基础。我们要继续坚定不移地沿着社会主义道路勇往直前,坚信共产主义必胜的信念。

        # 有人说:“在社会主义初级阶段只能讲树立中国特色社会主义共同理想,而不应提树立共产主义远大理想,否则就是脱离实际。” 请用共同理想和远大理想的关系来分析评论这一观点。

        • 答:共产主义理想是建立在科学基础上的社会理想,是人类最伟大的社会理想。在坚持和发展中国特色社会主义的实践中,我们不但要坚定中国特色社会主义共同理想,而且要进一步树立共产主义远大理想。

        • (1)正确认识和把握共产主义远大理想与中国特色社会主义共同理想的关系

          • ①从时间上看,远大理想与共同理想的关系是最终理想与阶段性理想的关系。

          • ②从层次上看,远大理想与共同理想的关系是最高纲领与最低纲领的关系。实现共产主义远大理想要坚持最高纲领和最低纲领的统一。

          • ③从范围来看,远大理想与共同理想的关系也是全人类理想与全体中国人民理想的关系。

        • (2)以辩证思维把握和处理远大理想和共同理想的关系
          必须以马克思主义的辩证思维和历史思维去把握远大理想和共同理想的关系。实现中国特色社会主义的共同理想,是为了实现共产主义远大理想而服务。中国特色社会主义共同理想在我国的成功实践,必然为实现共产主义远大理想奠定坚实的基础。

        # 中国特色社会主义进入了新时代。为了不辜负这个伟大的时代,当代大学生应该怎样确立和追求自己的人生理想?答:

        • (1)新时代的特征
          中国特色社会主义进入新时代。我们党对 “什么是社会主义、如何建设社会主义” 的认识又向前推进了一步,从理论和实践结合上系统回答了 “新时代坚持和发展什么样的中国特色社会主义、怎样坚持和发展中国特色社会主义” 这一重大历史课题,习近平新时代中国特色社会主义思想成为指引全国人民沿着中国特色社会主义道路继续前进的重要理论基础。

        • (2)对当代大学生的要求习近平指出:“青年是标志时代的最灵敏的晴雨表,时代的责任赋予青年,时代的光荣属于青年。” 青年是祖国的未来、民族的希望。青年兴则国兴,青年强则国强。当前,中国特色社会主义进入新时代,为当代青年特别是当代大学生提供了实现人生才华的极为有利的历史机遇。

          • ①要坚定理想信念。心中有信仰,脚下才有力量。为此,就要深入学习马克思主义基本原理及马克思主义中国化的理论成果,特别是学习习近平新时代中国特色社会主义思想。

          • ②要积极投身实现中华民族伟大复兴的中国梦,勇做担当中华民族伟大复兴大任的时代新人。

          • ③要投身新时代中国特色社会主义事业,当代大学生是祖国的未来、民族的希望,要以勇于担当的精神,做走在新时代前列的奋进者、开拓者、奉献者,投身于党和人民在中国特色社会主义新时代的伟大奋斗。

        \ No newline at end of file +马原课后习题 - 马克思主义基本原理 - 思政 | Balloon Party = 風船のパーティー = fuusen no party

        # 思维导图

        # 绪论部分

        绪论部分

        # 第一章 世界的物质性及发展规律

        第一章 世界的物质性及发展规律

        # 第二章 实践与认识及其发展规律

        第二章 实践与认识及其发展规律

        # 第三章 人类社会及其发展规律

        第三章 人类社会及其发展规律

        # 第四章 资本主义的本质及规律

        第四章 资本主义的本质及规律

        # 第五章 资本主义的发展及其趋势

        第五章 资本主义的发展及其趋势

        # 第六章社会主义的发展及其规律(本章仅供学习参考,不作考试要求)

        第六章 社会主义的发展及其规律(本章仅供学习参考,不作考试要求)

        # 第七章 共产主义是人类最崇高的社会理想

        第七章 共产主义是人类最崇高的社会理想

        # 大纲

        # 绪论部分

        # 什么是马克思主义?

        • a) 从创造者、继承者的认识成果讲,马克思主义是由马克思、恩格斯创立的,由其后继者不断丰富和发展的观点和学说的科学理论体系。

        • b) 从阶级属性讲,马克思主义是关于无产阶级解放、全人类解放和每个人自由而全面发展的学说,是指引人民创造美好生活的行动指南,具有鲜明的阶级性。

        • c) 从研究对象和主要内容讲,马克思主义是无产阶级的科学世界观和方法论,是关于自然、社会和人类思维发展的一般规律的学说,是关于社会主义必然代替资本主义、最终实现共产主义的学说。

        • d) 马克思主义哲学、马克思主义政治经济学和科学社会主义的有机统一并共同构成了马克思主义理论的主体内容。

        # 马克思主义的直接的理论来源?

        • a) 资本主义生产方式造成社会化大生产迅猛发展的同时也带来了严重的社会灾难(社会两极分化;周期性经济危机频发)

        • b) 无产阶级在反抗资产阶级剥削和压迫的斗争中,逐步走向自觉,并迫切渴望科学的理论指导

        • c) 19C 西欧三大先进思潮为马克思主义的创立提供直接的理论来源(德国古典哲学、英国古典政治经济学和英法两国的空想社会主义)

        • d) 19C 三大科学发现也为马克思主义的产生提供了自然科学前提(细胞学说、能量守恒和转化定律)

        • e) 还有主观条件,结合马恩二人的生平谈就行

        # 马克思主义的鲜明特征?

        • a) 科学性

          • a.马克思主义是对自然、社会和人类思维发展本质和规律的正确反映。它是在社会实践和科学发展的基础上产生的,并在自身发展过程中不断总结实践经验,吸取自然科学和社会科学发展的最新成就。

          • b.马克思主义具有科学的世界观和方法论基础,即辩证唯物主义和历史唯物主义,这是马克思主义的一个突出特征和理论优势,也是马克思主义科学性的重要体现。

          • c.马克思主义理论是一个逻辑严密的有机整体,它以事实为依据、以规律为对象,并以实践为检验标准。马克思主义的发展具有科学探索性,是一个不断探索和掌握客观规律的过程。

        • b) 人民性

          • a.人民至上是马克思主义的政治立场。马克思主义政党把人民放在心中最高位置,一切奋斗都致力于实现最广大人民的根本利益,这是马克思主义最鲜明的政治立场。

          • b.人民群众是历史的创造者,是社会主义事业的依靠力量,马克思主义的人民性是以阶级性为深刻基础的,是无产阶级先进性的体现。

          • c.马克思主义是关于无产阶级解放的学说。无产阶级解放和全人类解放是完全一致的。

        • c) 实践性

          • a.马克思主义具有突出的实践精神,它始终强调理论与实践的统一,始终坚持与社会主义实际运动紧密结合。马克思主义是从实践中来,到实践中去,在实践中接受检验,并随实践的发展而不断发展的学说。

          • b.从马克思主义的使命和作用来说,它不是一种纯粹解释世界的学说,而是直接服务于无产阶级和人民群众改造世界的实践活动的科学理论。

          • c.从马克思主义的内容来看,实践观点是马克思主义首要的和基本的观点,这一基本观点体现在马克思主义全部思想内容之中。

        • d) 发展性

          • a.马克思主义是不断发展的学说,具有与时俱进的理论品质。马克思主义是时代的产物,并随着时代、实践和科学的发展而不断发展。

          • b.马克思主义理论体系是开放的,它不断吸取人类最新的文明成果来充实和发展自己。

          • c.马克思主义在指导人们认识世界和改造世界的过程中,在指导社会主义事业发展的过程中,不断与时代特征和各国具体实际相结合,得到丰富和发展,并形成新的理论成果。

        # 结合当代世界所面临的课题和当代青年所肩负的使命,谈谈马克思主义的当代价值和指导意义?

        • (1)当代世界所面临的新课题和当代青年所肩负的新使命

          • a) 在经济全球化的背景下,资本主义内部矛盾越来越突出,经济政治发展的不平衡加剧了资本主义世界的两极分化,贫富差距日益扩大。20 世纪 90 年代末的东南亚金融危机以及由华盛顿开始波及全球的 2008 年全球金融危机,都暴露了资本主义的缺陷。当代世界的新课题是和平与发展,各国都希望在一个相对稳定的环境中谋求国内经济发展,提高人们的生活水平。因此,马克思主义思想对现代社会仍有指导意义。

          • b) 不同的时代有不同的历史责任与使命,当代青年是在中国改革开放新时期成长起来的,处在一个继往开来的历史时期,肩负着祖国人民更多的期望,也应肩负起民族复兴的重任。当代青年所处的时代正是社会主义现代化建设的关键时刻,中华民族的伟大复兴对当代青年提出了新的挑战,我们有责任也有义务来维护祖国的统一,保卫祖国,建设祖国。而大学生作为当代青年的主体和中坚力量,更应该发挥引领带头作用。

          • c) 总而言之,马克思主义作为具有科学性、真理性的学说,对当代青年的人生发展有重要的指引作用,有助于青年们树立正确的世界观、人生观和价值观,提高其应对和处理问题的能力。

        • (2)马克思主义的当代价值

          • ① 当代世界变化的认识工具

            • a.马克思主义给予我们观察当代世界的宏大视野。

            • b.马克思主义给予我们透视时代风云的锐利目光。

            • c.马克思主义给予我们展望未来世界的长远眼光和战略定力指引。

          • ② 当代中国发展的行动指南

            • a.马克思主义是指引当代中国发展的精神旗帜。

            • b.马克思主义是推动当代中国发展的精神动力。

            • c.马克思主义是引领当代中国实践的行动指南。

          • ③ 引领人类社会进步的科学真理

            • a.从人类历史发展的大视野来看,世界仍然处于马克思主义所指明的从资本主义走向社会主义的大时代。马克思主义所揭示的资本主义基本矛盾仍然存在,而且在近年来西方的金融危机和社会危机中呈现出某种激化的趋势。

            • b.人类的未来仍然需要马克思主义的启迪和指引。社会是在矛盾中进步的,每一个时代的社会进步总是伴随着相应的社会问题,面对当今社会的重大问题,还是需要从马克思主义中寻找智慧。

        • (3)对当代青年的指导意义

          • 青年是祖国的未来、事业的希望。青年强则国家强,青年一代有理想、有本领、有担当,国家就有前途,民族就有希望。学习马克思主义理论可以帮助青年提高分析和解决问题的能力,有助于其树立正确的世界观、人生观和价值观。当代青年在学习马克思主义的过程中,要有正确的态度和科学的方法:

            • ① 努力学习和掌握马克思主义的基本立场、观点、方法。
            • ② 坚持理论联系实际的马克思主义学风。
            • ③ 自觉将马克思主义内化于心、外化于行。

        # 阅读感受:

        • 《资本论》内容宏大,理论深奥,创造了一个崭新的思想体系。其研究世界的方法源于德国古典哲学、英法两国的空想社会主义和英国古典政治经济学。马克思吸收了黑格尔的辩证法思想,相信能够用一个辩证法公式概括人类的进化历程。他认为,所有哲学家所做的一切都在致力于解释世界,但他同时认为,问题的关键在于如何改造世界。马克思在黑格尔辩证法的基础之上,颠覆了传统的 “形而上学”,建立了一个现实中得以实践的最大的思想体系。《资本论》以唯物史观的基本思想为指导,通过深刻分析资本主义生产方式,揭示了资本主义社会发展的规律,同时也使唯物史观得到了科学的验证和进一步的丰富和发展。它运用唯物史观的观点和方法,将社会关系归结为生产关系,将生产关系归结于生产力水平,从而证明了社会形态的发展是一个不以人的意志为转移的自然历史过程。

        # 第一章 世界的物质性及发展规律

        # 马克思主义的物质范畴的理论意义

        • (1)马克思主义的物质观
          马克思主义认为,世界是物质的世界,物质是第一性,物质是不依赖于人类的意识而存在,并能为人类的意识所反映的客观存在。“物质” 范畴是唯物主义哲学关于世界本原和统一性的最高抽象,是唯物主义世界观的基石。马克思主义批判继承了以往唯物主义的传统,在总结现代自然科学重大成就的基础上,创立了辩证唯物主义的物质观。20 世纪初,列宁对物质概念做出了科学的概括:“物质是标志客观实在的哲学范畴,这种客观实在是人通过感觉感知的,它不依赖于我们的感觉而存在,为我们的感觉所复写、摄影、反映。”

        • (2)马克思主义的物质观具有深刻的理论意义

          • ① 坚持唯物主义一元论,同唯心主义一元论和二元论划清了界限。马克思主义指明了物质对于意识的独立性、根源性,以及意识对于物质的依赖性、派生性。因为意识不过是物质的反映,不能离开物质而独立存在,所以意识不可能成为世界的另一种本原。

          • ② 坚持了能动的反映论和可知论,批判了不可知论。目前世界上还有很多事物未被人类认识,但这并不意味着它们不可认识。人们已经认识到未知世界与已知世界都是客观存在的。世界上只有尚未认识之物,没有不可认识之物。

          • ③ 体现了唯物论和辩证法的统一,克服了形而上学唯物主义的缺陷。马克思主义的物质观认为,物质的唯一特性是客观实在性,既肯定了哲学物质范畴同自然科学物质结构理论的联系,又将它们相区别。从个性中看到共性,从暂时中发现永恒,从相对中找到绝对,这是马克思主义物质观体现的唯物辩证法。

          • ④ 体现了唯物主义自然观与唯物主义历史观的统一,为彻底的唯物主义奠定了理论基础。马克思主义的物质观建立了统一说明自然历史过程的唯物主义原则,揭示了自然和社会的物质性,实现了唯物主义自然观和历史观的辩证统一。

        # 物质与意识的辩证关系

        物质决定意识,意识对物质具有反作用,这种反作用就是意识的能动作用。

        • (1)物质决定意识
          意识是人脑的机能和属性,是客观世界的主观映像。物质对意识的决定作用表现在意识的起源、本质和作用上。

          • ① 从意识的起源来看,其形成和发展经历了三个阶段,即:由一切物质所具有的反应特性到低等生物的刺激感应性,再到高等动物的感觉和心理,最终发展成人类的意识。意识不仅是自然界长期发展的产物,而且是社会历史发展的产物。劳动为意识的产生和发展提供了客观需要和可能;人们在劳动和交往中形成的语言促进了意识的发展。

          • ② 从意识的本质来看,意识是客观世界的主观映像。马克思指出:“观念的东西不外是移入人的头脑并在人的头脑中改造过的物质的东西而已。”

        • (2)意识对物质具有反作用
          物质决定意识,意识的能动作用是指意识对物质具有反作用。意识的能动作用主要表现在:

          • ① 意识活动具有目的性和计划性。马克思认为:“人在劳动过程结束时得到的结果,在这个过程开始时就已经在劳动者的表象中存在着,即已经观念地存在着”。人的整个实践过程,就是围绕意识活动所构建的目标和蓝图来进行的。

          • ② 意识活动具有创造性。人的意识可以对客观世界中的感性材料加工和建构,在思维中构造一个现实中没有的理想世界。

          • ③ 意识具有指导实践改造客观世界的作用。在实践中形成观念,以这些观念为指导,通过实践使之变成客观现实。④ 意识具有调控人的行为和生理活动的作用。现代科学证明,意识、心理因素对人的行为选择和健康状况能够产生重要影响。

        • (3)主观能动性和客观规律性的统一
          正确认识和把握物质与意识的辩证关系,需要处理好主观能动性与客观规律性的关系,原因在于:

          • ① 尊重客观规律是正确发挥主观能动性的前提。人们正确地认识世界,有效地改造世界的前提是认识和掌握客观规律。

          • ② 只有充分发挥主观能动性,才能正确认识和利用客观规律。承认规律的客观性,并不是说人在规律面前无能为力、无所作为。人类能够按照客观规律改造世界,以达到自己的目的。

        # 运用矛盾的普遍性和特殊性辩证关系原理,说明把马克思主义基本真理与中国具体实际相结合的重要性。答:

        • (1)矛盾普遍性与特殊性的辩证关系

          • ① 矛盾的普遍性即矛盾的共性,矛盾的特殊性即矛盾的个性。矛盾的共性是无条件的、绝对的;矛盾的个性是有条件的、相对的。

          • ② 任何现实存在的事物的矛盾都是共性和个性的有机统一,共性寓于个性之中,没有离开个性的共性,也没有离开共性的个性。矛盾的共性和个性相统一的关系,既是客观事物固有的辩证法,也是科学的认识方法。

        • (2)把马克思主义基本真理与中国具体实际相结合的重要性
          矛盾的普遍性和特殊性辩证关系的原理,是把马克思主义普遍真理同各国具体实际相结合的理论基础,也是建设中国特色社会主义的理论根据。

          • ① 矛盾的普遍性和特殊性辩证统一的原理,是把马克思主义普遍真理与中国的具体实际相结合,走建设有中国特色社会主义道路的重要理论依据。矛盾的普遍性和特殊性的关系就是共性与个性、绝对与相对的关系。我国走中国特色社会主义道路,是中国共产党把马克思主义普遍真理同中国的具体实际相结合的过程中实现的第二次飞跃。在这里,“中国特色” 是中国不同于其他社会主义国家的个性、特殊性。“社会主义” 则是中国和其他社会主义国家的共性、普遍性。

          • ② 根据共性与个性相统一的原理,我们坚持中国特色社会主义道路。

            • a.不管中国的情况如何特殊,都必须坚持马克思主义的普遍真理,坚持社会主义根本制度和基本原则,这是共性。

            • b.社会主义的一般只能在各国特色的个别中存在,我们又必须建设中国特色的社会主义,从中国的国情出发,注意把握中国的特点。只有坚持党的基本路线不动摇,走自己的路,我们才能在建设中国特色社会主义的道路上不断取得胜利,这是个性。

        # 不断增强思维能力:辩证思维能力、历史思维能力、战略思维能力、底线思维能力、创新思维能力。

        答:创新思维能力是对常规思维的突破,是破除迷信、超越陈规,善于因时制宜、知难而进、开拓创新的能力。

        • (1)培养和提高创新思维能力的重要性

          • ① 坚持创新思维,才能推动发展、取得进步。“发展的观点” 是唯物辩证法的基本观点,马克思主义关于人类社会不断进步的发展观,不仅揭示了人类社会历史是不断向前发展的,而且说明了正是人类的创新思维和创造智慧推动了社会更好地发展。

          • ② 创新是一个民族进步的灵魂,是一个国家兴旺发达的不竭动力,也是中华民族最深沉的民族禀赋。这些年中华民族进步和贡献的取得,正是中华民族勇于创新和善于创新的结果。失去了创新思维,就会失去前进的动力,导致封闭和落后。

          改革开放以来,中国共产党高度重视思想解放和理论创新,开辟了中国特色社会主义道路,形成了中国特色社会主义理论体系,确立了中国特色社会主义制度,发展了中国特色社会主义文化,实现了中国人民从站起来到富起来、强起来的伟大飞跃。从某种意义上说,这些成就正是我们党坚持创新思维、大力推进理论创新和实践创新的结果。

        • (2)提高自己的创新思维能力的做法

          • ① 提高创新思维能力,必须坚决反对各种形而上学的思维方式,做到不唯书、不唯上,只唯实。提高创新思维能力,需要坚持和运用唯物辩证法,坚决反对形而上学。

            • a.要破除对 “本本” 的迷信;

            • b.要破除对 “经验” 的迷信;

            • c.要破除对 “权威” 的迷信。

          • ② 问题意识是提高创新思维能力的起点和动力源。

          增强问题意识、坚持问题导向,就是承认矛盾的普遍性客观性,就是善于把认识和化解矛盾作为打开工作局面的突破口,善于具体问题具体分析,透过现象看本质,抓主要矛盾和矛盾的主要方面,科学分析问题,深入研究问题。

          • ③ 善于继承才能善于创新。

            • a.创新思维不是凭空产生的,善于继承,才能将过去、现在和未来联系起来;

            • b.善于创新,才会有现在超越过去、未来超越现在的前进运动。

            • c.中国特色社会主义是科学社会主义理论逻辑和中国社会发展历史逻辑的辩证统一,是中国共产党人运用创新思维,在继承和发展中形成的创新成果。

          • ④ 在解放思想中锻造创新思维。

            • a.提高创新思维能力,就是要有敢为人先的锐气,打破惯性思维,以思想认识的新飞跃打开工作的新局面;

            • b.提高创新思维的过程,就是思想解放的过程。没有思想的大解放,就不可能有创新思维的大突破。

        # 第二章 实践与认识及其发展规律

        # 实践与认识的辩证关系及其规律

        • (1)毛泽东《实践论》的主要观点:
          毛泽东强调:“一个正确的认识,往往需要经过由物质到精神,由精神到物质,即由实践到认识,由认识到实践这样多次的反复,才能够完成。” 如此 “实践、认识、再实践、再认识,循环往复以至无穷,而实践和认识之每一循环的内容,都比较地进到了高一级的程度”。这就是认识辩证运动发展的基本过程,也是认识运动的总规律。认识是一个反复循环和无限发展的过程,是一个波浪式前进和螺旋式上升的过程。认识在实践基础上不断深化发展,并推动实践在认识指导下不断深入推进。这个过程既不是封闭式的循环,也不是直线式的发展,而是波浪式前进和螺旋式上升的。

        • (2)实践与认识的辩证关系

          • ① 实践是认识的基础,实践在认识活动中起决定性作用。主要表现在以下四个方面:

            • a.实践是认识的来源。认识的内容是在实践活动的基础上产生和发展的。只有在实践中实际改造和变革对象,人们才能形成对认识对象属性、本质、规律更为准确的认识,并以这种认识指导实践活动。

            • b.实践是认识发展的动力。认识的产生和发展源于实践的需要,也正是这种需要推动人类的科学发现和技术发明,推动人类的思想进步和理论创新。实践的需要是推动认识在深度和广度上不断发展之根本。此外,实践是认识发展的动力,还表现为实践为认识的发展提供了如经验资料、实验仪器和工具等的手段和条件。另外,实践改造了人的主观世界,锻炼和提高了人的认识能力。人们正是在实践的推动下,不断突破旧认识、旧思想,引起认识上的新飞跃,从而不断有新发现、有所前进。

            • c.实践是认识的目的。通过实践获得认识,其最终目的是为实践服务,指导实践,以满足生活和生产的需要。

            • d.实践是检验认识真理性的唯一标准。真理不是自封的。判断认识的真理性,不是依主观感觉,而是依客观上社会实践的结果。

          • ② 认识反作用于实践。

            • a.人的实践活动是受人的意识支配的,实践的这种本质特性决定着它不能离开认识的指导。

            • b.认识活动及其成果有相对独立性,遵循其特有的活动方式和发展规律,它一经形成,便能反作用于实践。

            • c.认识对实践的指导作用表现在:正确的认识能够推动实践的发展,而错误的认识则会对实践起阻碍作用。

        # 真理、真理的绝对性和相对性及其辩证关系、真理与谬误(自学);

        • (1)真理是绝对的又是相对的原因
          真理是一个过程。就真理的发展过程以及人们对它的认识和掌握程度来说,真理既具有绝对性,又具有相对性,它们是同一客观真理的两
          种属性,这是真理问题上的辩证法。任何真理都是绝对性和相对性的统一,二者相互联系、不可分割。两者关系具体如下:

          • ①真理的绝对性

            真理的绝对性是指真理主客观统一的确定性和发展的无限性。它有两个方面的含义:

            • a.任何真理都标志着主观与客观之间的符合,都包含着不依赖于人和人的意识的客观内容,都同谬误有原则的界限。这一点是绝对的、无条件的。在这个意义上,承认了真理的客观性也就是承认了真理的绝对性。

            • b.人类认识按其本性来说,能够正确认识无限发展着的物质世界,认识每前进一步,都是对无限发展着的物质世界的接近,这一点也是绝对的、无条件的。在这个意义上,承认了世界的可知性,承认人能够获得关于无限发展着的物质世界的正确认识,也就是承认了真理的绝对性。

          • ②真理的相对性

            真理的相对性是指人们在一定条件下对客观事物及其本质和发展规律的正确认识总是有限度的、不完善的。它具有两个方面的含义:

            • a.从客观世界的整体来看,任何真理都只是对客观世界的某一阶段、某一部分的正确认识,人类已经达到的认识的广度总是有限度的,因而,认识有待扩展。

            • b.就特定事物而言,任何真理都只是对客观对象一定方面、一定层次和一定程度的正确认识,认识反映事物的深度是有限度的,或是近似性的。因而,认识有待深化。任何真理都只能是主观对客观事物近似正确即相对正确的反映。

          • ③真理的绝对性和相对性的关系

            真理的绝对性和相对性是辩证统一的,具体如下:

            • a.二者相互依存。即人们对于客观事物及其本质和规律的每一个正确认识,都是在一定范围内、一定程度上、一定条件下的认识,因而是相对的和有局限的;但是,在这一定范围内、一定程度上、一定条件下,它也是对客观对象的正确反映,因而它也是无条件的、绝对的。

            • b.二者相互包含。一是真理的绝对性寓于真理的相对性之中。任何真理所包含的客观内容都只能是人们在特定历史条件下所把握到的,都只是对客观世界及其事物的一定范围、一定程度的正确反映。二是真理的相对性包含并表现着真理的绝对性。任何真理都与谬误有本质的区别,标志着人们在一定范围内和一定层次上达到对于无限发展着的物质世界的正确认识,包含着确定的客观内容。毛泽东把真理的绝对性与相对性的关系比喻为长河与水滴的关系,“马克思主义者承认,在绝对的总的宇宙发展过程中,各个具体过程的发展都是相对的,因而在绝对真理的长河中,人们对于在各个一定发展阶段上的具体过程的认识只具有相对的真理性。无数相对的真理之总和,就是绝对的真理”。

            • c.相对性向绝对性转化。真理永远处在由相对向绝对的转化和发展中,是从真理的相对性走向绝对性、接近绝对性的过程。任何真理性的认识都是由真理的相对性向绝对性转化过程中的一个环节,这是真理发展的规律。真理的绝对性与相对性根源于人认识世界的能力的无限性与有限性、绝对性与相对性的矛盾。人类的思维按其本性、可能和历史的终极目的来说,是能够认识无限发展着的物质世界的,思维是无限的和绝对的。但是,具体到每一个人乃至每一代人,由于受到客观事物及其本质的显露程度、社会历史的实践水平、主观的条件以及生命的有限性等各方面的限制,其思维也是有限的和相对的。人的认识能力、思维能力是无限性与有限性、绝对性与相对性的对立统一,作为人的正确认识成果的真理,也必然是绝对性和相对性的对立统一。

        • (2)真理的绝对性和相对性相统一对马克思主义的重要意义

          • ① 割裂真理绝对性与相对性之间的辩证关系,就会走向形而上学的真理观,即绝对主义和相对主义。绝对主义在实际工作中表现为教条主义、思想僵化,把马克思主义当作一成不变的公式。相对主义否认真理内容的客观性,把真理的相对性歪曲成了主观随意性,由此陷入主观真理论,导致不可知论和诡辩论。

          • ② 马克思主义是客观真理,是绝对性和相对性的统一

            马克思主义正确地反映了人类社会发展的规律,是经过长期革命、建设和改革的实践所证明了的真理,因而具有绝对性的一面。但是,马克思主义并没有穷尽对一切事物及其规律的认识,仍然需要随着社会实践的发展而发展,因而又具有相对性的一面。正因为马克思主义具有绝对性,所以我们必须坚持以马克思主义为指导思想;也因为它具有相对性,所以我们必须在实践中丰富和发展马克思主义。既坚持又发展,才是对待马克思主义的正确态度。

        # 真理和价值的辩证关系。

        • (1)真理尺度和价值尺度的含义

          • ① 人们的实践活动总是受着真理尺度和价值尺度的制约。

          • ② 实践的真理尺度是指在实践中人们必须遵循正确反映客观事物本质和规律的真理。只有按照真理办事,才能在实践中取得成功。

          • ③ 实践的价值尺度是指在实践中人们都是按照自己的尺度和需要去认识世界和改造世界。这一尺度体现了人的活动的目的性。

        • (2)真理尺度与价值尺度的关系

          • ① 任何实践活动都要受到这两种尺度的共同制约,任何活动都是合规律性和合目的性的统一。

          • ② 真理与价值或真理尺度与价值尺度之间是紧密联系、不可分割的。一方面,价值尺度必须以真理为前提。脱离了真理尺度,价值尺度就偏离了正确的轨道。另一方面,人类满足自身需要的内在尺度,推动新的真理不断被发现。脱离了价值尺度,真理就缺失了主体意义。

        • (3)帮助和启示
          因为实践具有具体性和历史性,真理尺度与价值尺度的统一也是具体的和历史的,二者的统一会随着实践的发展而不断发展到更高级的程度,真理由相对走向绝对,人的需要和利益也日益呈现出多元化特征。真理尺度与价值尺度是否达到了具体的、历史的统一,必须通过实践来验证。新时代中国特色社会主义的实践充分体现了真理尺度与价值尺度的辩证统一。正如习近平所指出的:“中国共产党人的理想信念,建立在马克思主义科学真理的基础之上,建立在马克思主义揭示的人类社会发展规律的基础之上,建立在为最广大人民谋利益的崇高价值的基础之上。我们坚定,是因为我们追求的是真理。我们坚定,是因为我们遵循的是规律。我们坚定,是因为我们代表的是最广大人民的根本利益。”

        # 第三章 人类社会及其发展规律

        # 为什么说物质资料的生产方式是社会发展的决定性力量?

        答:物质资料的生产方式是社会发展的决定性力量,主要是基于以下原因:

        • (1)物质资料的生产方式是人类其他一切活动的首要前提,是人类社会赖以存在和发展的物质基础。人们为了能够创造历史,必须能够生活,为了生活,首先就需要衣、食、住及其他东西。因此,人类历史的第一个历史活动就是生产满足这些需要的物质资料本身。这是一个简单的事实和起码的真理。发现并承认这一真理,是历史观的一个伟大革命。

        • (2)物质资料的生产方式决定着社会的结构、性质和面貌。有什么样的生产方式便有什么样的社会形态。正如马克思所说:“手推磨产生的是封建主为首的社会,蒸汽磨产生的是工业资本家为首的社会。”

        • (3)生产方式的发展变化决定着社会历史的发展变化和形态更替。在物质资料的生产方式中,生产力是最活跃、最能动的因素,它总是要向前发展的,而生产关系则是相对稳定的。随着生产力的发展,原有的生产关系便由生产力发展的形式变成生产力发展的桎梏,由适合生产力的发展变成阻碍生产力的发展。只有变革生产关系,才能解放和发展生产力。随着生产关系(即经济基础)的变革,全部的、庞大的上层建筑也会或迟或早地发展变革,从而引起社会形态的更替。

        • 总之,人类社会的历史就是物质资料生产的历史,是劳动发展史。马克思正是从劳动发展史中找到了理解人类社会发展史的 “钥匙”。

        # 运用社会基本矛盾运动的原理分析深化改革的客观依据与重要意义。答:

        • 与深化改革最相近的是生产力与生产关系的矛盾运动。

          • 社会基本矛盾运动包括

            • 生产力与生产关系之间的矛盾

            • 经济基础与上层建筑之间的矛盾

          • 它们是贯穿社会发展过程始终,规定社会发展过程的基本性质和基本趋势,并对社会历史发展起根本推动作用的矛盾。

          • 因此,社会基本矛盾的解决对历史发展具有重要作用。

          • 而要解决社会基本矛盾,改革是一种重要的方式和途径。

        • (1)客观依据
          社会基本矛盾,特别是生产力与生产关系的矛盾在历史发展中起根本的推动作用,为了推动社会发展,必须解决好社会基本矛盾,改革就是其中一种重要方式。深化改革的客观依据就在于社会基本矛盾在历史发展过程中的重要作用,主要表现在:

          • ① 生产力是社会基本矛盾运动中最基本的动力因素,是人类社会发展和进步的最终决定力量。

          • ② 社会基本矛盾特别是生产力和生产关系的矛盾,是 “一切历史冲突的根源”,决定着社会中其他矛盾的存在和发展。

          • ③ 社会基本矛盾有不同的表现形式和解决方式,并从根本上影响和促进社会形态的变化和发展。

          • 因此,当两者不能够及时适应时,就需要改革来调整两者的关系,因而改革也是必不可少的。

        • (2)重要意义
          改革的成功,是对社会基本矛盾的某一方面或某种程度的解决,从而促进社会发展。社会生活的基础是物质生产,推动社会进步的最活跃、最革命的要素是生产力,社会主义的根本任务是解放和发展社会生产力。全面深化改革中,要把坚持发展作为解决我国所有问题的关键,推动我国社会生产力不断发展前进,进而推动人的全面发展和社会的全面进步。同时,还应通过改革,使生产关系适应生产力的发展、上层建筑适应经济基础的发展,从而顺利解决社会基本矛盾,促进社会的进一步发展。

        # 结合我国科学技术的重大成就,如高铁、大飞机以及 “天宫”“蛟龙”“天眼”“悟空”“墨子” 等,谈谈对科学技术在社会发展中的作用的认识。答:

        • (1)科学技术是先进生产力的重要标志,对于推动社会发展有非常重要的作用。科技革命是推动经济和社会发展的强大杠杆,近代以来,科技革命极大地推动了社会历史的进步,其主要是通过促进人们的生产方式、生活方式和思维方式的深刻变化来推动社会发展:

          • ① 对生产方式产生了深刻影响:

            • a.改变了社会生产力的构成要素。科技发展使生产自动化程度提高,大大地改变了体力劳动与脑力劳动的比例,使劳动力结构向着智能化趋势发展。

            • b.改变了人们的劳动形式。科技发展使人们的劳动方式经历了由机械自动化走向智能自动化、由局部自动化走向大系统管理和控制自动化的根本性变革。

            • c.改变了社会经济结构,特别是导致产业结构发生变革。科技的发展使第三产业在国民经济中所占的比重日益提高。

          • ② 对生活方式产生了巨大影响:

            • a.现代科技革命把人们带入了信息时代,要求人们不断更新和充实知识,以适应时代发展的需要。

            • b.现代信息技术为人们提供了处理、存储和传递信息的手段,给学习、工作带来了极大便利。

            • c.现代化的交通、通信等手段,为人们的交往提供了方便。

            • d.劳动生产率的提高,使人们自由支配的闲暇时间增多,为人的自由而全面的发展创造了更多条件。

          • ③促进了思维方式的变革。主要表现在新的科学理论和技术手段通过影响思维主体、思维客体和思维工具,引起了思维方式的变革正确把握科学技术的社会作用科学技术社会作用的两重性:

            • a.科学技术能够通过促进经济和社会发展造福于人类,科学技术的作用既受到一定客观条件的影响,也受到一定主观条件的影响。

            • b.科学技术的发展标志着人类改造自然能力的增强,意味着人们能够创造出更多的物质财富,对社会发展有巨大的推动作用。

        • (2)科学技术是社会发展的重要动力。近代中国落后于西方国家主要原因是科学技术水平低。习近平总书记提出 “科技兴则民族兴,科技强则国家强”。

        • (3)新时期我国的科学技术发展取得重大成就:2016 年 6 月 20 日,使用中国自主芯片制造的超级计算机 “神威 - 太湖之光” 登上全球超级计算机 500 强榜首。2016 年 8 月 16 日,长征二号运载火箭成功将世界首颗量子科学实验卫星 “墨子号” 发射升空。2016 年 8 月 25 日,我国研发的自主遥控水下机器人 “海斗” 号潜深达到 10767 米,首次进入万米时代。2016 年 9 月 15 日,“天宫二号” 顺利升空。2016 年 9 月 25 日,被誉为 “中国天眼” 的 FAST 射电望远镜落成启用。
          这些科学技术的发展不仅提高了我国的综合国力和国际地位,而且对我们的生产生活产生巨大影响。比如我国研制的全球超级计算机将对我们所进行的科学实验起到促进作用;“墨子号” 的发射升空将促进我国科研事业的发展,从而改善普通群众的生活。

        # 请结合人民群众是历史创造者的原理,谈谈对坚持以人民为中心重要性的认识。答:

        • (1)人民群众的含义
          人民群众是社会历史的主体,是历史的创造者,这是马克思主义最基本的观点之一。人民群众是一个历史范畴。从质上看,人民群众是指一切对社会历史发展起推动作用的人;从量上看,人民群众是指社会人口中的绝大多数。在不同的历史时期,人民群众有着不同的内容,包含着不同的阶级、阶层和集团,但其中最稳定的主体部分始终是从事物质资料生产的劳动群众。

        • (2)对坚持以人民为中心重要性的认识
          在实际生活中,我们必须坚持以人民为中心的思想,这是因为人民群众在创造历史过程中的决定作用。

          • ①人民群众是社会物质财富的创造者。人类社会赖以存在和发展的基础是物质资料的生产方式。人民群众创造的社会物质财富,是社会得以存在和发展的物质保障。人民群众的这一创造作用同生产力是社会发展的最终决定力量这一原理具有逻辑上的一致性,因为作为人民群众主体的劳动群众,乃是生产力的体现者。

          • ②人民群众是社会精神财富的创造者。首先,人民群众的社会实践活动是科学、文化、艺术的唯一源泉;其次,劳动群众为人们从事精神文化活动提供了一切物质手段和物质条件;再次,劳动知识分子在精神财富的创造过程中起着极其重要的作用。

          • ③人民群众是社会变革的决定力量。人民群众在创造社会财富的同时,也创造并改造着社会关系。人民群众既是社会革命的决定力量,又是社会改革的决定力量。社会变革、社会改革根源于社会基本矛盾,但生产关系一定要适合生产力发展状况的规律、上层建筑一定要适合经济基础发展状况的规律不可能自发地起作用,必须通过人民群众这一社会变革的主体才能实现其作用。

        # 习近平指出:“我们既要绿水青山,也要金山银山。宁要绿水青山,不要金山银山,而且绿水青山就是金山银山。” 请结合自然地理环境在社会生存和发展中的作用,谈谈应怎样认识和处理经济发展与环境保护的关系。答:

        • 自然地理环境是人类赖以存在和发展的必要前提,是社会存在的组成部分。没有自然界,没有感性的外部世界,工人什么也不能创造。没有地理环境,就不会有人和人类社会。可以看出,自然环境对人类的生存发展具有重要意义,因此,我们在发展经济的同时,不能忽略自然环境,不能以牺牲环境为代价来取得
          经济发展。我们必须坚持可持续的绿色发展理念,在发展经济的同时保护环境,“既要金山银山,也要绿水青山”。为此,我们必须做到以下四点:

        • (1)强化环境意识,树立生态理念要树立 “绿水青山就是金山银山” 的环境价值理念,实现以环境换取经济增长向以环境优化经济增长转变。绿水青山意味着优美的人居环境、清洁的水源和清新的空气,可以大大减少因环境污染和生态破坏造成的直接和间接损失,大大减轻因污染治理和生态恢复所需付出的巨大代价,大大缓解生态环境问题引发的各种社会矛盾,有利于维护社会稳定。

        • (2)把环境保护作为决策的重要环节,从源头落实环保基本国策
          环保从源头抓起,最重要的是各级政府、各级管理部门、各级领导要依法承担起改善环境质量和环境管理的责任,牢牢树立科学发展观念,转变把环境因素置于决策之外的决策模式,实行环境与发展综合决策。同时,必须树立正确的政绩观。

        • (3)把环境保护作为生产和消费过程中的重要环节,大力发展循环经济解决经济高速增长与生态环境日益恶化这一矛盾的根本出路是转变经济增长方式,用绿色核算体系来重新审视和把握经济发展的途径,走新型工业化道路,积极推动发展循环经济,实现经济与环境 “双赢”。

        • (4)把环境保护作为改善人居环境的重要环节,集中精力解决突出的环境问题
          改善人居环境不仅仅是改善市民的住房条件,要充分发挥环境保护的文明发展与协调稳定功能,以人为本,下决心解决老百姓广泛关注的影响环境质量和日常环境生活质量的突出问题。强化饮用水源的环保严管措施,保证饮用水安全。采取有效措施保护水源地,确保生态环境改善。大力改善区域环境,创建绿色环保文明社区,提高人民生活质量。

        • (5)正确认识和利用科学技术
          科学技术是把双刃剑,既能通过促进经济和社会发展造福于人类,也能在一定条件下对人类的生存和发展带来消极后果。

        # 第四章 资本主义的本质及规律

        # 为什么说 “资本来到世间,从头到脚,每个毛孔都滴着血和肮脏的东西”?答:

        • (1)资本主义的产生与发展
          资本主义于 14 世纪末 15 世纪初开始在地中海沿岸的一些城市出现,一是从小商品经济分化出来,二是从商人和高利贷者转化而来。资本主义生产关系产生之后,其成长是一个缓慢的过程。新兴资产阶级和资产阶级化的贵族用暴力剥夺小生产者从而加速形成资本主义生产方式。

        • (2)资本的原始积累
          资本原始积累,是生产者与生产资料相分离,货币资本迅速集中于少数人手中的历史过程。它主要是通过两个途径进行的:

          • ① 用暴力手段剥夺农民的土地,是资本原始积累过程的基础。资本家和封建贵族还通过 “掠夺教会地产,欺骗性地出让国有土地,盗窃公有地,用剥夺方法、用残暴的恐怖手段把封建财产和克兰财产转化为现代私有财产”,建立了资本主义的土地私有制,从而奠定了资本主义私有财产制度的基础。

          • ② 利用国家政权的力量进行残酷的殖民掠夺。新兴资产阶级在国外进行疯狂掠夺的同时,还通过国债制度、课税制度和保护关税制度等,加强对国内人民的剥削,在少数人手中积累起了大量的资本。这些极大地促进了资本主义的发展,缩短了封建生产方式转变为资本主义生产方式的历史过程。

        资本原始积累的事实表明,资产阶级的发家史就是一部罪恶的掠夺史。正如马克思所说:“资本来到世间,从头到脚,每个毛孔都滴着血和肮脏的东西。”

        # 如何理解商品二因素的矛盾来自劳动二重性的矛盾,归根结底来源于私人劳动和社会劳动之间的矛盾?答:

        • (1)商品经济存在着一系列的矛盾,其中私人劳动和社会劳动的矛盾是商品经济的基本矛盾,它决定着劳动二重性的矛盾和商品二因素的矛盾。
          商品经济存在的一系列的矛盾包括:

          • 商品的使用价值和价值的矛盾(即商品二因素的矛盾)

          • 具体劳动和抽象劳动的矛盾(即劳动二重性的矛盾)

          • 私人劳动和社会劳动的矛盾

        • (2)在私有制商品经济条件下,商品二因素的矛盾来源于生产商品的劳动二重性的矛盾。在商品交换过程中,生产商品使用价值的具体劳动不能直接进行量的比较,只有将具体劳动还原为抽象劳动,即从具体劳动中抽象出无差别的人类劳动,才可以进行量的比较。

        • (3)具体劳动能否还原为抽象劳动,取决于私人劳动和社会劳动能否实现一。一切商品,只有换成货币,商品生产者的私人劳动才会转化为社会劳动,具体劳动才会还原为抽象劳动,商品的使用价值和价值的矛盾才能得到解决。商品生产者的私人劳动生产的产品如果与社会的需求不相适应,该私人劳动就不被承认为社会劳动,其作为具体劳动的有用性质也就不被社会所承认,从而不能还原为抽象劳动,这意味着商品的价值不能实现,商品的使用价值和价值之间的矛盾没有解决。反过来,如果私人劳动生产的产品为社会所接受,则该私人劳动就被承认并转化为社会劳动,其作为具体劳动的有用性质就被社会所承认,从而可以还原为抽象劳动,这意味着商品的价值得到了实现,商品的使用价值和价值之间的矛盾得到了解决。

        # 如何理解 “资本是带来剩余价值的价值”?

        答:资本是带来剩余价值的价值,可以作以下理解:

        • (1)劳动力商品在使用价值上有一个很大的特点,其使用价值是价值的源泉,消费过程中能够创造新的价值,并且这个新的价值超过了劳动力本身的价值。由于该特点,货币所有者购买到劳动力以后,在消费过程中,不仅能够收回他在购买这种商品时支付的价值,还能得到增值的价值即剩余价值。

        • (2)资本是自行增殖的价值,是能够带来剩余价值的价值。在现实生活中,资本总是表现为一定的物,例如货币、机器设备、商品等,但这些物本身并不是资本,只有在一定的社会关系下,这些物被用来从事以获得剩余价值为目的的生产活动,也就是成为带来剩余价值的手段时,它才成为资本。所以马克思强调指出,资本的本质不是物,而是在物的外壳掩盖下的一种社会生产关系,即资本主义剥削关系。

        • (3)资本主义条件下,资本家剥削剩余价值具有隐蔽性。资本在运动过程中使用了例如货币、生产资料、商品等物质形态,资本家购买劳动力依照等价交换原则进行,于是在人们的观念上容易形成一种错觉,好像这些物天然就是资本,资本变成一种神秘的东西,似乎劳动的一切社会生产力并非劳动本身所有,而为资本所有,资本本身具有一种能使价值增殖的魔力,这就是资本拜物教。马克思指出:“资本家用货币购买的是雇佣工人的劳动力商品,劳动力商品的使用价值是劳动,劳动本身并不是商品,劳动力商品的使用价值具有成为价值源泉的特殊属性,它的实际使用本身就是劳动的物化,从而创造了剩余价值,使货币转化为资本。” 马克思的分析揭露了资本拜物教产生的实质,为科学认识资本主义经济制度的本质提供了理论武器。

        # 运用历史和现实的事实说明经济危机是资本主义基本矛盾的集中体现。答:

        • 资本主义发展到一定阶段,就会出现生产的相对过剩,这是经济危机的基本特征。经济危机发生时,大量商品积压,企业减产甚至停工,伴随着许多金融机构倒闭,整个社会经济生活一片混乱。对该论述理解如下:

        • (1)在资本主义经济的发展过程中,经济危机是周期性地重演的,危机与危机之间的间隔表现了一定的规律性。第二次世界大战后,各主要资本主义国家发生了次数不等的经济危机。在战后各国的历次危机中,有的是属于部分国家同期发生的,有的是普遍性的资本主义世界经济危机。属于世界性经济危机的有三次,即 1957—1958 年、1973—1975 年和 1980—1982 年的经济危机,因为这三次危机表现了明显的国际同期性。
          第二次世界大战结束以来,随着国家垄断资本主义的形成和发展,虽然市场机制依然在资源配置过程中发挥着基础性调节作用,但它已不是唯一的经济调节机制了。调节机制发生了一些明显的变化,其主要表现是资产阶级国家对经济的干预不断加强。国家已经承担起了提供财产保护、增强国家竞争力、实现经济增长和充分就业、保持经济稳定、提高社会福利水平以及维护竞争秩序等重要职能。它与市场机制相辅相成,共同推动着资本主义经济的运行和发展。

        • (2)生产过剩不是生产的商品绝对超过了人们的物质生活需要,而是一种相对过剩,即生产的商品相对于人民群众有支付能力的需求来说是过剩了。生产相对过剩是资本主义经济危机的实质。经济危机的抽象的一般的可能性,是由货币作为流通手段和支付手段引起的。以货币为媒介的商品买卖在时间上分为两个相互独立的行为。如果一些商品生产者在出卖自己的商品后不接着购买他人生产的商品,就会有另一些商品生产者的商品卖不出去。同时,在商品买卖有更多的部分采取赊购赊销的情况下,若某些债务人在债务到期时不能支付,就会使整个信用关系体系遭到破坏。但是,这仅仅是危机在形式上的可能性。

        • (3)资本主义经济危机是资本主义基本矛盾存在、激化和集中暴露的证明。这一基本矛盾具体表现在两个方面:①生产无限扩大的趋势与劳动人民现有支付能力和需求相对缩小的矛盾;②个别企业内部生产的有组织性和整个社会生产的无政府状态之间的矛盾。正如马克思所指出的:“一切现实的危机的最终原因始终是:群众贫穷和群众的消费受到限制,而与此相对立,资本主义生产却竭力发展生产力,好像只有社会的绝对的消费能力才是生产力发展的界限。”

        • (4)在经济调节机制变化的同时,经济危机形态也发生了变化,表现为:危机对社会经济运行的干扰减轻,破坏性减弱,生产下降的幅度减小,失业率有所降低,企业破产的数量减少;危机周期的长度缩短;经济危机的四个阶段(即危机阶段、萧条阶段、复苏阶段和高涨阶段)之间的差别有所减弱,各阶段的交替过程已不如过去那样明显;金融危机对整个经济危机的影响加强。随着调节机制的变化,资本主义经济危机的形态尽管发生了这样或那样的变化,但是由于资本主义的基本矛盾所决定,资产阶级不论采取什么样的调节措施和手段,经济危机是克服不了的。经济危机是资本主义基本矛盾的集中体现。

        # 有人认为,资本主义民主是囿于 “钱主” 的民主,迷于 “游戏” 的民主,止于 “选举” 的民主。你如何看待这种说法,为什么?

        • 答:资本主义民主存在着自身无法克服的 “软肋” 和 “硬伤”,这是因为资本主义政治制度为资产阶级专政服务,其本质上是资产阶级进行政治统治和社会管理的手段和方式,不可避免地有其阶级和历史的局限性。

        • (1)资本主义的民主是金钱操纵下的民主,是资产阶级精英统治下的民主。民主是具体的、相对的,而不是抽象的、绝对的。资本主义社会的主导逻辑是资本逻辑,不仅经济领域服从这一逻辑的统治,民主政治领域同样服从这一逻辑的主宰。西式民主的实质是 “钱主”,金钱是西式民主的 “母乳”,无论是政党还是政客都在母乳的滋养下成长,政客与财团之间结成生死与共的命运共同体,也常被金钱、媒体、黑势力、财团等影响和操纵。在实际的选举过程中,为得到选民的支持和认可,参选者往往需要支付巨额的竞选活动经费,这笔经费是普通公民根本没有能力支付的,只有依靠大财团才能支付得起。资本主义政治制度中的选举只能是有钱人的游戏,是资本玩弄民意的过程。号称民主典范的美国,其总统和参众两院议员的选举在现实中已经沦为烧钱的游戏,一人一票异化为 “一元一票”,筹集竞选经费的能力早已成为问鼎白宫的风向标,金钱是 “打开权力之门的金钥匙”。历史学家作过统计,从 1860 年以来的历次美国总统大选中,竞选经费占优的一方几乎都获得了胜利。“钱主” 的后果就是 “钱权联姻”。在西方,钱与权具有天然的近亲关系,“钱能生权,权又能生更多的钱”“政治献金” 与 “政治分赃” 总是如影随形。美国民主的实质就是 “1% 所有,1% 统治,1% 享用”。

        • (2)资本主义的民主是迷于 “游戏” 的民主。资本家和富人拥有特权,这对劳动者和穷人严重不平等,资产阶级法律将这种不平等合法化。民主的游戏化、娱乐化,这是西方的又一杰作。西方将民主变成了游戏,选民以娱乐的心态对待民主。西式民主在游戏中沉沦,选民在娱乐中迷茫。在这场游戏中,赢的永远是政客,输的始终是选民。

        • (3)资本主义的民主是止于 “选举” 的民主。民主不等于选举,民主和选举不能等量齐观、等同视之。真正的民主体制应该包括民主选举、民主决策、民主管理、民主监督等各个环节,覆盖起点、过程、结果等各个阶段。资本主义国家的多党制仍是资产阶级选择自己的国家管理者,以实现其内部利益平衡的政治机制。然而,现代资本主义民主却大都沉迷于 “选举” 这一初始环节,“人民主权” 被置换成 “人民的选举权”,民主被简化为选举,选举进一步简化为投票,而对于决策是否民主、管理是否民主、监督是否民主等方面却并不感兴趣。止于 “选
          举” 的民主只顾开头,不顾过程和结尾。这种民主即便有始却不一定有终,即便能善始却未必能善终。其实,民主是基于文化传统和现实国情的长期实践,采取何种民主形式必须与各个国家的历史文化传统、社会状况、人口结构、宗教信仰、民族构成、经济发展水平、法制意识、国民素质等因素相结合,否则很难有效运转。

        # 第五章 资本主义的发展及其趋势

        # 垄断是怎样产生的?为什么说垄断并没有消除竞争?答:

        • (1)垄断产生的原因

          • ① 当生产集中发展到相当高的程度,极少数企业就会联合起来,操纵和控制本部门的生产和销售,实行垄断,以获得高额利润。

          • ② 企业规模巨大,形成对竞争的限制,也会产生垄断。

          • ③ 激烈的竞争给竞争各方带来的损失越来越严重,为了避免两败俱伤,企业之间会达成妥协,联合起来,实行垄断垄断组织垄断是通过一定的垄断组织形式实现的。垄断组织是指在一个或几个经济部门中,占据垄断地位的大企业联合。垄断组织的形式多种多样,如初级的短期价格协定。但本质上都是通过联合达到独占和瓜分商品生产和销售市场,操纵垄断价格,以攫取高额垄断利润。

        • (2)垄断没有消除竞争的原因

          • ① 垄断没有消除产生竞争的经济根源。

          • ② 垄断必须通过竞争来维持。

          • ③ 社会生产是复杂多样的,任何垄断组织都不可能把包罗万象的社会生产都包下来。

        • 为什么说国家垄断资本主义体现了资本主义生产关系的部分质变?
          答:国家垄断资本主义是垄断资本和国家政权密切结合的垄断资本主义。国家垄断资本主义的产生,标志着资本主义发展进入了新的阶段。国家垄断资本主义体现了资本主义生产关系的部分质变的原因是:

          • (1)国家垄断资本主义是垄断资本主义的新发展,它对资本主义经济的发展产生了积极的作用。

            • ① 国家垄断资本主义的出现在一定程度上有利于社会生产力的发展,资本主义国家的政府通过公共投资和支出,开办大型新兴工业企业,部分克服了社会化大生产与私人垄断资本之间的矛盾。
            • ② 私人垄断统治扩大了资本主义生产与消费的矛盾,资产阶级国家采取一定手段干预国民经济,减轻经济危机的程度,缓和了社会各阶级的利益矛盾,有利于社会稳定,促进经济较为协调地发展。
            • ③ 宏观调控的发展,使国家对经济的干预和调节与市场机制日益有机结合起来,自由放任的市场经济逐渐演变成为由国家干预的现代市场经济,一定程度上弥补了自由市场经济的缺陷,对于国家的经济运行和社会发展产生了积极作用。
            • ④ 资产阶级国家通过税收和收入再分配手段,提高了劳动人民的生活水平。
          • (2)国家垄断资本主义的出现并没有根本改变垄断资本主义的性质。

            • ①国家垄断资本主义仍然是以资本主义私有制为基础,维护资产阶级利益和资本主义制度。
            • ②国家垄断资本主义没有从根本上消除资本主义的基本矛盾,它代表的是资产阶级的总利益。

        # 如何认识和把握经济全球化及其影响?

        • 答:经济全球化是指在生产不断发展、科技加速进步、社会分工和国际分工不断深化、生产的社会化和国际化程度不断提高的情况下,世界各国、各地区的经济活动越来越超出某一国家和地区的范围而相互联系、相互依赖的过程。

        • (1)经济全球化的表现

          • ① 国际分工进一步深化。

          • ② 贸易全球化。

          • ③ 金融全球化。

          • ④ 企业生产经营全球化。

        • (2)经济全球化的动因从本质上讲,经济全球化是生产力发展和社会化大生产的必然要求。导致经济全球化迅猛发展的因素主要有三点:

          • ① 科学技术的进步和生产力的发展提供了坚实的物质基础和根本的推动力。

          • ② 跨国公司的发展提供了适宜的企业组织形式。

          • ③ 各国经济体制的变革是经济全球化的体制保障。

        • (3)经济全球化的影响

          • ① 积极影响

            • a.社会分工得以在更大的范围内进行,资金、技术等生产要素可以在国际社会流动和优化配置,由此带来巨大的分工利益,推动世界生产力的发展。

            • b.体现了社会化生产的要求,不仅发达国家从中受益,一些发展中国家在参与经济全球化进程中也得到了快速发展。全球化为发展中国家提供先进技术和管理经验、就业机会,推动发展中国家国际贸易和跨国公司发展。

          • ② 消极影响

            • a.发达国家与发展中国家在经济全球化过程中的地位和收益不平等、不平衡。

            • b.加剧了发展中国家资源短缺和环境污染恶化。

            • c.一定程度上增加经济风险。

          • ③ 正确看待经济全球化

            • a.“把困扰世界的问题简单归咎于经济全球化,既不符合事实,也无助于问题解决。” 国际金融危机不是经济全球化发展的必然产物,而是金融资本过度逐利、金融监管严重缺失的结果。

            • b.经济全球化不是一部分国家的独角戏,而是世界各国、各民族共同实现发展的大舞台。

            • c.面对不同国家在生产方式、发展水平、文化背景等方面的差异,要以共同构建人类命运共同体的理念引领经济全球化。

        # 近年来资本主义社会发生了哪些新变化?试分析其原因及影响。答:

        • (1)资本主义社会的新变化

          • ① 生产资料所有制的变化:

            • a.国家资本所有制是指生产资料由国家占有并服务于垄断资本的所有制形式。国家资本所有制主要存在于基础设施和公共事业部门,所以对整个社会经济的发展有着重要的影响。但就其性质而言,仍然是资本主义形式,体现着总资本家剥削雇佣劳动者的关系。

            • b.法人资本所有制是资本主义生产资料所有制发展的新形式,是法人股东化的产物。法人资本所有制有两种形式:一种是企业法人资本所有制,另一种是机构法人资本所有制,但其在性质上是一种基于资本雇佣劳动的垄断资本。

          • ② 集体所有制劳资关系和分配关系的变化

            • a.职工参与决策。这一制度旨在协调劳资关系,缓和阶级矛盾。

            • b.终身雇佣。目的是增强工人对企业的归属意识,从而更加自觉地服从资本家的统治。

            • c.职工持股。旨在通过使职工持有一部分本公司的股份来调动工人的生产积极性,使工人产生归属感,在生产中努力提高劳动生产率,增加剩余价值生产。

            • d.普及化、全民化的社会福利制度。在一定程度上满足劳动者的安全和保障需求,保证劳动者维持最低生活水平,改善劳动者的社会状况。

          • ③ 社会阶层和阶级结构的变化

            • a.资本家的地位和作用已经发生很大变化。

            • b.高级职业经理成为大公司经营活动的实际控制者。

            • c.知识型和服务型劳动者的数量不断增加,劳动方式发生了新变化。

          • ④ 经济调节机制的变化 a.去工业化和产业空心化日趋严重,产业竞争力下降。

            • b.经济高度金融化,虚拟经济与实体经济严重脱节。

            • c.财政严重债务化,债务危机频繁爆发。

            • d.两极分化和社会对立加剧。

            • e.经济增长乏力,发展活力不足,周期性危机与结构性危机交织在一起。

            • f.金融危机频发,全球经济屡受打击。

          • ⑤ 政治制度的变化

            • a.政治制度出现多元化的趋势,公民权利有所扩大。

            • b.重视并加强法制建设。

            • c.改良主义政党在政治舞台上的影响日益扩大。

        • (2)资本主义社会的新变化的原因

          • ① 科学技术革命和生产力的发展,是当代资本主义发生新变化的根本推动力量。

          • ② 工人阶级争取自身权利和利益的斗争,是推动当代资本主义发生新变化的重要力量。

          • ③ 社会主义制度初步显示的优越性对当代资本主义产生了重要影响。

          • ④ 主张改良主义的政党对资本主义制度的改革,也对当代资本主义新变化发挥了重要作用。

        • (3)资本主义社会的新变化的影响

          • ① 当代资本主义发生的变化从根本上说是人类社会发展一般规律和资本主义经济规律作用的结果。

          • ② 当代资本主义发生的变化是在资本主义制度基本框架内的变化,并不意味着资本主义生产关系的根本性质发生了变化。

          • ③ 没有改变其为资产阶级利益服务,仍然是服从于资产阶级进行统治和压迫需要的政治工具的本质属性,没有改变马克思主义关于资本主义的基本论断的科学性。

        # 如何理解资本主义的历史地位及其为社会主义所代替的历史必然性?

        • 答:从人类社会发展的长河看,社会主义取代资本主义,即社会主义公有制取代资本主义私有制,这是历史发展的基本趋势。

        • (1)资本主义的内在矛盾决定了资本主义必然被社会主义所代替

          • ① 资本主义基本矛盾 “包含着现代的一切冲突的萌芽”。

            • a.资本主义基本矛盾表现在阶级关系上,是无产阶级和资产阶级的对立。

            • b.资本主义基本矛盾表现在经济关系上,是个别工厂中生产的组织性和整个社会中生产的无政府状态之间的对立。

            • c.资本主义经济危机的爆发正是这个基本矛盾发展的结果。

          • ② 资本积累推动资本主义基本矛盾不断激化并最终否定资本主义自身。

            • a.从资本主义积累过程来看,资本主义基本矛盾在资本积累过程中不断发展。

            • b.资本的不断积累必然提高生产的社会化程度,这在客观上势必导致生产的集中和资本的集中,使资本的社会化占有成为可能。

            • c.资本的不断积累使对生产过程的管理社会化,相应地派生出管理社会化大生产的管理人员和专业的管理机构,而这些都弱化甚至排斥资本家个人在管理中的地位和作用。

            当资本主义基本矛盾及其派生的各种矛盾在资本积累中不断发展、激化到资本主义制度自身无法使之释放时,社会主义取代资本主义就将成为不可避免的结果。这是资本主义积累过程所具有的客观历史趋势。

          • ③ 国家垄断资本主义中孕育着某些社会主义的因素,将成为社会主义的前奏。

          • ④ 资本主义社会中存在着资产阶级和无产阶级两大阶级之间的矛盾和斗争。无产阶级和资产阶级之间不可调和的矛盾、对立和斗争,必然导致资本主义政治危机。资产阶级的灭亡和无产阶级的胜利是不可避免的。通过无产阶级革命推翻资本主义制度,用社会主义公有制代替资本主义私有制,成为人类社会由低级向高级发展的客观规律的必然要求。

        • (2)社会主义代替资本主义是一个复杂的、长期的历史过程。其原因在于:

          • ① 社会主义自身的巩固、发展和完善是一个长期的历史过程。

            • a.经济上的强烈反差使社会主义国家面临严峻挑战。

            • b.西方敌对势力的颠覆、破坏与和平演变活动,加重了社会主义建设的曲折性、复杂性和长期性。

            • c.对社会主义建设规律的探索需要经过一个长期曲折的过程。

          • ② 社会主义在世界范围代替资本主义是一个长期的历史过程。

            • a.社会主义的本质决定了它在全世界取代资本主义的长期性、曲折性与艰巨性。

            • b.资本主义的暂时强大决定了它在全世界被社会主义所代替的长期性、曲折性与艰巨性。

            • c.20 世纪末世界社会主义运动遭受了重大挫折,导致社会主义顺利发展、再造辉煌需要较长的时间。

        # 第六章 社会主义的发展及其规律(本章仅供学习参考,不作考试要求)

        # 如何认识世界社会主义五百年的发展历程?

        • 答:世界社会主义五百年的发展过程实际上经历了四个阶段,分别是社会主义从空想到科学、社会主义从理论到实践、社会主义从一国到多国、在中国焕发强大生机活力。

        • (1)社会主义从空想到科学

          • ①空想社会主义

          空想社会主义产生于 16 世纪初期,到 19 世纪上半叶达到顶峰。这 300 多年正是欧洲从封建主义生产方式向资本主义生产方式转变的时期。其发展大致分为三个阶段:16 世纪和 17 世纪的早期空想社会主义,有理想社会制度的空想的描写,代表著作是英国人莫尔的《乌托邦》和意大利人康帕内拉的《太阳城》;18 世纪的空想平均共产主义;19 世纪初期出现了批判空想社会主义。主要代表是法国的圣西门、傅立叶和英国的欧文,以三大空想社会主义者为代表的空想社会主义学说,它对资本主义的揭露和批判,提供了启发工人觉悟的极为宝贵的材料。空想社会主义是早期无产阶级意识和利益的先声,反映了早期无产阶级迫切要求改造现存社会、建立理想的新社会的愿望。但其没有能够指出真正的出路。空想社会主义看不到无产阶级负有的历史使命,反对政治斗争,幻想依靠统治阶级的帮助,通过和平的方式来实现社会主义。

          • ② 科学社会主义

          19 世纪 40 年代,资本主义生产方式在西欧先进国家已占统治地位,随着资本主义的发展,资本主义内部矛盾日益尖锐,无产阶级反对资产阶级的斗争日益高涨。马克思和恩格斯参加了当时阶级斗争的实践。在此基础上,周密地研究了资本主义生产方式的矛盾,批判地继承了 19 世纪三大空想社会主义者法国的圣西门、傅立叶和英国的欧文的思想成果,创立了唯物史观和剩余价值论,为实现社会主义从空想到科学的飞跃奠定了坚实的理论基础。他们阐明了生产社会性和生产资料资本主义私人占有形式之间的矛盾的发展,必然导致社会主义取代资本主义,以生产资料的公有制取代生产资料私有制,科学地论述了资本主义必然灭亡、社会主义必然胜利的客观规律,从根本上超越了空想社会主义,实现了社会主义从空想到科学的伟大飞跃。

        • (2)社会主义从理论到现实

          • ① 第一国际与巴黎公社

          1848 年欧洲革命后,资本主义在各国飞速发展,无产阶级力量不断壮大。到 19 世纪 50 年代末 60 年代初,反压迫反剥削的斗争实践使各国无产阶级认识到,他们有着共同的利益和共同的敌人,而以往分散的斗争常常使他们遭到同样的失败,无产阶级必须在国际范围内联合起来。1864 年,国际工人协会(第一国际)应运而生。马克思是第一国际的真正领袖,在他指导下,第一国际把无产阶级先进分子团结在自己周围,率领各国工人群众向资产阶级和压迫者进行坚决斗争,促进了马克思主义的传播和与国际工人运动的结合,初步确立了马克思主义在工人运动中的指导地位。1871 年爆发的巴黎公社革命,深受第一国际精神的影响,是建立无产阶级专政的第一次伟大尝试。公社在经济方面采取了措施,对资本主义企业进行了一定程度的限制和剥夺;成立了管理经济工作的最高权力机构 —— 粮食委员会、财政委员会和劳动、就业及交换委员会;对工人群众的生活状况和劳动条件尽可能加以改善,提出了劳动者八小时工作制的原则。巴黎公社是世界上无产阶级武装暴力直接夺取城市政权的第一次尝试。它丰富和发展了马克思主义关于阶级斗争和社会主义的学说,在国际共产主义运动上写下了光辉、伟大而悲壮的一页。

          • ② 十月革命与第一个社会主义国家的建立

          1917 年 11 月 7 日(俄历 10 月 25 日)作为伟大的十月社会主义革命胜利日载入史册。俄国十月社会主义革命是人类历史上第一次获得胜利的社会主义革命,世界上第一个社会主义国家由此诞生。十月革命的胜利沉重打击了帝国主义的统治,推动了国际社会主义运动的发展,鼓舞了殖民地半殖民地人民的解放斗争。它改变了俄国历史的发展方向,用社会主义方式改造俄国的道路,对整个人类社会的发展都产生了巨大的影响。十月革命是二十世纪国际共产主义运动的序幕,触发了此后各国社会主义运动在全球范围的扩张,许多殖民地或半殖民地的解放运动也因此得到了更多支持,加速了世界范围内帝国主义殖民体系的整体瓦解,深刻改变了国际力量对比和世界格局。

        • (3)社会主义从一国到多国

          • ① 社会主义在苏联一国的实践

          苏维埃俄国是世界上第一个社会主义国家。年轻的苏维埃政权建立之后,既面临着巩固政权的任务,又面临着在经济文化相对落后的条件下建设社会主义的任务。列宁对苏维埃俄国社会主义的探索大体经历了巩固苏维埃政权时期、战时共产主义时期和新经济政策时期。

          • ② 社会主义发展到多个国家俄国十月革命的胜利后,第一个社会主义国家诞生,开辟了人类历史新纪元,极大地鼓舞了世界各国人民,促进了世界社会主义运动的发展。二战以后,社会主义在世界范围内获得大发展,在欧洲、亚洲、拉丁美洲,先后有 15 个国家建立社会主义制度,在国家数量上,占世界的 1/14,在领土上,占世界的 1/4,在人口上,占总人口的 1/3。社会主义国家相继开展了大规模的社会主义建设,推动了国家经济、政治、文化和社会各项事业的发展,世界形成了社会主义和资本主义两种制度共处的新格局。
        • (4)社会主义在中国焕发出强大生机活力
          俄国十月革命的胜利,给中国人民送来了马克思列宁主义。1921 年 7 月,中国共产党成立并成为中国社会主义运动的领导力量。中国共产党率领中国人民进行的社会主义事业经历了从新民主主义革命到社会主义革命、建设、改革的发展过程,中国共产党在近百年的奋斗中不断发展壮大,在 21 世纪焕发出强大的生机活力。习近平总书记在十九大报告中指出:“中国特色社会主义进入新时代,意味着近代以来久经磨难的中华民族迎来了从站起来、富起来到强起来的伟大飞跃,迎来了实现中华民族伟大复兴的光明前景;意味着科学社会主义在 21 世纪的中国焕发出强大生机活力,在世界上高高举起了中国特色社会主义伟大旗帜;意味着中国特色社会主义道路、理论、制度、文化不断发展,拓展了发展中国家走向现代化的途径,给世界上那些既希望加快发展又希望保持自身独立性的国家和民族提供了全新选择,为解决人类问题贡献了中国智慧和中国方案。” 总之,
          中国特色社会主义是科学社会主义 “中国化” 的伟大成果,是在当代中国的成功实践,充分表明了社会主义在世界上人口最多的国家成功开辟出了通向繁荣富强的正确道路,显著地展现了社会主义制度的优越性,标志着社会主义在世界范围内正在开拓新的历史征程。

        # 如何认识和把握科学社会主义一般原则?答:

        • (1)科学社会主义一般原则的主要内容
          科学社会主义一般原则是社会主义事业发展规律的集中体现,是马克思主义政党领导人民进行社会主义革命、建设、改革的基本遵循。

          • ① 人类社会发展规律和资本主义基本矛盾是 “资本主义必然灭亡、社会主义必然胜利” 的根本依据。

          • ② 无产阶级是最先进最革命的阶级,肩负着推翻资本主义旧世界、建立社会主义和共产主义新世界的历史使命。

          • ③ 无产阶级革命是无产阶级进行斗争的最高形式,以建立无产阶级专政的国家为目的。

          • ④ 社会主义社会要在生产资料公有制基础上组织生产,以满足全体社会成员的需要为生产的根本目的。

          • ⑤ 社会主义社会要对社会生产进行有计划的指导和调节,实行按劳分配原则。

          • ⑥ 社会主义社会要合乎自然规律地改造和利用自然,努力实现人与自然的和谐共生。

          • ⑦ 社会主义社会必须坚持科学的理论指导,大力发展社会主义先进文化。

          • ⑧ 无产阶级政党是无产阶级的先锋队,社会主义事业必须始终坚持无产阶级政党的领导。

          • ⑨ 社会主义社会要大力解放和发展生产力,逐步消灭剥削和消除两极分化,实现共同富裕和社会全面进步,并最终向共产主义社会过渡。

        • (2)正确对待科学社会主义一般原则的科学态度

          • ① 必须始终坚持社会主义一般原则,反对任何背离科学社会主义一般原则的错误倾向。

          • ② 要善于把科学社会主义一般原则与本国实际相结合,创造性地回答和解决革命、建设、改革中的重大问题。

          • ③ 紧跟时代和实践的发展,在不断总结新鲜经验中进一步丰富和发展科学社会主义一般原则。

        # 请结合科学社会主义一般原则,谈谈为什么说中国特色社会主义是科学社会主义,而不是其他什么主义。答:

        • (1)科学社会主义一般原则
          科学社会主义一般原则是社会主义事业发展规律的集中体现,是马克思主义政党领导人民进行社会主义革命、建设、改革的基本遵循。马克思、恩格斯在深刻揭示人类社会发展一般规律的基础上,深入阐发资本主义基本矛盾及其发展趋势,并在指导国际工人运动的过程中不断总结经验,逐步形成了科学社会主义一般原则。这些原则在后来的社会主义革命和建设中得到了证实、丰富和发展。这些原则具体如下:

          • ① 人类社会发展规律和资本主义基本矛盾是 “资本主义必然灭亡、社会主义必然胜利” 的根本依据。

          • ② 无产阶级是最先进最革命的阶级,肩负着推翻资本主义旧世界、建立社会主义和共产主义新世界的历史使命。

          • ③ 无产阶级革命是无产阶级进行斗争的最高形式,以建立无产阶级专政的国家为目的。

          • ④ 社会主义社会要在生产资料公有制基础上组织生产,以满足全体社会成员的需要为生产的根本目的。

          • ⑤ 社会主义社会要对社会生产进行有计划的指导和调节,实行按劳分配原则。

          • ⑥ 社会主义社会要合乎自然规律地改造和利用自然,努力实现人与自然的和谐共生。

          • ⑦ 社会主义社会必须坚持科学的理论指导,大力发展社会主义先进文化。

          • ⑧ 无产阶级政党是无产阶级的先锋队,社会主义事业必须始终坚持无产阶级政党的领导。

          • ⑨ 社会主义社会要大力解放和发展生产力,逐步消灭剥削和消除两极分化,实现共同富裕和社会全面进步,并最终向共产主义社会过渡。

        • (2)中国特色社会主义是科学的社会主义的原因
          中国特色社会主义,是科学社会主义理论逻辑和中国社会发展历史逻辑的辩证统一。中国特色社会主义既坚持了科学社会主义的基本原则,又根据时代条件赋予其鲜明的中国特色,是一百多年来科学社会主义理论与实践发展的结晶,是当代中国的科学社会主义。

          • ① 新中国成立后,中国共产党领导人民开始社会主义革命和建设的伟大实践。在只有苏联模式可学的情况下,在开始进行社会主义建设时我们走了照搬苏联经验的路子。但是,我们党很快就觉察到苏联模式的种种弊端,果断决定独立探索适合中国国情的社会主义建设道路。这一探索尽管曾出现过脱离实际的严重失误,却还是取得了巨大的成绩,形成了关于社会主义建设的许多独创性的理论成果和实践成果。在新的历史时期,我们党坚持解放思想、实事求是,深刻总结历史经验教训,正确判断时代主题和基本国情,经过 30 多年的努力,形成了中国特色社会主义道路、中国特色社会主义理论体系、中国特色社会主义制度三位一体的伟大成果,取得了中国特色社会主义建设的巨大成就。

          • ② 中国特色社会主义之所以是社会主义而不是别的什么主义,就在于它始终坚持以科学社会主义的基本原则为理论源泉和理论核心。科学社会主义创始人对社会主义的设想和他们提出的科学社会主义基本原则的最终实现,需要一个过程。我们党始终将其作为核心价值追求,始终将其作为进行改革开放和实践创新、理论创新、制度创新的 “魂”。这样一种追求,这样的 “魂”,鲜明地体现在改革开放新时期我们党所制定的路线、方针、政策中。

          • ③ 科学社会主义包含极其丰富的内容,集中到一点,就是科学地论证了社会主义必然代替资本主义的历史趋势,论证了无产阶级取代资本主义、建设社会主义和共产主义的伟大历史使命。科学社会主义理论创立以来的历史证明它的理论是正确的,社会主义具有强大的生命力。同时,它又不是一成不变的教条。它在指导实践的过程中,又要接受实践的检验,随着实践的发展而不断扩展、不断深化、不断丰富、不断完善。中国特色社会主义理论体系,就是科学社会主义在当代中国的新发展。

          • ④ 当前,资本主义现实和社会主义实践同马克思、恩格斯两位伟大思想家所处的历史环境相比,都发生了巨大的变化,但从世界社会主义发展的五百年大视野来看,目前依然处在马克思主义所指明的历史时代,处在资本主义向社会主义转变的历史进程之中,科学社会主义从没有过时,仍然有强大的生命力,必须始终坚持、不动摇。正如习近平指出:“中国特色社会主义是社会主义而不是其他什么主义,科学社会主义基本原则不能丢,丢了就不是社会主义。”

        # 请谈谈中国特色社会主义的成功实践对世界社会主义发展的贡献。答:

        • 党的十九大报告作出了 “中国特色社会主义进入新时代” 的重大政治论断。习近平总书记在报告中指出:“中国特色社会主义进入新时代,意味着近代以来久经磨难的中华民族迎来了从站起来、富起来到强起来的伟大飞跃,迎来了实现中华民族伟大复兴的光明前景;意味着科学社会主义在二十一世纪的中国焕发出强大生机活力,在世界上高高举起了中国特色社会主义伟大旗帜;意味着中国特色社会主义道路、理论、制度、文化不断发展,拓展了发展中国家走向现代化的途径,给世界上那些既希望加快发展又希望保持自身独立性的国家和民族提供了全新选择,为解决人类问题贡献了中国智慧和中国方案。” 这一重大政治论断给我们许多启示,增加了我们对中国特色社会主义的自信,也使我们充分认识到中国特色社会主义不仅在 21 世纪的中国充分显现出生机与活力,取得伟大成功,使科学社会主义理论在中国实践中获得巨大发展,也使得中国特色社会主义超越了国界,具有世界性的意义,为发展中国家走向现代化提供了新的路径选择,也为世界发展和人类问题的解决贡献了中国智慧和中国方案。

        • (1)中国特色社会主义为世界贡献了中国智慧
          中国走上中国特色社会主义道路,是几代中国共产党人带领全国人民在艰难曲折中艰辛探索、不断实践创新的结果,体现了中国共产党人的不屈不挠,也体现了中国共产党人的卓越智慧,其中最核心的就是党的实事求是的思想路线,一切从实际出发,把马克思主义基本原理与中国客观实际相结合,走适合中国国情的社会主义道路。改革开放以来,我们党坚持实事求是的思想路线,在实践中不断探索和回答了什么是社会主义、怎样建设社会主义,建设什么样的党、怎样建设党,实现什么样的发展、怎样发展,并继续探索和回答新时代坚持和发展什么样的中国特色社会主义、怎样坚持和发展中国特色社会主义这些基本问题。党的思想路线体现的是马克思主义的世界观、方法论。因此,中国特色社会主义的成功,是马克思主义世界观和方法论的成功,是马克思主义中国化、时代化的成功。中国特色社会主义贡献给世界的智慧,最为根本的是实事求是,国家建设一定要从自己的国情出发,探索适合自己国情的建设道路,任何脱离实际、不符合国情的道路、方法、模式都是不可能取得成功的。

        • (2)中国特色社会主义为世界贡献了中国方案
          中国特色社会主义不仅为世界贡献了指导国家建设的正确的思想方法,还为世界贡献了国家建设的具体方案和治国理政的基本方略。中国特色社会主义治国方案既包括总目标、总任务、总体布局、战略布局和发展方向、发展方式、发展动力、战略步骤、外部条件、政治保证等基本方面的总体设计,也包括经济、政治、科技、文化、教育、民生、民族、宗教、社会、生态、国家安全、国防和军队、国家统一、统一战线、外交、党的建设等各方面的基本规定。还有就是,坚持党对一切工作的领导,坚持以人民为中心,坚持全面深化改革,坚持新发展理念,坚持人民当家作主,坚持全面依法治国,坚持社会主义核心价值体系,坚持在发展中保障和改善民生,坚持人与自然和谐共生,坚持总体国家安全观,坚持党对人民军队的绝对领导,坚持 “一国两制” 和推进祖国统一,坚持推动构建人类命运共同体,坚持全面从严治党等基本方略。这些既是新时代坚持和发展中国特色社会主义的基本方略,也是中国特色社会主义为解决人类问题向世界贡献的中国方案,给世界上那些既希望加快发展又希望保持自身独立性的国家和民族提供了全新选择。

        # 如何正确认识和把握共产党执政规律?

        答:可以从以下几个方面正确认识和把握共产党执政规律:

        • (1)共产党执政规律,最根本的是人心向背定理。习近平总书记指出:“一个政党,一个政权,其前途和命运最终取决于人心向背。”“人心是最大的政治。” 要始终把人民放在心中最高位置。坚持一切为了人民、一切依靠人民。

        • (2)共产党执政首要的是坚持党中央权威和集中统一领导,党是最高政治领导力量,党的领导是全面领导。党的十九大报告指明 “坚持党中央权威和集中统一领导,是党的政治建设的首要任务”,揭示了保证党的团结和集中统一的根本性问题,阐明了进行伟大斗争、建设伟大工程、推进伟大事业、实现伟大梦想的根本保证,意义重大而深远。

        • (3)共产党在执政过程中,要把党的政治建设摆在首位。要坚持党要管党、全面从严治党,以应对各种考验,坚定不移地惩治和预防腐败,勇于自我革命。只有全面从严治党,才能顺利实现中华民族伟大复兴的中国梦。

        • (4)要始终抓住科学执政、民主执政、依法执政这个要律,为实现国家治理现代化指明方向路径。

        # 第七章 共产主义是人类最崇高的社会理想

        # 请谈谈在未来理想社会的认识上,马克思主义经典作家与空想社会主义者有何本质区别。

        • 答:在未来理想社会的认识上,马克思主义经典作家与空想社会主义者的本质区别是:是否坚持科学的立场、观点和方法来预见未来。预见未来社会的方法论原则主要包括:

        • (1)在揭示人类社会发展一般规律的基础上指明社会发展的方向。

        • (2)在剖析资本主义旧世界的过程中阐发未来新世界的特点。

        • (3)在社会主义社会发展中不断深化对未来共产主义社会的认识。

        • (4)立足于揭示未来社会的一般特征,而不作详尽的细节描绘。

        # 既然共产主义理想的实现是历史的必然,为什么又要人们去努力追求?既然共产主义的最终实现是一个漫长的过程,为什么又说 “共产主义渺茫论” 是错误的?请用马克思主义的辩证观点予以解答。答:

        • (1)实现共产主义的历史必然

          • ① 共产主义理想的实现是历史规律的必然要求

            • a.共产主义理想是能够实现的理想,它与一切空想和幻想有着本质区别。

            • b.共产主义理想一定会实现,是以人类社会发展规律以及资本主义社会的基本矛盾发展为依据的。

            • c.社会主义运动的实践,特别是社会主义国家的兴起和不断发展,已经且正在用事实证明共产主义理想实现的必然性。从一定意义上讲,社会主义革命的胜利本身就是对共产主义理想可以实现的证明。

          • ② 实现共产主义是人类最伟大的事业

            • a.实现共产主义理想是广大人民群众的共同愿望。

            • b.实现共产主义,必须找到现实的阶级力量,这就是现代工人阶级或无产阶级。

            • c.无产阶级的解放与全人类的解放是完全一致的。

        • (2)共产主义的实现是一个漫长的过程

          • ① 资本主义的灭亡和向社会主义的转变是一个长期的过程

            • a.资本主义作为一个社会形态,其走向灭亡是一个长期的历史过程。资本主义国家经历过严重的经济和社会危机,在与社会主义国家的竞争中也伴随着巨大的压力。但资本主义并没有退出历史舞台,而是仍然有一定发展空间。这些危机背后隐藏着资本主义衰败的必然趋势,但目前谁也无法准确判断资本主义何时走到尽头。

            • b.社会主义革命取得胜利后,从资本主义转变为社会主义也需要一段时期。这个重要的过渡期是不以人的意志为转移的,它不能省略,更不可随意缩短。

            • c.在完成转变以后,仍然需要经历一段社会主义的发展阶段,才能最终逐渐走向共产主义。在任何国家,实现共产主义都不能超越社会主义发展阶段。不管资本主义国家发达到何种程度,当发生根本性的制度变革时,也不可能直接达到共产主义社会,只能首先进入社会主义社会,也就是共产主义社会的低级阶段。

            • d 要真正迎来新社会,消除旧社会痕迹,也需要经过一个很长的时期。

          • ② 社会主义社会自身巩固、发展和完善并最终成为共产主义也是一个长期的过程。

            • a.想要在全世界范围内实现共产主义,必须取决于社会主义国家的巩固和发展,取决于这些国家所经历的社会主义社会的时间长短。因此,共产主义只有在社会主义社会充分发展和高度发达的基础上才能实现。

            • b.经济上的强烈反差使社会主义国家面临严峻挑战。社会主义革命没有像马克思和恩格斯所设想的那样,在发达的资本主义国家首先发生,而是在少数经济和文化比较落后的国家首先取得胜利。这些国家基础差、起点低,要在经济上赶上资本主义发达国家,提高社会化生产水平,建设和完善社会主义政治、经济、文化等各项制度,体现出社会主义优于资本主义,需要付出更大、更艰苦和更长期的努力,是一个漫长的历史过程。

            • c.对社会主义建设规律的探索需要经过一个长期曲折的过程。建设社会主义是前无古人的崭新事业,没有现成固定的模式可循,需要各个社会主义国家在实践中把马克思主义的基本原理与本国的具体实践相结合,探索适合本国发展的社会主义建设道路和模式。而探索的过程不会是一帆风顺的,难免会出现不同程度的失误,甚至发生曲折和暂时倒退现象。这种探索过程的曲折性,无疑会增加这些国家建设、巩固和发展社会主义的艰巨性与长期性。

            • d.社会主义是共产主义的低级阶段,也是实现共产主义的必由之路。作为同一社会形态发展进程中的两个阶段,社会主义是共产主义的必要准备,共产主义是社会主义发展的必然趋势。随着社会主义生产力和其他因素的不断发展,属于社会主义的一些因素将逐步消失,而共产主义的所有因素将逐步增长。社会主义社会不断发展和完善的过程,就是共产主义因素不断扩大的过程。一旦条件完全成熟,社会主义社会就自然过渡到共产主义社会。

        • (3)实现共产主义是历史发展规律的必然要求,“共产主义渺茫论” 是错误的

          • ① 共产主义渺茫论者不了解共产主义的含义。

          共产主义有两方面的含义:一方面指的是在共产主义思想体系指导下的共产主义运动;另一方面指的是共产主义的社会制度。当然,共产主义社会制度彻底实现,是需要经过若干代人的不懈努力的,是一个远大的奋斗目标。但是,作为在共产主义思想体系指导下的共产主义运动,作为一种亿万人民在共产党领导下改造社会的革命实践,则是一点也不 “渺茫”,而是实实在在地、每日每时地存在于我们的现实生活中。一个多世纪以来,各国无产阶级及其政党为了实现共产主义制度这一伟大理想,进行了艰苦卓绝的斗争,诸如巴黎公社、十月社会主义革命以及我国的社会主义革命和社会主义建设,每向前发展一步都意味着向共产主义靠拢一步。今天,我们的一切革命工作都是伟大的共产主义运动的具体体现和组成部分。

          • ② 共产主义渺茫论者割裂了共产主义与社会主义之间的联系。

          共产主义渺茫论者忽视了一个很重要的事实,即共产主义运动不仅在全世界,而且在中国也早已存在,而且已经取得了显著成果,这就是共产主义社会的过渡阶段 —— 社会主义社会,已经在中国和世界一些国家建立起来并取得重大发展,从而使共产主义运动发生了质的飞跃。我国社会主义制度的建立,不仅使人民从被压迫和被剥削的旧制度下解放出来,而且,随着社会主义改革和建设的不断深入,人们生活的不断改善,社会主义制度的不断完善以及社会道德水准和文化教育科学事业的向前推进,共产主义理想逐渐深入人心。

          • ③共产主义渺茫论者不能正确对待社会主义事业中的曲折和困难。

          马克思主义辩证法告诉我们,事物的发展不是直线上升,而是在曲折中前进的。社会主义事业也是如此,它有成功,也有失误,有前进,也有暂时的困难。中国的改革和经济建设取得了举世瞩目的伟大成就,但也存在不少问题。只有坚持中国特色社会主义道路,集中精力把经济建设搞上去,人民生活得到了不断改善,人们才能看到社会主义的前途和希望。只要把社会主义建设好,才能为将来的共产主义打下了坚实的基础。我们要继续坚定不移地沿着社会主义道路勇往直前,坚信共产主义必胜的信念。

        # 有人说:“在社会主义初级阶段只能讲树立中国特色社会主义共同理想,而不应提树立共产主义远大理想,否则就是脱离实际。” 请用共同理想和远大理想的关系来分析评论这一观点。

        • 答:共产主义理想是建立在科学基础上的社会理想,是人类最伟大的社会理想。在坚持和发展中国特色社会主义的实践中,我们不但要坚定中国特色社会主义共同理想,而且要进一步树立共产主义远大理想。

        • (1)正确认识和把握共产主义远大理想与中国特色社会主义共同理想的关系

          • ①从时间上看,远大理想与共同理想的关系是最终理想与阶段性理想的关系。

          • ②从层次上看,远大理想与共同理想的关系是最高纲领与最低纲领的关系。实现共产主义远大理想要坚持最高纲领和最低纲领的统一。

          • ③从范围来看,远大理想与共同理想的关系也是全人类理想与全体中国人民理想的关系。

        • (2)以辩证思维把握和处理远大理想和共同理想的关系
          必须以马克思主义的辩证思维和历史思维去把握远大理想和共同理想的关系。实现中国特色社会主义的共同理想,是为了实现共产主义远大理想而服务。中国特色社会主义共同理想在我国的成功实践,必然为实现共产主义远大理想奠定坚实的基础。

        # 中国特色社会主义进入了新时代。为了不辜负这个伟大的时代,当代大学生应该怎样确立和追求自己的人生理想?答:

        • (1)新时代的特征
          中国特色社会主义进入新时代。我们党对 “什么是社会主义、如何建设社会主义” 的认识又向前推进了一步,从理论和实践结合上系统回答了 “新时代坚持和发展什么样的中国特色社会主义、怎样坚持和发展中国特色社会主义” 这一重大历史课题,习近平新时代中国特色社会主义思想成为指引全国人民沿着中国特色社会主义道路继续前进的重要理论基础。

        • (2)对当代大学生的要求习近平指出:“青年是标志时代的最灵敏的晴雨表,时代的责任赋予青年,时代的光荣属于青年。” 青年是祖国的未来、民族的希望。青年兴则国兴,青年强则国强。当前,中国特色社会主义进入新时代,为当代青年特别是当代大学生提供了实现人生才华的极为有利的历史机遇。

          • ①要坚定理想信念。心中有信仰,脚下才有力量。为此,就要深入学习马克思主义基本原理及马克思主义中国化的理论成果,特别是学习习近平新时代中国特色社会主义思想。

          • ②要积极投身实现中华民族伟大复兴的中国梦,勇做担当中华民族伟大复兴大任的时代新人。

          • ③要投身新时代中国特色社会主义事业,当代大学生是祖国的未来、民族的希望,要以勇于担当的精神,做走在新时代前列的奋进者、开拓者、奉献者,投身于党和人民在中国特色社会主义新时代的伟大奋斗。

        \ No newline at end of file diff --git a/si-zheng/ma-yuan/ti-gang/index.html b/si-zheng/ma-yuan/ti-gang/index.html index d4cdb1aa..476a5885 100644 --- a/si-zheng/ma-yuan/ti-gang/index.html +++ b/si-zheng/ma-yuan/ti-gang/index.html @@ -1 +1 @@ -马原题纲 - 马克思主义基本原理 - 思政 | Balloon Party = 風船のパーティー = fuusen no party

        # 思维导图

        # 绪论部分

        绪论部分

        # 第一章 世界的物质性及发展规律

        第一章 世界的物质性及发展规律

        # 第二章 实践与认识及其发展规律

        第二章 实践与认识及其发展规律

        # 第三章 人类社会及其发展规律

        第三章 人类社会及其发展规律

        # 第四章 资本主义的本质及规律

        第四章 资本主义的本质及规律

        # 第六章社会主义的发展及其规律(本章仅供学习参考,不作考试要求)

        第六章 社会主义的发展及其规律(本章仅供学习参考,不作考试要求)

        # 第七章 共产主义是人类最崇高的社会理想

        第七章 共产主义是人类最崇高的社会理想

        # 大纲

        # 绪论部分

        # 马克思主义的含义、理论来源、经典著作、鲜明特征;马克思主义基本原理的含义(马克思主义的基本立场、基本观点、基本方法);

        • 含义:马克思主义是由马克思和恩格斯创立的,为他们的后继承者所发展的,以反对资本主义、建设社会主义和实现共产主义为目标的科学理论体系,是关于无产阶级和人类解放的科学。

        • 理论来源: 德国古典哲学、英国古典政治经济学、英法两国的空想社会主义。

        • 经典著作:

          • 两人《神圣家族》 《德意志意识形态》 《共产党宣言》 《新莱茵报》

            • 马《资本论》 《法兰西内战》 《哥达纲领批判》

            • 恩《反杜林论》 《家庭、私有制和国家的起源》 《路德维希・费尔巴哈和德国古典哲学的终结》

        整理出版《资本论》二三卷

        • 鲜明特征:

          • (1)科学性,它是对客观世界特别是人类社会本质和规律的正确反应。(表现在坚持世界的物质性和真理的客观性)

          • (2)革命性, 它是无产阶级和广大人民群众推翻旧世界,建设新世界的理论,按本质来说,它是批判的和革命的,它具有鲜明的无产阶级政治立场(表现在坚持唯物辩证法)。

          • (3)实践性,马克思主义的科学性和革命性是统一的,以社会主义运动的实践为基础,致力于实现无产阶级和广大人民群众的根本利益。

        • 马克思主义基本原理的含义

          • (1)基本立场:是马克思主义观察、分析和解决问题的根本立足点和出发点

          • (2)基本观点:是关于自然、社会和人类思维发展一般规律的科学认识,是对人类思想成果和社会实践经验的科学总结

          • (3)基本方法:是建立在辩证唯物主义和历史唯物主义世界观和方法论基础上的指导我们正确认识世界和改造世界的思想方法和工作方法。

        # 第一章 世界的物质性及发展规律

        # 哲学的基本问题、哲学派别的划分及各派别的基本思想和代表人物(唯物主义 VS 唯心主义;可知论 VS 不可知论;唯物史观 VS 唯心史观;辩证法 VS 形而上学)

        • 哲学的基本问题:存在和思维、物质和精神的关系问题

          • 存在和思维、物质和精神何者是第一性

          • 存在和思维、物质和精神是否具有同一性

        • 哲学派别的划分及各派别的基本思想和代表人物

          • 存在和思维究竟谁是世界的本原,即物质和精神何者是第一性、何者是第二性的问题

            • 唯物主义

            存在决定思维

            • 唯心主义

            思维决定存在

          • 存在和思维是否具有同一性?我们关于我们周围世界的思想对这个世界本身的关系是怎样的?我们的思维能不能认识现实世界?我们能不能在我们关于现实世界的表象和概念中正确的反映现实?

            • 可知论

            认为思维和存在有同一性,人的意识能够认识客观世界及其规律

            • 不可知论

            认为思维和存在没有同一性,人的意识不能或不能彻底认识客观世界及其规律
            古希腊怀疑论者 皮浪,

          • 社会意识和社会存在的关系

            • 唯物史观

            社会存在决定社会意识

            • 唯心史观

            社会意识决定社会存在

          • 是否承认事物的内部矛盾是事物发展的根本动力

            • 辩证法

            承认事物的内部矛盾是事物发展的根本动力

            • 形而上学

            否认事物的内部矛盾是事物发展的根本动力

        # 物质观(物质的含义、意识的含义;物质与意识的辩证关系;主观能动性与客观规律性的统一)、运动观(运动的含义、运动与静止、运动与物质的关系)、时空观(时间的含义及其特点、空间的含义及其特点、时空与物质运动的关系)

        • (一)物质观

          • 物质的含义:物质就是不依赖于人类意识而存在并能为人类的意识所反映的客观存在

          • 意识的含义:意识是人脑的机能和属性,是客观世界的主观映像

          • 物质与意识的辩证关系:

            • (一) 物质决定意识。物质对意识的决定作用表现在意识的起源、本质和作用上。

            • (二) 意识对物质具有反作用。

              • 1. 意识活动具有目的性和计划性;

              • 2. 意识活动具有创造性;

              • 3. 意识具有指导实践改造客观世界的作用;

              • 4. 意识具有控制人的行为和生理活动的作用。

            • (三) 主观能动性和客观规律性的统一

          • 主观能动性与客观规律性的统一:

            • 首先,尊重客观规律是正确发挥主观能动性的前提

            • 其次,只有充分发挥主观能动性,才能正确认识和利用客观规律。

            • 要正确发挥主观能动性,应注意:

              • 一,从实际出发,努力认识和把握事物的发展规律。

              • 二,实践是发挥人的主观能动性的途径。

              • 三,主观能动性的发挥还依赖于一定的物质条件和物质手段

        • (二)运动观

          • 运动的含义:运动是标志一切事物和现象的变化及其过程的哲学范畴

          • 运动与静止:物质世界的运动是绝对的,而物质在运动的过程中又有某种相对的静止。相对静止是物质运动在一定条件下的稳定状态,包括空间的相对位置和事物的根本性质暂时未变这样两种运动的特殊状态。

            • 运动的绝对性体现了物质运动的变动性、无条件性,静止的相对性体现了物质运动的稳定性,有条件性。

            • 运动和静止相互依赖、相互渗透、相互包含,“静中有动,动中有静”。

            • 无条件的绝对运动和有条件的相对静止构成对立统一关系

          • 运动与物质的关系:物质和运动是不可分割的,运动是物质的运动,物质是运动着的物质,离开物质的运动和离开运动的物质都是不可想象的。

        • (三)时空观

          • 时间的含义:是指物质运动的持续性、顺序性

          特点:一维性,即一去不复返

          • 空间的含义:是指物质运动的广延性、伸张性

          特点:三维性

          • 时空与物质运动的关系:

        物质运动总是在一定的时间和空间中进行的,没有离开物质运动的 “纯粹” 时间和空间,也没有离开时间和空间的物质运动。

        # 世界的物质统一性原理

        • (一)意识统一于物质。

        • (二)人类社会也统一于物质。

          • 1. 人类社会依赖于自然界,是整个物质世界的组成部分。

          • 2. 人类谋取物资生活资料的实践活动虽然有意识作指导,但仍然是以物质力量改造物质力量的活动,仍然是物质活动。

          • 3. 物质资料的生产方式是人类社会存在和发展的基础,集中体现着人类社会的物质性。

        # 唯物辩证法的总特征:联系与发展的观点(联系的含义、特点;发展的含义、实质等)

        • 联系的含义:作为一个普遍的哲学范畴,联系是指事物内部各要素之间和事物之间相互影响、相互制约、相互作用的关系。

        • 联系的特点:客观性、普遍性、多样性、条件性

        • 发展的含义:变化的基本趋势是发展,发展是前进的上升的运动

        • 发展的实质:

          • 新事物的产生和旧事物的灭亡。

            • 新事物是指合乎历史前进方向,具有远大前途的东西

            • 旧事物是指丧失历史必然性、日趋灭亡的东西

        # 唯物辩证法的五对范畴(原因与结果、现象与本质、内容与形式、必然性与偶然性、现实性与可能性)

        • 原因与结果:这是揭示事物引起和被引起的一对范畴。

          原因是引起某种现象的现象,结果是被某种现象引起的现象。

          因果关系就是包括时间先后顺序在内的一种现象必然引起另一种现象的普遍联系

        • 现象与本质:这是揭示事物外在联系和内在联系的一对范畴。

          现象是事物的外部联系和表面特征,本质是事物的根本性质和构成要素的内在联系。

          本质是现象的根据,现象是本质的表现

        • 内容与形式:这是从构成要素和表现方式两个方面反映事物的一对范畴。

          内容是指构成事物的一切要素总和,形式是指把内容诸要素统一起来的结构或表现内容的方式

        • 必然性与偶然性:这是揭示事物发生、发展和灭亡的不同趋势的一对范畴。

          必然性是事物联系和发展过程中合乎规律的确定不移的趋势,

          偶然性是联系和发展过程中并非必定如此的、不确定的趋势。

        • 现实性与可能性:这是反映事物过去、现在和将来关系的一对范畴。

          现实性包括内在根据的、合乎必然性的存在,

          可能性是指包含在现实事物中、预示着事物发展前途的种种趋势。

        # 唯物辩证法的三大规律:对立统一规律(矛盾的同一性与斗争性的辩证关系;矛盾的普遍性与特殊性的辩证关系)、质量互变规律(质、量、度等范畴的含义;量变与质变的辩证关系)、否定之否定规律;对立统一规律是唯物辩证法的实质与核心

        • 一、对立统一规律:是事物发展的根本规律,是唯物辩证法的实质与核心。

          ①矛盾是反应事物内部和事物之间对立统一关系的哲学范畴。

          矛盾的同一性和斗争性是相互联结,相辅相成的,没有斗争性就没有同一性,没有同一性也没有斗争性,斗争性寓于同一性之中。

          ②矛盾的普遍性与特殊性是辩证统一的关系,任何现实存在的事物的矛盾都是共性和个性的有机统一,共性寓于个性之中,没有离开个性的共性,也没有离开共性的个性。

        • 二、质量互变规律:

          • ① 事物的联系和发展都采取量变和质变两种状态和形式。

          • ② 质是一事物区别于其他事物的内在规定,量是事物的规模、程度、速度等可以用数量关系表示的规定性。

          度是保持事物质的稳定性的数量界限,即事物的限度、幅度和范围。

          度的两端叫关节点或临界点,超出度的范围,此物就转化为他物。

          “度” 启示我们,在认识和处理问题时要掌握适度的原则。

          • ③ 辩证关系:

            • (1)量变是质变的必要准备

            • (2)质变是量变的必然结果

            • (3)量变和质变是相互渗透的。

        • 三、否定之否定规律:

          • ① 辩证否定观

            • (1)否定是事物的自我否定,是事物内部矛盾运动的结果

            • (2)否定是事物发展的环节,是新事物向旧事物的转变,是旧质到新质的飞跃

            • (3)否定是新旧事物联系的环节

            • (4)辩证否定的实质是 “扬弃” 即新事物对旧事物既批判又继承,既克服其消极因素又保留其积极因素。

          • ② 否定之否定规律揭示了事物发展的前进性与曲折性的统一。

          • ③ 按照否定之否定规律办事,要求我们对待一切事物都要采取科学分析的态度,反对肯定一切和否定一切的形而上学否定观。

        # 如何理解 “以唯物辩证法为指导,不断增强思维能力”P53(五大思维)

        • 一、思维能力包括:战略思维能力、历史思维能力、辩证思维能力、创新思维能力、底线思维能力

        // 战历辩创底

        • 二、战略思维能力,就是高瞻远瞩、统揽全局,善于把握事物发展总体趋势和方向的能力。(树立大局意识,放眼未来。善于观大势,把握工作主权。开阔视野,紧跟时代步伐。)

        • 三、历史思维能力就是以史为鉴、知古鉴今,善于运用历史眼光认识发展规律、把握前进方向、知道现实工作的能力。(善于运用历史智慧推进 ** 的思想观点。加强对中国历史,世界历史的学习,总结历史经验,把握认清规律,更好的走向未来。)

        • 四、辩证思维能力就是承认矛盾,分析矛盾、解决矛盾,善于抓住关键,找准重点,洞察事物发展规律的能力。(坚持 “两点论”, 一分为二的看问题。要勇于进取,稳扎稳打。发展的眼光,全面,普遍联系)

        • 五、创新思维能力就是破除迷信、超越过时的陈规,善于因时制宜、知难而进、开拓创新的能力。(勇于创新,不因循守旧,满足现状。有探索真知、求真务实的态度。百折不挠、勇往直前,不断积累经验。)

        • 六、底线思维能力就是客观地设定最低目标,立足最低点,争取最大期望值的一种积极的思维能力.(基点放在出现较大的锋线上,做好对策,争取最好结果。要居安思危,增强忧患意识。)

        # 第二章 实践与认识及其发展规律

        # 实践观(实践的含义、特点、类型、实践在认识中的决定性作用、实践是检验真理的唯一标准)

        • 一、实践的本质含义:实践是人类能动地改造世界的感性物质活动。

        • 二、实践的基本特征:

          • 第一,实践是改造世界的客观物质活动,具有直接现实性。

          • 第二,实践是人类有意识的活动,体现了自觉能动性。

          • 第三,实践是社会的、历史的活动,具有社会历史性。

        • 三、实践的类型:

          • (1)物质生产实践(最基本的):人类最基本的实践活动,决定着社会的基本性质和面貌。

            • (2)社会政治实践:是改造社会关系的实践活动,形成了复杂的社会政治关系

            • (3)科学文化实践:是创造精神文化产品的实践活动。

        • 四、实践在认识中的决定作用:

          • (1)辩证唯物主义认识论认为,实践决定认识,实践是认识的基础。

          • (2)实践是认识的来源

          • (3)实践是认识发展的动力

          • (4)实践是认识的目的

          • (5)实践是检验认识真理性的唯一标准。

        # 辩证唯物主义和旧唯物主义对认识本质的不同回答;感性认识和理性认识及其辩证关系、非理性因素;认识运动的基本规律

        • Ⅰ. 辩证唯物主义和旧唯物主义对认识本质的不同回答:

          • ② 辩证唯物主义和旧唯物主义都坚持反映论,但是两者之间又有着本质的区别

          • ② 旧唯物主义认识论以感性直观为基础,把人的认识看成是消极地、被动地反映和接受外界对象,不能把认识看作一个不断发展的过程,认为认识是一次性完成的。这种直观的,消极被动的反映论是不科学的

          • ③ 辩证唯物主义认识论继承了旧唯物主义反映论的合理前提,同时又克服了它的严重缺陷。它把实践的观点引入认识论,把辩证法应用于反映论,把认识看成是一个由不知到知、由浅入深的充满矛盾的能动的认识过程,是能动的反映论。

        • Ⅱ. 感性认识和理性认识及其辩证关系、非理性因素:

          • ① 感性认识:是认识的低级阶段,是人在实践中通过感官对事物外部形态的直接的、具体的反映。它包括感觉、知觉、表象三种形式。它的突出特点是直接现实性。它只是对事物表面现象的反映,没有反映事物的本质,因而具体局限性。

          • ② 理性认识:人们通过抽象思维得到的对事物本质、内在联系和规律性的认识,是认识的高级阶段,包括概念、判断和推理三种形式。其特点是间接性和抽象性。

          • ③ 感性认识和理性认识的辩证关系:

            • (1)感性认识是理性认识的基础,理性认识依赖于感性认识。

            • (2)感性认识有待于发展和深化为理性认识。

            • (3)感性认识和感性认识相互渗透、相互包含。

            • ④ 从感性认识上升到理性认识需要具备的条件:

              • (1)勇于实践、深入调查,获取丰富的感性材料,这是基础。

              • (2)必须经过理性思考,将感性材料加以去粗取精、去伪存真、由此及彼、由表及里加工,才能将感性认识上升为理性认识。

            • ⑤ 非理性因素:主要是指认识主体的情感和意志。也包括不自觉、非逻辑等特点的认识形式(如:联想、想象、猜测、直觉、顿悟、灵感等)

              • (1)人的认识过程是理性因素和非理性因素协同作用的结果。

              • (2)非理性因素对于人的认识能力和人是活动具有激活,驱动和控制作用我们应当在理性认识的指导下,充分发挥非理性因素的积极作用)

        • Ⅲ. 认识运动的基本规律:

          • (1)认识运动的辩证过程,是从实践到认识的过程,即在实践基础上由感性认识能动的飞跃到理性认识的过程。

          • (2)认识过程的第二个阶段是从认识到实践,即由 “精神到物质的阶段,由思想到存在的阶段”。

          • (3)认识运动是不断反复和无限发展的,是一个波浪式前进和螺旋式上升的过程。

        # 真理的含义及其属性;价值评价的含义

        • 真理的定义

        真理是人们对于客观事物及其规律的正确认识。

        • 真理的属性

          • (1) 真理的客观性

            真理具有客观性,凡真理都是客观真理。真理的客观性或客观真理有两层含义:一是指真理的内容是客观的,是不以人的意志为转移的客观世界;二是指检验真理的标准是客观的。检验真理的唯一标准是实践,而实践是物质性的活动。

            • (2) 真理的绝对性

            真理的绝对性或者绝对真理是指真理的内容具有正确性,真理的发展具有无限性。其具体含义有:其一,就真理的内容而言,任何真理都必然包含着同客观对象相符合的客观内容,都同谬误有原则的界限。因此,承认了客观真理就是承认了绝对真理。其二,就真理的发展而言,其前景是无限的。由于人类依其认识本性来说,能够正确认识无限发展着的物质世界。所以,承认世界的可知性,也就是承认了绝对真理。其三,就真理发展的最终结果而言,无数相对真理的总和构成为绝对真理。所以,承认认识发展的无限性,就必然承认绝对真理。

            • (3) 真理的相对性

            真理的相对性是指真理的有限性、有条件性,是指人们在一定条件下对世界的正确把握具有近似性。其含义有三层:其一,从认识的广度上看,任何真理只能是对无限的物质世界的某一阶段、某一方面、某一领域的认识。其二,从认识的深度上看,任何真理对特定事物的一定程度、一定层次的近似正确的反映。其三,从认识的进程看,任何真理只是对事物的一定发展阶段的正确认识。

        • 价值评价的定义:价值评价即对主体和客体的价值关系进行评判,是一种关于价值现象的评价性的认识活动。

        # 第三章 人类社会及其发展规律

        # 社会存在与社会意识的含义及其辩证关系(社会存在的含义,社会存在的要素及其对人类社会发展的作用;社会意识的含义及分类;社会存在与社会意识的辩证关系、社会意识相对独立性的表现)

        • 社会存在的含义:

        社会存在是指社会生活的物质方面,是社会实践和物质生活条件的总和。

        • 社会存在的要素:物质生产方式、自然地理环境和人口因素。

          • Ⅰ. 生产方式是社会历史发展的决定力量。

            • 其一,物质生产活动及生产方式是人类社会赖以存在和发展的基础,是人类其他一切活动的首要前提。

            • 其二,物质生产活动及生产方式决定着社会的结构、性质和面貌,制约着人们的经济生活、政治生活和精神生活等全部社会生活。

            • 其三,物质生产活动及生产方式的变化发展决定整个社会历史的变化发展,决定社会形态从低级向高级的更替和发展。

          • Ⅱ. 自然地理环境是人类社会生存和发展的必要和永恒的条件。是人们生活和生产的自然基础。自然地理环境提供了社会生活和生产资料的来源。

          • Ⅲ. 人口因素也是重要的社会物质生活条件,对社会发展起着制约和影响的作用。人是从事物质生产活动和其他一切社会活动的主体,是一切社会关系的承担者。

        • 社会意识的含义:社会意识是社会生活的精神方面,是社会存在的反映。

        • 社会意识的构成
          社会意识具有复杂的结构,可以按照不同的标准进行划分。

          • 根据社会意识的主体,可以分为个体意识、群体意识;

          • 根据其发展水平,可以分为

            • 低级的自发的社会心理

            • 高级的社会意识形式

              • 作为上层建筑的意识形式,又称为社会意识形态、观念上层建筑
                主要包括政治法律思想、道德、艺术、宗教、哲学等。
                在阶级社会中,社会意识形态本质上是占统治阶级的思想文化,反映经济基础,具有鲜明的阶级性。

              • 非上层建筑的意识形式
                包括语言学、逻辑学以及自然科学等。
                它不具有阶级性。

        • 社会存在和社会意识的关系

          社会存在和社会意识是辩证统一的关系。社会存在决定社会意识,社会意识是社会存在的反映,并反作用于社会存在。

          • (1) 社会存在对社会意识的决定作用表现在:

            • 其一,社会存在是社会意识内容的客观来源,社会意识是社会物质生活过程及其条件的主观反映。这是社会意识对社会存在的依赖性。

            • 其二,社会存在决定社会意识的性质。

            • 其三,随着社会存在的发展,社会意识也相应地或迟或早地发生变化和发展。

          • (2) 社会意识有其相对独立性,对社会存在具有反作用。其具体表现有:

            • 其一,社会意识具有相对独立性,它与社会存在之间存在发展的不平衡性、不同步性。

            • 其二,社会意识内部各种形式之间的相互影响及各自具有的历史继承性。

            • 其三,社会意识对社会存在的能动的反作用。这是社会意识相对独立性的突出表现。先进的社会意识,反映了社会发展的客观规律,可以对社会发展起积极的促进作用;而落后的社会意识,对社会发展起着阻碍的作用。

          • (3) 社会意识起作用的形式

        社会意识的能动作用是通过指导人们的实践活动实现的。思想本身并不能实现什么,要实现思想就要诉诸实践。而社会实践的主体是人民群众。因此,一种社会意识发挥作用的程度及范围大小、时间久暂,同它实际掌握群众的深度和广度密切联系在一起。

        # 生产力与生产关系的含义及其辩证关系;经济基础与上层建筑的含义及其辩证关系

        • 生产力的含义:是人类在生产事件中形成的改造和影响自然以使其适合社会需要的物质力量。

        • 生产关系的含义:生产关系是人们在物质生产过程中形成的不以人的意志为转移的经济关系。

        • 生产力与生产关系的关系

          生产力和生产关系是社会生产不可分割的两个方面。在社会生产中,生产力是生产的物质内容,生产关系是生产的社会形式,二者的有机结合和统一,构成社会的生产方式。

          • Ⅰ. 生产力决定生产关系:

            • (1) 生产力状况决定生产关系的性质。

            • (2) 生产力的发展决定生产关系的变化。

          • Ⅱ. 生产关系对生产力具有能动的反作用。其主要表现为:当生产关系适合生产力发展的客观要求时,它对生产力的发展起推动作用;当生产关系不适合生产力发展的客观要求时,它就会阻碍生产力的发展。

        • 经济基础的含义:经济基础是指由社会一定发展阶段的生产力所决定的生产关系的总和。其实质是社会一定发展阶段上的基本经济制度,是制度化的物质社会关系。其表现形式是经济体制,即社会基本经济制度所采取的组织形式和管理方式。

        • 上层建筑的含义:上层建筑是建立在一定经济基础之上的意识形态以及相应的制度、组织和设施。

        • 经济基础与上层建筑的关系 辩证统一

          • (1) 经济基础决定上层建筑。具体表现有:

            • 第一,经济基础的需要决定了上层建筑的产生;

            • 第二,经济基础的性质决定上层建筑的性质,有什么样的经济基础就有什么样的上层建筑;

            • 第三,经济基础的变更必然引起上层建筑的变革,并决定着其变革的方向。

          • (2) 上层建筑对经济基础具有反作用。

        这种反作用集中表现在,上层建筑保护和促进自己的经济基础的形成巩固和发展;同时反对和排除与自己性质不一致的生产关系。

        # 社会基本矛盾是人类社会发展的根本动力;人类社会发展动力学说

        • Ⅰ. 社会基本矛盾是人类社会发展的根本动力

          • 含义:社会基本矛盾就是指贯穿社会发展始终,规定社会发展过程的基本性质和基本趋势,并对社会历史发展起根本推动作用的矛盾。生产力和生产关系、经济基础和上层建筑的矛盾是社会的基本矛盾。

          • 作用:

            • (1)生产力是社会基本矛盾运动中最基本的动力因素,是人类社会发展和进步的最终决定力量。生产力是社会存在和发展的物质基础,是不能任意选择的物质力量和历史活动的前提。

            • (2)生产力是社会进步的根本内容,是衡量社会进步的根本尺度。人类社会是在生产力与生产关系的矛盾运动中前进的。

            • (3)社会基本矛盾特别是生产力和生产关系的矛盾,决定着社会中其他矛盾的存在和发展。

            • (4)经济基础和上层建筑的矛盾也会影响和制约生产力和生产关系的矛盾。

            • (5)社会基本矛盾具有不同的表现形式,并从根本上影响和促进社会形态的变化和发展。

        • Ⅱ. 人类社会发展动力学

          • (1)阶级斗争是社会基本矛盾在阶级社会中表现,是阶级社会发展的直接动力。离开了阶级斗争,就无法理解阶级社会发展。“没有对抗就没有进步。这是文明直到今天所遵循的规律。”

          • (2)社会革命是阶级斗争发展到一定阶级的产物,是推动社会发展的重要动力。

        # 第四章 资本主义的本质及规律

        # 商品经济的含义、产生条件、特点

        • 含义:是以交换为目的而进行生产的经济形式,它是一定社会历史条件产物。

        • 产生条件:

          • (1)存在社会分工

          • (2)生产资料和劳动产品属于不同的所有者。

        • 特点

          • (1)商品经济本质上是交换经济,在商品经济条件下,生产要素和消费资料的全部或大部分都要通过市场交换来获得,商品生产者以追求价值为目的,并通过市场交换来实现。

          • (2)商品经济是开放型经济,商品经济以社会分工为基础,强调生产过程中的分工与协作,人与人之间,商品生产者之间以及生产单位之间的经济联系随着商品经济的发展而日益紧密,范围也日益扩大。

          • (3)商品经济是开拓进取型经济。

          • (4)商品经济以扩大再生产为特征。商品生产者为获得更多的价值实现和在竞争中处于有利地位,必然不断增加投入,改进技术和改善经营管理,从而使生产规模不断扩大。

        # 商品的含义、商品二因素及其辩证关系、商品的价值量

        • 含义:是用来交换,能满足人们某种需要的劳动产品,具有使用价值和价值两种因素或者两种属性,是使用价值和价值的统一体。

        • 商品的二因素及其辩证关系

          • 1. 商品的二因素:使用价值和价值。

            • 关于使用价值:

              • (1)使用价值反映的是人与自然之间的物质关系,是商品的自然属性,是一切劳动产品所共有的属性,使用价值由商品本身的自然属性决定,离开了商品它就不复存在。

              • (2)使用价值是交换价值的物质承担者。

            • 关于价值:

              • (1)价值是凝结在商品中无差别的一般人类劳动,即人类脑力和体力的耗费。

              • (2)价值是商品所特有地位社会属性。

              • (3)价值是交换价值的基础,交换价值是价值的表现形式。

          • 2. 辩证关系

            • (1)商品的价值和使用价值之间是对立统一的关系。

            • (2)其对立表现在:商品的使用价值和价值是相互排斥的,二者不可兼得。要获得商品的价值,就必须放弃商品的使用价值;要得到商品的使用价值,就不能得到商品的价值。

            • (3)其统一性表现在:作为商品,必须同时具有使用价值和价值两个因素。使用价值是价值的物质承担者,价值寓于使用价值之中。一种物品尽管具有使用价值,但如果不是劳动产品,也没有价值。

        • 商品的价值量

          • (1)首先,价值量是由劳动者生产商品所耗费的劳动量决定的,而劳动量则是按照劳动时间来计量。决定商品价值量的不是生产商品的个别劳动时间,而只能是社会必要劳动时间。生产商品所需要的社会必要劳动时间随着劳动生产力的变化而变化。

          • (2)商品价值量与生产商所耗费的劳动时间成正比,与劳动生产力成反比。

          • (3)商品的价值量同简单劳动与复杂劳动有密切饿关系。简单劳动书指不需要经过专门训练和培养的一般劳动者能从事的劳动。复杂劳动是指需要经过专门的训练和培养,具有一定的文化知识和技术专长的劳动者所从事的劳动。

        # 劳动二重性

        • 生产商品的劳动可以区分为具体劳动和抽象劳动

          • 具体劳动是指生产一定使用价值的具体形式的劳动;

          • 抽象劳动是指撇开一切具有形式的、无差别的一般人类劳动,即人的体力和脑力消耗。

          • 具体劳动形成商品的使用价值,抽象劳动形成商品的价值实体。

        • 劳动的二重性决定了商品的二因素。具体劳动和抽象劳动是对立统一的:

          • 一方面,具体劳动和抽象劳动在时间上和空间上是统一的,是商品生产者的同一劳动过程的不可分割的两方面;

          • 另一方面,具体劳动所反映的是人与自然的关系,是劳动的自然属性,抽象劳动所反映的是商品生产者的社会关系,是劳动的社会属性。

        # 价值规律的内容、表现形式、作用

        • 价值规律的基本内容:

        • ①商品的价值量是由生产这种商品的社会必要劳动时间决定的。

        • ②商品交换要以价值量为基础,按照等价交换的原则进行。

        • 价值规律的表现形式:
          商品的价格围绕商品的价值上下波动。

        • 价值规律的作用:

          • ①自发地调节生产资料和劳动力在社会各生产部门之间的分配比例。

          • ②自发地刺激社会生产力的发展。

          • ③自发地调节社会收入的分配。

        # 劳动力商品的特点

        • 劳动力商品的使用价值是价值的源泉

        • 劳动力成为商品的基本条件:

          • 第一、劳动者是自由人,能够把自己的劳动力当做自己的商品来支配;

          • 第二、劳动者自由的一无所有,没有任何实现自己的劳动力所必需的物质条件。

        • 劳动力的价值,是由生产、发展、维持和延续劳动力所必需的生活必需品的价值决定的。
          它包括三个部分:

          • ① 是维持劳动者自身生存所必需的生活资料的价值,用以再生产他的劳动力。

          • ② 是劳动者家属的生存必需的生活资料的价值,用以延续劳动力的供给。

          • ③ 劳动者接受教育和训练所支出的费用,用以培训适合资本主义再生产所需要的劳动力。

        # 资本主义生产的直接目的;绝对剩余价值、相对剩余价值

        • 资本主义生产的直接目的是获取更多剩余价值。

        • 资本主义生产是以雇佣劳动为基础,以获取剩余价值为目的的生产。

        • 其生产过程具有二重性:一方面是生产使用价值的劳动过程;另一方面又是生产剩余价值的价值增殖过程。

        • 绝对剩余价值是指在必要劳动时间不变的条件下,由于延长工作日的长度和提高劳动强度而生产的剩余价值。

        • 相对剩余价值是指在工作日长度不变的条件下,通过缩短必要劳动时间而相对延长剩余劳动时间所生产的剩余价值

        # 资本主义扩大再生产的途径

        • 第一,第 Ⅰ 部类原有可变资本加上追加的可变资本,再加上本部类资本家用于个人消费的剩余价值之和,必须等于第 Ⅱ 部类的不变资本加上追加的不变资本,即
          Ⅰ(v+△v+m/x)=Ⅱ(c+△c)

        • 第二,第 Ⅰ 部类全部产品的价值必须等于两大部类原有的不变资本加上追加的不变资本,即
          Ⅰ(c+v+m)=Ⅰ(c+△c)+Ⅱ(c+△c)

        • 第三,第 Ⅱ 部类全部产品的价值必须等于两大部类原有的可变资本加上追加的可变资本,再加上资本家用于个人消费的剩余价值之和,即
          Ⅱ(c+v+m)=Ⅰ(v+△v+m/x)+Ⅱ(v+△v+m/x)

        # 资本主义基本矛盾、资本主义经济危机的实质

        • 基本矛盾:生产社会化和生产资料资本主义私人占有之间的矛盾。

        • 资本主义经济危机的实质:相对过剩的危机,即生产和销售矛盾造成的相对过剩。

        # 资本主义选举制度

        • 是资产阶级制定某种原则和程序,通过竞选产生议会和国家元首的一种政治机制。

        • 在资本主义国家中,选举已经成为国家政治制度运行中对社会发展和稳定产生举足轻重影响的一个不可或缺的政治机制。

        # 第五章 资本主义的发展及其趋势

        # 第六章 社会主义的发展及其规律(本章仅供学习参考,不作考试要求)

        # 科学社会主义的基本原则

        • 第一,在生产资料公有制基础上组织生产,满足全体社会成员的需要是社会主义生产的根本目的。

        • 第二,对社会生产进行有计划的指导和调节,实行等量劳动领取等量产品的按劳分配原则。

        • 第三,合乎自然规律地改造和利用自然。

        • 第四,无产阶级革命是无产阶级进行斗争的最高形式,必须以建立无产阶级专政的国家为目的。

        • 第五,社会主义事业必须坚持无产阶级政党的领导。

        • 第六,通过无产阶级专政和社会主义高度发展,最终实现向消灭阶级、消灭剥削,实现人的全面而自由发展的共产主义的过渡。

        • 一方面,科学社会主义基本原则任何时候都不能违背;另一方面,科学社会主义基本原则在实践中不能僵化地教条地对待它。

        # 经济文化相对落后国社会主义革命理论

        经济文化相对落后的国家社会主义建设的艰巨性和长期性

        • 第一、生产力发展状态的约束。社会主义国家在经济上落后于发达的资本主义国家,有的甚至落后很远,社会主义制度下必须把大力发展生产力作为根本任务,努力完成生产社会化、商品化、现代化的艰巨任务。

        • 第二、经济基础和上层建筑发展状况的制约。发展公有经济,改造小农经济,建立、巩固和完善社会主义的经济基础,就成为无产阶级政权的一项十分巨大的任务。社会主义民主政治建设受到这些国家的经济、政治、文化条件的严重制约,在这一过程中,还要进一步消灭阶级和阶级差别,改变几千年形成的传统观念,建设社会主义的先进文化,实现真正的社会公正的平等。

        • 第三,国际环境的严峻挑战。如果说在社会主义国家成立之初,国际资本主义对资本主义的进攻主要是武力方式,那么在社会主义建设取得重大成就后,进攻方式则往往转变为一和平演变为主。一是通过军事、政治压力盒有限的经济、科技的合作和援助,迫使社会主义国家屈从其经济政治。二是通过文化渗透,在社会主义国家内部制造经济、政治、思想等种种混乱,阻挠和破坏社会主义国家的发展。

        • 第四,马克思主义执政党和对社会主义发展的道路的探索和对社会主义建设规律的认识,需要一个长期的艰苦的

        # 第七章 共产主义是人类最崇高的社会理想

        # 实现共产主义是一个长期的历史过程(辨析题)p286

        • (1)理论上,马克思主义所揭示的社会形态发展与更替的规律是一般的历史规律,是只有在漫长的历史过程中才能体现出来的规律。

        • (2)每一个社会形态的产生发展,都会经历一个很长的历史时期,而旧的社会形态走向没落并为新的社会形态所代替也是一个长期的历史过程。

        • (3)从资本主义到共产主义的转变是一种根本的转变,它不是具体制度的更新,而是整个社会的根本改造,因而必然是个长期而艰难的过程。

        # 共产主义的基本特征 p275

        • (1)物质财富极大丰富,消费资料按需分配。

        • (2)社会关系高度和谐,人们精神境界极大提高。

        • (3)每个人自由而全面的发展,人类从必然王国向自由王国飞跃。

        \ No newline at end of file +马原题纲 - 马克思主义基本原理 - 思政 | Balloon Party = 風船のパーティー = fuusen no party

        # 思维导图

        # 绪论部分

        绪论部分

        # 第一章 世界的物质性及发展规律

        第一章 世界的物质性及发展规律

        # 第二章 实践与认识及其发展规律

        第二章 实践与认识及其发展规律

        # 第三章 人类社会及其发展规律

        第三章 人类社会及其发展规律

        # 第四章 资本主义的本质及规律

        第四章 资本主义的本质及规律

        # 第六章社会主义的发展及其规律(本章仅供学习参考,不作考试要求)

        第六章 社会主义的发展及其规律(本章仅供学习参考,不作考试要求)

        # 第七章 共产主义是人类最崇高的社会理想

        第七章 共产主义是人类最崇高的社会理想

        # 大纲

        # 绪论部分

        # 马克思主义的含义、理论来源、经典著作、鲜明特征;马克思主义基本原理的含义(马克思主义的基本立场、基本观点、基本方法);

        • 含义:马克思主义是由马克思和恩格斯创立的,为他们的后继承者所发展的,以反对资本主义、建设社会主义和实现共产主义为目标的科学理论体系,是关于无产阶级和人类解放的科学。

        • 理论来源: 德国古典哲学、英国古典政治经济学、英法两国的空想社会主义。

        • 经典著作:

          • 两人《神圣家族》 《德意志意识形态》 《共产党宣言》 《新莱茵报》

            • 马《资本论》 《法兰西内战》 《哥达纲领批判》

            • 恩《反杜林论》 《家庭、私有制和国家的起源》 《路德维希・费尔巴哈和德国古典哲学的终结》

        整理出版《资本论》二三卷

        • 鲜明特征:

          • (1)科学性,它是对客观世界特别是人类社会本质和规律的正确反应。(表现在坚持世界的物质性和真理的客观性)

          • (2)革命性, 它是无产阶级和广大人民群众推翻旧世界,建设新世界的理论,按本质来说,它是批判的和革命的,它具有鲜明的无产阶级政治立场(表现在坚持唯物辩证法)。

          • (3)实践性,马克思主义的科学性和革命性是统一的,以社会主义运动的实践为基础,致力于实现无产阶级和广大人民群众的根本利益。

        • 马克思主义基本原理的含义

          • (1)基本立场:是马克思主义观察、分析和解决问题的根本立足点和出发点

          • (2)基本观点:是关于自然、社会和人类思维发展一般规律的科学认识,是对人类思想成果和社会实践经验的科学总结

          • (3)基本方法:是建立在辩证唯物主义和历史唯物主义世界观和方法论基础上的指导我们正确认识世界和改造世界的思想方法和工作方法。

        # 第一章 世界的物质性及发展规律

        # 哲学的基本问题、哲学派别的划分及各派别的基本思想和代表人物(唯物主义 VS 唯心主义;可知论 VS 不可知论;唯物史观 VS 唯心史观;辩证法 VS 形而上学)

        • 哲学的基本问题:存在和思维、物质和精神的关系问题

          • 存在和思维、物质和精神何者是第一性

          • 存在和思维、物质和精神是否具有同一性

        • 哲学派别的划分及各派别的基本思想和代表人物

          • 存在和思维究竟谁是世界的本原,即物质和精神何者是第一性、何者是第二性的问题

            • 唯物主义

            存在决定思维

            • 唯心主义

            思维决定存在

          • 存在和思维是否具有同一性?我们关于我们周围世界的思想对这个世界本身的关系是怎样的?我们的思维能不能认识现实世界?我们能不能在我们关于现实世界的表象和概念中正确的反映现实?

            • 可知论

            认为思维和存在有同一性,人的意识能够认识客观世界及其规律

            • 不可知论

            认为思维和存在没有同一性,人的意识不能或不能彻底认识客观世界及其规律
            古希腊怀疑论者 皮浪,

          • 社会意识和社会存在的关系

            • 唯物史观

            社会存在决定社会意识

            • 唯心史观

            社会意识决定社会存在

          • 是否承认事物的内部矛盾是事物发展的根本动力

            • 辩证法

            承认事物的内部矛盾是事物发展的根本动力

            • 形而上学

            否认事物的内部矛盾是事物发展的根本动力

        # 物质观(物质的含义、意识的含义;物质与意识的辩证关系;主观能动性与客观规律性的统一)、运动观(运动的含义、运动与静止、运动与物质的关系)、时空观(时间的含义及其特点、空间的含义及其特点、时空与物质运动的关系)

        • (一)物质观

          • 物质的含义:物质就是不依赖于人类意识而存在并能为人类的意识所反映的客观存在

          • 意识的含义:意识是人脑的机能和属性,是客观世界的主观映像

          • 物质与意识的辩证关系:

            • (一) 物质决定意识。物质对意识的决定作用表现在意识的起源、本质和作用上。

            • (二) 意识对物质具有反作用。

              • 1. 意识活动具有目的性和计划性;

              • 2. 意识活动具有创造性;

              • 3. 意识具有指导实践改造客观世界的作用;

              • 4. 意识具有控制人的行为和生理活动的作用。

            • (三) 主观能动性和客观规律性的统一

          • 主观能动性与客观规律性的统一:

            • 首先,尊重客观规律是正确发挥主观能动性的前提

            • 其次,只有充分发挥主观能动性,才能正确认识和利用客观规律。

            • 要正确发挥主观能动性,应注意:

              • 一,从实际出发,努力认识和把握事物的发展规律。

              • 二,实践是发挥人的主观能动性的途径。

              • 三,主观能动性的发挥还依赖于一定的物质条件和物质手段

        • (二)运动观

          • 运动的含义:运动是标志一切事物和现象的变化及其过程的哲学范畴

          • 运动与静止:物质世界的运动是绝对的,而物质在运动的过程中又有某种相对的静止。相对静止是物质运动在一定条件下的稳定状态,包括空间的相对位置和事物的根本性质暂时未变这样两种运动的特殊状态。

            • 运动的绝对性体现了物质运动的变动性、无条件性,静止的相对性体现了物质运动的稳定性,有条件性。

            • 运动和静止相互依赖、相互渗透、相互包含,“静中有动,动中有静”。

            • 无条件的绝对运动和有条件的相对静止构成对立统一关系

          • 运动与物质的关系:物质和运动是不可分割的,运动是物质的运动,物质是运动着的物质,离开物质的运动和离开运动的物质都是不可想象的。

        • (三)时空观

          • 时间的含义:是指物质运动的持续性、顺序性

          特点:一维性,即一去不复返

          • 空间的含义:是指物质运动的广延性、伸张性

          特点:三维性

          • 时空与物质运动的关系:

        物质运动总是在一定的时间和空间中进行的,没有离开物质运动的 “纯粹” 时间和空间,也没有离开时间和空间的物质运动。

        # 世界的物质统一性原理

        • (一)意识统一于物质。

        • (二)人类社会也统一于物质。

          • 1. 人类社会依赖于自然界,是整个物质世界的组成部分。

          • 2. 人类谋取物资生活资料的实践活动虽然有意识作指导,但仍然是以物质力量改造物质力量的活动,仍然是物质活动。

          • 3. 物质资料的生产方式是人类社会存在和发展的基础,集中体现着人类社会的物质性。

        # 唯物辩证法的总特征:联系与发展的观点(联系的含义、特点;发展的含义、实质等)

        • 联系的含义:作为一个普遍的哲学范畴,联系是指事物内部各要素之间和事物之间相互影响、相互制约、相互作用的关系。

        • 联系的特点:客观性、普遍性、多样性、条件性

        • 发展的含义:变化的基本趋势是发展,发展是前进的上升的运动

        • 发展的实质:

          • 新事物的产生和旧事物的灭亡。

            • 新事物是指合乎历史前进方向,具有远大前途的东西

            • 旧事物是指丧失历史必然性、日趋灭亡的东西

        # 唯物辩证法的五对范畴(原因与结果、现象与本质、内容与形式、必然性与偶然性、现实性与可能性)

        • 原因与结果:这是揭示事物引起和被引起的一对范畴。

          原因是引起某种现象的现象,结果是被某种现象引起的现象。

          因果关系就是包括时间先后顺序在内的一种现象必然引起另一种现象的普遍联系

        • 现象与本质:这是揭示事物外在联系和内在联系的一对范畴。

          现象是事物的外部联系和表面特征,本质是事物的根本性质和构成要素的内在联系。

          本质是现象的根据,现象是本质的表现

        • 内容与形式:这是从构成要素和表现方式两个方面反映事物的一对范畴。

          内容是指构成事物的一切要素总和,形式是指把内容诸要素统一起来的结构或表现内容的方式

        • 必然性与偶然性:这是揭示事物发生、发展和灭亡的不同趋势的一对范畴。

          必然性是事物联系和发展过程中合乎规律的确定不移的趋势,

          偶然性是联系和发展过程中并非必定如此的、不确定的趋势。

        • 现实性与可能性:这是反映事物过去、现在和将来关系的一对范畴。

          现实性包括内在根据的、合乎必然性的存在,

          可能性是指包含在现实事物中、预示着事物发展前途的种种趋势。

        # 唯物辩证法的三大规律:对立统一规律(矛盾的同一性与斗争性的辩证关系;矛盾的普遍性与特殊性的辩证关系)、质量互变规律(质、量、度等范畴的含义;量变与质变的辩证关系)、否定之否定规律;对立统一规律是唯物辩证法的实质与核心

        • 一、对立统一规律:是事物发展的根本规律,是唯物辩证法的实质与核心。

          ①矛盾是反应事物内部和事物之间对立统一关系的哲学范畴。

          矛盾的同一性和斗争性是相互联结,相辅相成的,没有斗争性就没有同一性,没有同一性也没有斗争性,斗争性寓于同一性之中。

          ②矛盾的普遍性与特殊性是辩证统一的关系,任何现实存在的事物的矛盾都是共性和个性的有机统一,共性寓于个性之中,没有离开个性的共性,也没有离开共性的个性。

        • 二、质量互变规律:

          • ① 事物的联系和发展都采取量变和质变两种状态和形式。

          • ② 质是一事物区别于其他事物的内在规定,量是事物的规模、程度、速度等可以用数量关系表示的规定性。

          度是保持事物质的稳定性的数量界限,即事物的限度、幅度和范围。

          度的两端叫关节点或临界点,超出度的范围,此物就转化为他物。

          “度” 启示我们,在认识和处理问题时要掌握适度的原则。

          • ③ 辩证关系:

            • (1)量变是质变的必要准备

            • (2)质变是量变的必然结果

            • (3)量变和质变是相互渗透的。

        • 三、否定之否定规律:

          • ① 辩证否定观

            • (1)否定是事物的自我否定,是事物内部矛盾运动的结果

            • (2)否定是事物发展的环节,是新事物向旧事物的转变,是旧质到新质的飞跃

            • (3)否定是新旧事物联系的环节

            • (4)辩证否定的实质是 “扬弃” 即新事物对旧事物既批判又继承,既克服其消极因素又保留其积极因素。

          • ② 否定之否定规律揭示了事物发展的前进性与曲折性的统一。

          • ③ 按照否定之否定规律办事,要求我们对待一切事物都要采取科学分析的态度,反对肯定一切和否定一切的形而上学否定观。

        # 如何理解 “以唯物辩证法为指导,不断增强思维能力”P53(五大思维)

        • 一、思维能力包括:战略思维能力、历史思维能力、辩证思维能力、创新思维能力、底线思维能力

        // 战历辩创底

        • 二、战略思维能力,就是高瞻远瞩、统揽全局,善于把握事物发展总体趋势和方向的能力。(树立大局意识,放眼未来。善于观大势,把握工作主权。开阔视野,紧跟时代步伐。)

        • 三、历史思维能力就是以史为鉴、知古鉴今,善于运用历史眼光认识发展规律、把握前进方向、知道现实工作的能力。(善于运用历史智慧推进 ** 的思想观点。加强对中国历史,世界历史的学习,总结历史经验,把握认清规律,更好的走向未来。)

        • 四、辩证思维能力就是承认矛盾,分析矛盾、解决矛盾,善于抓住关键,找准重点,洞察事物发展规律的能力。(坚持 “两点论”, 一分为二的看问题。要勇于进取,稳扎稳打。发展的眼光,全面,普遍联系)

        • 五、创新思维能力就是破除迷信、超越过时的陈规,善于因时制宜、知难而进、开拓创新的能力。(勇于创新,不因循守旧,满足现状。有探索真知、求真务实的态度。百折不挠、勇往直前,不断积累经验。)

        • 六、底线思维能力就是客观地设定最低目标,立足最低点,争取最大期望值的一种积极的思维能力.(基点放在出现较大的锋线上,做好对策,争取最好结果。要居安思危,增强忧患意识。)

        # 第二章 实践与认识及其发展规律

        # 实践观(实践的含义、特点、类型、实践在认识中的决定性作用、实践是检验真理的唯一标准)

        • 一、实践的本质含义:实践是人类能动地改造世界的感性物质活动。

        • 二、实践的基本特征:

          • 第一,实践是改造世界的客观物质活动,具有直接现实性。

          • 第二,实践是人类有意识的活动,体现了自觉能动性。

          • 第三,实践是社会的、历史的活动,具有社会历史性。

        • 三、实践的类型:

          • (1)物质生产实践(最基本的):人类最基本的实践活动,决定着社会的基本性质和面貌。

            • (2)社会政治实践:是改造社会关系的实践活动,形成了复杂的社会政治关系

            • (3)科学文化实践:是创造精神文化产品的实践活动。

        • 四、实践在认识中的决定作用:

          • (1)辩证唯物主义认识论认为,实践决定认识,实践是认识的基础。

          • (2)实践是认识的来源

          • (3)实践是认识发展的动力

          • (4)实践是认识的目的

          • (5)实践是检验认识真理性的唯一标准。

        # 辩证唯物主义和旧唯物主义对认识本质的不同回答;感性认识和理性认识及其辩证关系、非理性因素;认识运动的基本规律

        • Ⅰ. 辩证唯物主义和旧唯物主义对认识本质的不同回答:

          • ② 辩证唯物主义和旧唯物主义都坚持反映论,但是两者之间又有着本质的区别

          • ② 旧唯物主义认识论以感性直观为基础,把人的认识看成是消极地、被动地反映和接受外界对象,不能把认识看作一个不断发展的过程,认为认识是一次性完成的。这种直观的,消极被动的反映论是不科学的

          • ③ 辩证唯物主义认识论继承了旧唯物主义反映论的合理前提,同时又克服了它的严重缺陷。它把实践的观点引入认识论,把辩证法应用于反映论,把认识看成是一个由不知到知、由浅入深的充满矛盾的能动的认识过程,是能动的反映论。

        • Ⅱ. 感性认识和理性认识及其辩证关系、非理性因素:

          • ① 感性认识:是认识的低级阶段,是人在实践中通过感官对事物外部形态的直接的、具体的反映。它包括感觉、知觉、表象三种形式。它的突出特点是直接现实性。它只是对事物表面现象的反映,没有反映事物的本质,因而具体局限性。

          • ② 理性认识:人们通过抽象思维得到的对事物本质、内在联系和规律性的认识,是认识的高级阶段,包括概念、判断和推理三种形式。其特点是间接性和抽象性。

          • ③ 感性认识和理性认识的辩证关系:

            • (1)感性认识是理性认识的基础,理性认识依赖于感性认识。

            • (2)感性认识有待于发展和深化为理性认识。

            • (3)感性认识和感性认识相互渗透、相互包含。

            • ④ 从感性认识上升到理性认识需要具备的条件:

              • (1)勇于实践、深入调查,获取丰富的感性材料,这是基础。

              • (2)必须经过理性思考,将感性材料加以去粗取精、去伪存真、由此及彼、由表及里加工,才能将感性认识上升为理性认识。

            • ⑤ 非理性因素:主要是指认识主体的情感和意志。也包括不自觉、非逻辑等特点的认识形式(如:联想、想象、猜测、直觉、顿悟、灵感等)

              • (1)人的认识过程是理性因素和非理性因素协同作用的结果。

              • (2)非理性因素对于人的认识能力和人是活动具有激活,驱动和控制作用我们应当在理性认识的指导下,充分发挥非理性因素的积极作用)

        • Ⅲ. 认识运动的基本规律:

          • (1)认识运动的辩证过程,是从实践到认识的过程,即在实践基础上由感性认识能动的飞跃到理性认识的过程。

          • (2)认识过程的第二个阶段是从认识到实践,即由 “精神到物质的阶段,由思想到存在的阶段”。

          • (3)认识运动是不断反复和无限发展的,是一个波浪式前进和螺旋式上升的过程。

        # 真理的含义及其属性;价值评价的含义

        • 真理的定义

        真理是人们对于客观事物及其规律的正确认识。

        • 真理的属性

          • (1) 真理的客观性

            真理具有客观性,凡真理都是客观真理。真理的客观性或客观真理有两层含义:一是指真理的内容是客观的,是不以人的意志为转移的客观世界;二是指检验真理的标准是客观的。检验真理的唯一标准是实践,而实践是物质性的活动。

            • (2) 真理的绝对性

            真理的绝对性或者绝对真理是指真理的内容具有正确性,真理的发展具有无限性。其具体含义有:其一,就真理的内容而言,任何真理都必然包含着同客观对象相符合的客观内容,都同谬误有原则的界限。因此,承认了客观真理就是承认了绝对真理。其二,就真理的发展而言,其前景是无限的。由于人类依其认识本性来说,能够正确认识无限发展着的物质世界。所以,承认世界的可知性,也就是承认了绝对真理。其三,就真理发展的最终结果而言,无数相对真理的总和构成为绝对真理。所以,承认认识发展的无限性,就必然承认绝对真理。

            • (3) 真理的相对性

            真理的相对性是指真理的有限性、有条件性,是指人们在一定条件下对世界的正确把握具有近似性。其含义有三层:其一,从认识的广度上看,任何真理只能是对无限的物质世界的某一阶段、某一方面、某一领域的认识。其二,从认识的深度上看,任何真理对特定事物的一定程度、一定层次的近似正确的反映。其三,从认识的进程看,任何真理只是对事物的一定发展阶段的正确认识。

        • 价值评价的定义:价值评价即对主体和客体的价值关系进行评判,是一种关于价值现象的评价性的认识活动。

        # 第三章 人类社会及其发展规律

        # 社会存在与社会意识的含义及其辩证关系(社会存在的含义,社会存在的要素及其对人类社会发展的作用;社会意识的含义及分类;社会存在与社会意识的辩证关系、社会意识相对独立性的表现)

        • 社会存在的含义:

        社会存在是指社会生活的物质方面,是社会实践和物质生活条件的总和。

        • 社会存在的要素:物质生产方式、自然地理环境和人口因素。

          • Ⅰ. 生产方式是社会历史发展的决定力量。

            • 其一,物质生产活动及生产方式是人类社会赖以存在和发展的基础,是人类其他一切活动的首要前提。

            • 其二,物质生产活动及生产方式决定着社会的结构、性质和面貌,制约着人们的经济生活、政治生活和精神生活等全部社会生活。

            • 其三,物质生产活动及生产方式的变化发展决定整个社会历史的变化发展,决定社会形态从低级向高级的更替和发展。

          • Ⅱ. 自然地理环境是人类社会生存和发展的必要和永恒的条件。是人们生活和生产的自然基础。自然地理环境提供了社会生活和生产资料的来源。

          • Ⅲ. 人口因素也是重要的社会物质生活条件,对社会发展起着制约和影响的作用。人是从事物质生产活动和其他一切社会活动的主体,是一切社会关系的承担者。

        • 社会意识的含义:社会意识是社会生活的精神方面,是社会存在的反映。

        • 社会意识的构成
          社会意识具有复杂的结构,可以按照不同的标准进行划分。

          • 根据社会意识的主体,可以分为个体意识、群体意识;

          • 根据其发展水平,可以分为

            • 低级的自发的社会心理

            • 高级的社会意识形式

              • 作为上层建筑的意识形式,又称为社会意识形态、观念上层建筑
                主要包括政治法律思想、道德、艺术、宗教、哲学等。
                在阶级社会中,社会意识形态本质上是占统治阶级的思想文化,反映经济基础,具有鲜明的阶级性。

              • 非上层建筑的意识形式
                包括语言学、逻辑学以及自然科学等。
                它不具有阶级性。

        • 社会存在和社会意识的关系

          社会存在和社会意识是辩证统一的关系。社会存在决定社会意识,社会意识是社会存在的反映,并反作用于社会存在。

          • (1) 社会存在对社会意识的决定作用表现在:

            • 其一,社会存在是社会意识内容的客观来源,社会意识是社会物质生活过程及其条件的主观反映。这是社会意识对社会存在的依赖性。

            • 其二,社会存在决定社会意识的性质。

            • 其三,随着社会存在的发展,社会意识也相应地或迟或早地发生变化和发展。

          • (2) 社会意识有其相对独立性,对社会存在具有反作用。其具体表现有:

            • 其一,社会意识具有相对独立性,它与社会存在之间存在发展的不平衡性、不同步性。

            • 其二,社会意识内部各种形式之间的相互影响及各自具有的历史继承性。

            • 其三,社会意识对社会存在的能动的反作用。这是社会意识相对独立性的突出表现。先进的社会意识,反映了社会发展的客观规律,可以对社会发展起积极的促进作用;而落后的社会意识,对社会发展起着阻碍的作用。

          • (3) 社会意识起作用的形式

        社会意识的能动作用是通过指导人们的实践活动实现的。思想本身并不能实现什么,要实现思想就要诉诸实践。而社会实践的主体是人民群众。因此,一种社会意识发挥作用的程度及范围大小、时间久暂,同它实际掌握群众的深度和广度密切联系在一起。

        # 生产力与生产关系的含义及其辩证关系;经济基础与上层建筑的含义及其辩证关系

        • 生产力的含义:是人类在生产事件中形成的改造和影响自然以使其适合社会需要的物质力量。

        • 生产关系的含义:生产关系是人们在物质生产过程中形成的不以人的意志为转移的经济关系。

        • 生产力与生产关系的关系

          生产力和生产关系是社会生产不可分割的两个方面。在社会生产中,生产力是生产的物质内容,生产关系是生产的社会形式,二者的有机结合和统一,构成社会的生产方式。

          • Ⅰ. 生产力决定生产关系:

            • (1) 生产力状况决定生产关系的性质。

            • (2) 生产力的发展决定生产关系的变化。

          • Ⅱ. 生产关系对生产力具有能动的反作用。其主要表现为:当生产关系适合生产力发展的客观要求时,它对生产力的发展起推动作用;当生产关系不适合生产力发展的客观要求时,它就会阻碍生产力的发展。

        • 经济基础的含义:经济基础是指由社会一定发展阶段的生产力所决定的生产关系的总和。其实质是社会一定发展阶段上的基本经济制度,是制度化的物质社会关系。其表现形式是经济体制,即社会基本经济制度所采取的组织形式和管理方式。

        • 上层建筑的含义:上层建筑是建立在一定经济基础之上的意识形态以及相应的制度、组织和设施。

        • 经济基础与上层建筑的关系 辩证统一

          • (1) 经济基础决定上层建筑。具体表现有:

            • 第一,经济基础的需要决定了上层建筑的产生;

            • 第二,经济基础的性质决定上层建筑的性质,有什么样的经济基础就有什么样的上层建筑;

            • 第三,经济基础的变更必然引起上层建筑的变革,并决定着其变革的方向。

          • (2) 上层建筑对经济基础具有反作用。

        这种反作用集中表现在,上层建筑保护和促进自己的经济基础的形成巩固和发展;同时反对和排除与自己性质不一致的生产关系。

        # 社会基本矛盾是人类社会发展的根本动力;人类社会发展动力学说

        • Ⅰ. 社会基本矛盾是人类社会发展的根本动力

          • 含义:社会基本矛盾就是指贯穿社会发展始终,规定社会发展过程的基本性质和基本趋势,并对社会历史发展起根本推动作用的矛盾。生产力和生产关系、经济基础和上层建筑的矛盾是社会的基本矛盾。

          • 作用:

            • (1)生产力是社会基本矛盾运动中最基本的动力因素,是人类社会发展和进步的最终决定力量。生产力是社会存在和发展的物质基础,是不能任意选择的物质力量和历史活动的前提。

            • (2)生产力是社会进步的根本内容,是衡量社会进步的根本尺度。人类社会是在生产力与生产关系的矛盾运动中前进的。

            • (3)社会基本矛盾特别是生产力和生产关系的矛盾,决定着社会中其他矛盾的存在和发展。

            • (4)经济基础和上层建筑的矛盾也会影响和制约生产力和生产关系的矛盾。

            • (5)社会基本矛盾具有不同的表现形式,并从根本上影响和促进社会形态的变化和发展。

        • Ⅱ. 人类社会发展动力学

          • (1)阶级斗争是社会基本矛盾在阶级社会中表现,是阶级社会发展的直接动力。离开了阶级斗争,就无法理解阶级社会发展。“没有对抗就没有进步。这是文明直到今天所遵循的规律。”

          • (2)社会革命是阶级斗争发展到一定阶级的产物,是推动社会发展的重要动力。

        # 第四章 资本主义的本质及规律

        # 商品经济的含义、产生条件、特点

        • 含义:是以交换为目的而进行生产的经济形式,它是一定社会历史条件产物。

        • 产生条件:

          • (1)存在社会分工

          • (2)生产资料和劳动产品属于不同的所有者。

        • 特点

          • (1)商品经济本质上是交换经济,在商品经济条件下,生产要素和消费资料的全部或大部分都要通过市场交换来获得,商品生产者以追求价值为目的,并通过市场交换来实现。

          • (2)商品经济是开放型经济,商品经济以社会分工为基础,强调生产过程中的分工与协作,人与人之间,商品生产者之间以及生产单位之间的经济联系随着商品经济的发展而日益紧密,范围也日益扩大。

          • (3)商品经济是开拓进取型经济。

          • (4)商品经济以扩大再生产为特征。商品生产者为获得更多的价值实现和在竞争中处于有利地位,必然不断增加投入,改进技术和改善经营管理,从而使生产规模不断扩大。

        # 商品的含义、商品二因素及其辩证关系、商品的价值量

        • 含义:是用来交换,能满足人们某种需要的劳动产品,具有使用价值和价值两种因素或者两种属性,是使用价值和价值的统一体。

        • 商品的二因素及其辩证关系

          • 1. 商品的二因素:使用价值和价值。

            • 关于使用价值:

              • (1)使用价值反映的是人与自然之间的物质关系,是商品的自然属性,是一切劳动产品所共有的属性,使用价值由商品本身的自然属性决定,离开了商品它就不复存在。

              • (2)使用价值是交换价值的物质承担者。

            • 关于价值:

              • (1)价值是凝结在商品中无差别的一般人类劳动,即人类脑力和体力的耗费。

              • (2)价值是商品所特有地位社会属性。

              • (3)价值是交换价值的基础,交换价值是价值的表现形式。

          • 2. 辩证关系

            • (1)商品的价值和使用价值之间是对立统一的关系。

            • (2)其对立表现在:商品的使用价值和价值是相互排斥的,二者不可兼得。要获得商品的价值,就必须放弃商品的使用价值;要得到商品的使用价值,就不能得到商品的价值。

            • (3)其统一性表现在:作为商品,必须同时具有使用价值和价值两个因素。使用价值是价值的物质承担者,价值寓于使用价值之中。一种物品尽管具有使用价值,但如果不是劳动产品,也没有价值。

        • 商品的价值量

          • (1)首先,价值量是由劳动者生产商品所耗费的劳动量决定的,而劳动量则是按照劳动时间来计量。决定商品价值量的不是生产商品的个别劳动时间,而只能是社会必要劳动时间。生产商品所需要的社会必要劳动时间随着劳动生产力的变化而变化。

          • (2)商品价值量与生产商所耗费的劳动时间成正比,与劳动生产力成反比。

          • (3)商品的价值量同简单劳动与复杂劳动有密切饿关系。简单劳动书指不需要经过专门训练和培养的一般劳动者能从事的劳动。复杂劳动是指需要经过专门的训练和培养,具有一定的文化知识和技术专长的劳动者所从事的劳动。

        # 劳动二重性

        • 生产商品的劳动可以区分为具体劳动和抽象劳动

          • 具体劳动是指生产一定使用价值的具体形式的劳动;

          • 抽象劳动是指撇开一切具有形式的、无差别的一般人类劳动,即人的体力和脑力消耗。

          • 具体劳动形成商品的使用价值,抽象劳动形成商品的价值实体。

        • 劳动的二重性决定了商品的二因素。具体劳动和抽象劳动是对立统一的:

          • 一方面,具体劳动和抽象劳动在时间上和空间上是统一的,是商品生产者的同一劳动过程的不可分割的两方面;

          • 另一方面,具体劳动所反映的是人与自然的关系,是劳动的自然属性,抽象劳动所反映的是商品生产者的社会关系,是劳动的社会属性。

        # 价值规律的内容、表现形式、作用

        • 价值规律的基本内容:

        • ①商品的价值量是由生产这种商品的社会必要劳动时间决定的。

        • ②商品交换要以价值量为基础,按照等价交换的原则进行。

        • 价值规律的表现形式:
          商品的价格围绕商品的价值上下波动。

        • 价值规律的作用:

          • ①自发地调节生产资料和劳动力在社会各生产部门之间的分配比例。

          • ②自发地刺激社会生产力的发展。

          • ③自发地调节社会收入的分配。

        # 劳动力商品的特点

        • 劳动力商品的使用价值是价值的源泉

        • 劳动力成为商品的基本条件:

          • 第一、劳动者是自由人,能够把自己的劳动力当做自己的商品来支配;

          • 第二、劳动者自由的一无所有,没有任何实现自己的劳动力所必需的物质条件。

        • 劳动力的价值,是由生产、发展、维持和延续劳动力所必需的生活必需品的价值决定的。
          它包括三个部分:

          • ① 是维持劳动者自身生存所必需的生活资料的价值,用以再生产他的劳动力。

          • ② 是劳动者家属的生存必需的生活资料的价值,用以延续劳动力的供给。

          • ③ 劳动者接受教育和训练所支出的费用,用以培训适合资本主义再生产所需要的劳动力。

        # 资本主义生产的直接目的;绝对剩余价值、相对剩余价值

        • 资本主义生产的直接目的是获取更多剩余价值。

        • 资本主义生产是以雇佣劳动为基础,以获取剩余价值为目的的生产。

        • 其生产过程具有二重性:一方面是生产使用价值的劳动过程;另一方面又是生产剩余价值的价值增殖过程。

        • 绝对剩余价值是指在必要劳动时间不变的条件下,由于延长工作日的长度和提高劳动强度而生产的剩余价值。

        • 相对剩余价值是指在工作日长度不变的条件下,通过缩短必要劳动时间而相对延长剩余劳动时间所生产的剩余价值

        # 资本主义扩大再生产的途径

        • 第一,第 Ⅰ 部类原有可变资本加上追加的可变资本,再加上本部类资本家用于个人消费的剩余价值之和,必须等于第 Ⅱ 部类的不变资本加上追加的不变资本,即
          Ⅰ(v+△v+m/x)=Ⅱ(c+△c)

        • 第二,第 Ⅰ 部类全部产品的价值必须等于两大部类原有的不变资本加上追加的不变资本,即
          Ⅰ(c+v+m)=Ⅰ(c+△c)+Ⅱ(c+△c)

        • 第三,第 Ⅱ 部类全部产品的价值必须等于两大部类原有的可变资本加上追加的可变资本,再加上资本家用于个人消费的剩余价值之和,即
          Ⅱ(c+v+m)=Ⅰ(v+△v+m/x)+Ⅱ(v+△v+m/x)

        # 资本主义基本矛盾、资本主义经济危机的实质

        • 基本矛盾:生产社会化和生产资料资本主义私人占有之间的矛盾。

        • 资本主义经济危机的实质:相对过剩的危机,即生产和销售矛盾造成的相对过剩。

        # 资本主义选举制度

        • 是资产阶级制定某种原则和程序,通过竞选产生议会和国家元首的一种政治机制。

        • 在资本主义国家中,选举已经成为国家政治制度运行中对社会发展和稳定产生举足轻重影响的一个不可或缺的政治机制。

        # 第五章 资本主义的发展及其趋势

        # 第六章 社会主义的发展及其规律(本章仅供学习参考,不作考试要求)

        # 科学社会主义的基本原则

        • 第一,在生产资料公有制基础上组织生产,满足全体社会成员的需要是社会主义生产的根本目的。

        • 第二,对社会生产进行有计划的指导和调节,实行等量劳动领取等量产品的按劳分配原则。

        • 第三,合乎自然规律地改造和利用自然。

        • 第四,无产阶级革命是无产阶级进行斗争的最高形式,必须以建立无产阶级专政的国家为目的。

        • 第五,社会主义事业必须坚持无产阶级政党的领导。

        • 第六,通过无产阶级专政和社会主义高度发展,最终实现向消灭阶级、消灭剥削,实现人的全面而自由发展的共产主义的过渡。

        • 一方面,科学社会主义基本原则任何时候都不能违背;另一方面,科学社会主义基本原则在实践中不能僵化地教条地对待它。

        # 经济文化相对落后国社会主义革命理论

        经济文化相对落后的国家社会主义建设的艰巨性和长期性

        • 第一、生产力发展状态的约束。社会主义国家在经济上落后于发达的资本主义国家,有的甚至落后很远,社会主义制度下必须把大力发展生产力作为根本任务,努力完成生产社会化、商品化、现代化的艰巨任务。

        • 第二、经济基础和上层建筑发展状况的制约。发展公有经济,改造小农经济,建立、巩固和完善社会主义的经济基础,就成为无产阶级政权的一项十分巨大的任务。社会主义民主政治建设受到这些国家的经济、政治、文化条件的严重制约,在这一过程中,还要进一步消灭阶级和阶级差别,改变几千年形成的传统观念,建设社会主义的先进文化,实现真正的社会公正的平等。

        • 第三,国际环境的严峻挑战。如果说在社会主义国家成立之初,国际资本主义对资本主义的进攻主要是武力方式,那么在社会主义建设取得重大成就后,进攻方式则往往转变为一和平演变为主。一是通过军事、政治压力盒有限的经济、科技的合作和援助,迫使社会主义国家屈从其经济政治。二是通过文化渗透,在社会主义国家内部制造经济、政治、思想等种种混乱,阻挠和破坏社会主义国家的发展。

        • 第四,马克思主义执政党和对社会主义发展的道路的探索和对社会主义建设规律的认识,需要一个长期的艰苦的

        # 第七章 共产主义是人类最崇高的社会理想

        # 实现共产主义是一个长期的历史过程(辨析题)p286

        • (1)理论上,马克思主义所揭示的社会形态发展与更替的规律是一般的历史规律,是只有在漫长的历史过程中才能体现出来的规律。

        • (2)每一个社会形态的产生发展,都会经历一个很长的历史时期,而旧的社会形态走向没落并为新的社会形态所代替也是一个长期的历史过程。

        • (3)从资本主义到共产主义的转变是一种根本的转变,它不是具体制度的更新,而是整个社会的根本改造,因而必然是个长期而艰难的过程。

        # 共产主义的基本特征 p275

        • (1)物质财富极大丰富,消费资料按需分配。

        • (2)社会关系高度和谐,人们精神境界极大提高。

        • (3)每个人自由而全面的发展,人类从必然王国向自由王国飞跃。

        \ No newline at end of file diff --git a/tags/AI/index.html b/tags/AI/index.html index 3e7ccfbb..a80e3619 100644 --- a/tags/AI/index.html +++ b/tags/AI/index.html @@ -1 +1 @@ -Tag: AI | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: AI | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git a/tags/DevOps/index.html b/tags/DevOps/index.html index 4782153a..eae0dee9 100644 --- a/tags/DevOps/index.html +++ b/tags/DevOps/index.html @@ -1 +1 @@ -Tag: DevOps | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: DevOps | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git a/tags/FT/index.html b/tags/FT/index.html index 2a84e6a9..c345a27a 100644 --- a/tags/FT/index.html +++ b/tags/FT/index.html @@ -1 +1 @@ -Tag: FT | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: FT | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git a/tags/JS/index.html b/tags/JS/index.html index 668e15f3..3c70ba89 100644 --- a/tags/JS/index.html +++ b/tags/JS/index.html @@ -1 +1 @@ -Tag: JS | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: JS | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git a/tags/Linux/index.html b/tags/Linux/index.html index 3ce32527..b4d0a52b 100644 --- a/tags/Linux/index.html +++ b/tags/Linux/index.html @@ -1 +1 @@ -Tag: Linux | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: Linux | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git a/tags/Node-js/index.html b/tags/Node-js/index.html index 7479c819..9f64983f 100644 --- a/tags/Node-js/index.html +++ b/tags/Node-js/index.html @@ -1 +1 @@ -Tag: Node.js | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: Node.js | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git a/tags/Rust/index.html b/tags/Rust/index.html index 671cf7f5..fab4e6d0 100644 --- a/tags/Rust/index.html +++ b/tags/Rust/index.html @@ -1 +1 @@ -Tag: Rust | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: Rust | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git a/tags/algebra/index.html b/tags/algebra/index.html index 8c589599..42778a50 100644 --- a/tags/algebra/index.html +++ b/tags/algebra/index.html @@ -1 +1 @@ -Tag: 代数 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 代数 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git a/tags/algebra/page/2/index.html b/tags/algebra/page/2/index.html index 0a1f3f27..68c91e1a 100644 --- a/tags/algebra/page/2/index.html +++ b/tags/algebra/page/2/index.html @@ -1 +1 @@ -Tag: 代数 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 代数 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git a/tags/chisel/index.html b/tags/chisel/index.html index 040c5135..9e862730 100644 --- a/tags/chisel/index.html +++ b/tags/chisel/index.html @@ -1 +1 @@ -Tag: Chisel | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: Chisel | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git a/tags/combinatorics/index.html b/tags/combinatorics/index.html index 93b4dcf5..8e26c173 100644 --- a/tags/combinatorics/index.html +++ b/tags/combinatorics/index.html @@ -1 +1 @@ -Tag: 组合数学 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 组合数学 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git a/tags/coursework/index.html b/tags/coursework/index.html index 7de66ad3..5c4ca6c9 100644 --- a/tags/coursework/index.html +++ b/tags/coursework/index.html @@ -1 +1 @@ -Tag: 大作业 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 大作业 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git a/tags/coursework/page/2/index.html b/tags/coursework/page/2/index.html index 71dfee7b..1d9e3bc0 100644 --- a/tags/coursework/page/2/index.html +++ b/tags/coursework/page/2/index.html @@ -1 +1 @@ -Tag: 大作业 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 大作业 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git a/tags/cs/index.html b/tags/cs/index.html index 50a69ebb..5b3730cb 100644 --- a/tags/cs/index.html +++ b/tags/cs/index.html @@ -1 +1 @@ -Tag: CS | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: CS | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git a/tags/cs/page/2/index.html b/tags/cs/page/2/index.html index 85a0feca..3b66f01d 100644 --- a/tags/cs/page/2/index.html +++ b/tags/cs/page/2/index.html @@ -1 +1 @@ -Tag: CS | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: CS | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git a/tags/ctf/index.html b/tags/ctf/index.html index d4b4e108..541894e8 100644 --- a/tags/ctf/index.html +++ b/tags/ctf/index.html @@ -1 +1 @@ -Tag: ctf | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: ctf | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git a/tags/discrete-math/index.html b/tags/discrete-math/index.html index 5c66b9f5..0d560849 100644 --- a/tags/discrete-math/index.html +++ b/tags/discrete-math/index.html @@ -1 +1 @@ -Tag: 离散数学 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 离散数学 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git a/tags/discrete-math/page/2/index.html b/tags/discrete-math/page/2/index.html index b57306ef..b035fe99 100644 --- a/tags/discrete-math/page/2/index.html +++ b/tags/discrete-math/page/2/index.html @@ -1 +1 @@ -Tag: 离散数学 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 离散数学 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git a/tags/discrete-math/page/3/index.html b/tags/discrete-math/page/3/index.html index 825055ee..2d5558e1 100644 --- a/tags/discrete-math/page/3/index.html +++ b/tags/discrete-math/page/3/index.html @@ -1 +1 @@ -Tag: 离散数学 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 离散数学 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git a/tags/graph/index.html b/tags/graph/index.html index d8d1f621..e71ee155 100644 --- a/tags/graph/index.html +++ b/tags/graph/index.html @@ -1 +1 @@ -Tag: 图论 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 图论 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git a/tags/graph/page/2/index.html b/tags/graph/page/2/index.html index 46d53534..44296ddf 100644 --- a/tags/graph/page/2/index.html +++ b/tags/graph/page/2/index.html @@ -1 +1 @@ -Tag: 图论 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 图论 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git a/tags/index.html b/tags/index.html index 679eed92..41927465 100644 --- a/tags/index.html +++ b/tags/index.html @@ -1 +1 @@ -AllTag | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +AllTag | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git a/tags/number/index.html b/tags/number/index.html index 64d0ba60..71bb5f2c 100644 --- a/tags/number/index.html +++ b/tags/number/index.html @@ -1 +1 @@ -Tag: 数论 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 数论 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git a/tags/os/index.html b/tags/os/index.html index e227eef4..9d2988e3 100644 --- a/tags/os/index.html +++ b/tags/os/index.html @@ -1 +1 @@ -Tag: OS | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: OS | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git a/tags/os/page/2/index.html b/tags/os/page/2/index.html index 3f17ee84..a6b4a2a2 100644 --- a/tags/os/page/2/index.html +++ b/tags/os/page/2/index.html @@ -1 +1 @@ -Tag: OS | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: OS | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git a/tags/principles-of-computer-composition/index.html b/tags/principles-of-computer-composition/index.html index 3212bc7f..092ee666 100644 --- a/tags/principles-of-computer-composition/index.html +++ b/tags/principles-of-computer-composition/index.html @@ -1 +1 @@ -Tag: 计组 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 计组 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git a/tags/principles-of-computer-composition/page/2/index.html b/tags/principles-of-computer-composition/page/2/index.html index b31c4cbb..3b509cb0 100644 --- a/tags/principles-of-computer-composition/page/2/index.html +++ b/tags/principles-of-computer-composition/page/2/index.html @@ -1 +1 @@ -Tag: 计组 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 计组 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git a/tags/scala/index.html b/tags/scala/index.html index c6f14de3..2d1619e3 100644 --- a/tags/scala/index.html +++ b/tags/scala/index.html @@ -1 +1 @@ -Tag: Scala | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: Scala | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git "a/tags/\344\277\241\345\217\267/index.html" "b/tags/\344\277\241\345\217\267/index.html" index e82f1c28..683fa77a 100644 --- "a/tags/\344\277\241\345\217\267/index.html" +++ "b/tags/\344\277\241\345\217\267/index.html" @@ -1 +1 @@ -Tag: 信号 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 信号 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git "a/tags/\345\202\205\351\207\214\345\217\266\345\217\230\346\215\242/index.html" "b/tags/\345\202\205\351\207\214\345\217\266\345\217\230\346\215\242/index.html" index 7bf9dbf1..247e8664 100644 --- "a/tags/\345\202\205\351\207\214\345\217\266\345\217\230\346\215\242/index.html" +++ "b/tags/\345\202\205\351\207\214\345\217\266\345\217\230\346\215\242/index.html" @@ -1 +1 @@ -Tag: 傅里叶变换 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 傅里叶变换 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git "a/tags/\345\211\215\347\253\257/index.html" "b/tags/\345\211\215\347\253\257/index.html" index fbf79369..b05f0cdf 100644 --- "a/tags/\345\211\215\347\253\257/index.html" +++ "b/tags/\345\211\215\347\253\257/index.html" @@ -1 +1 @@ -Tag: 前端 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 前端 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git "a/tags/\345\215\267\347\247\257/index.html" "b/tags/\345\215\267\347\247\257/index.html" index 11f4fc51..aa90d38d 100644 --- "a/tags/\345\215\267\347\247\257/index.html" +++ "b/tags/\345\215\267\347\247\257/index.html" @@ -1 +1 @@ -Tag: 卷积 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 卷积 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git "a/tags/\345\244\215\345\217\230\345\207\275\346\225\260/index.html" "b/tags/\345\244\215\345\217\230\345\207\275\346\225\260/index.html" index bc27497d..8165828a 100644 --- "a/tags/\345\244\215\345\217\230\345\207\275\346\225\260/index.html" +++ "b/tags/\345\244\215\345\217\230\345\207\275\346\225\260/index.html" @@ -1 +1 @@ -Tag: 复变函数 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 复变函数 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git "a/tags/\345\257\206\347\240\201\345\255\246/index.html" "b/tags/\345\257\206\347\240\201\345\255\246/index.html" index d61800e2..7e1594b2 100644 --- "a/tags/\345\257\206\347\240\201\345\255\246/index.html" +++ "b/tags/\345\257\206\347\240\201\345\255\246/index.html" @@ -1 +1 @@ -Tag: 密码学 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 密码学 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git "a/tags/\345\271\266\350\241\214/index.html" "b/tags/\345\271\266\350\241\214/index.html" index 169f21f9..cfda7966 100644 --- "a/tags/\345\271\266\350\241\214/index.html" +++ "b/tags/\345\271\266\350\241\214/index.html" @@ -1 +1 @@ -Tag: 并行 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 并行 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git "a/tags/\345\274\272\345\214\226\345\255\246\344\271\240/index.html" "b/tags/\345\274\272\345\214\226\345\255\246\344\271\240/index.html" index 33a21277..588d24f2 100644 --- "a/tags/\345\274\272\345\214\226\345\255\246\344\271\240/index.html" +++ "b/tags/\345\274\272\345\214\226\345\255\246\344\271\240/index.html" @@ -1 +1 @@ -Tag: 强化学习 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 强化学习 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git "a/tags/\346\200\235\346\224\277/index.html" "b/tags/\346\200\235\346\224\277/index.html" index a41dbdc4..90d2a673 100644 --- "a/tags/\346\200\235\346\224\277/index.html" +++ "b/tags/\346\200\235\346\224\277/index.html" @@ -1 +1 @@ -Tag: 思政 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 思政 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git "a/tags/\346\220\234\347\264\242/index.html" "b/tags/\346\220\234\347\264\242/index.html" index b118d5eb..64324698 100644 --- "a/tags/\346\220\234\347\264\242/index.html" +++ "b/tags/\346\220\234\347\264\242/index.html" @@ -1 +1 @@ -Tag: 搜索 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 搜索 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git "a/tags/\346\225\260\346\215\256\347\273\223\346\236\204/index.html" "b/tags/\346\225\260\346\215\256\347\273\223\346\236\204/index.html" index d1df52fd..a4a12eb9 100644 --- "a/tags/\346\225\260\346\215\256\347\273\223\346\236\204/index.html" +++ "b/tags/\346\225\260\346\215\256\347\273\223\346\236\204/index.html" @@ -1 +1 @@ -Tag: 数据结构 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 数据结构 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git "a/tags/\346\225\260\347\224\265/index.html" "b/tags/\346\225\260\347\224\265/index.html" index 9aa35e3c..8036fab6 100644 --- "a/tags/\346\225\260\347\224\265/index.html" +++ "b/tags/\346\225\260\347\224\265/index.html" @@ -1 +1 @@ -Tag: 数电 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 数电 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git "a/tags/\346\240\221\350\216\223\346\264\276/index.html" "b/tags/\346\240\221\350\216\223\346\264\276/index.html" index 110226ac..f2d7896d 100644 --- "a/tags/\346\240\221\350\216\223\346\264\276/index.html" +++ "b/tags/\346\240\221\350\216\223\346\264\276/index.html" @@ -1 +1 @@ -Tag: 树莓派 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 树莓派 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git "a/tags/\346\240\274/index.html" "b/tags/\346\240\274/index.html" index 97f97b2b..d9deff27 100644 --- "a/tags/\346\240\274/index.html" +++ "b/tags/\346\240\274/index.html" @@ -1 +1 @@ -Tag: 格 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 格 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git "a/tags/\346\246\202\347\216\207\350\256\272/index.html" "b/tags/\346\246\202\347\216\207\350\256\272/index.html" index 77c091e0..d0ebe8df 100644 --- "a/tags/\346\246\202\347\216\207\350\256\272/index.html" +++ "b/tags/\346\246\202\347\216\207\350\256\272/index.html" @@ -1 +1 @@ -Tag: 概率论 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 概率论 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git "a/tags/\347\216\257/index.html" "b/tags/\347\216\257/index.html" index 8b69efd6..fa1d006e 100644 --- "a/tags/\347\216\257/index.html" +++ "b/tags/\347\216\257/index.html" @@ -1 +1 @@ -Tag: 环 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 环 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git "a/tags/\347\256\227\346\263\225/index.html" "b/tags/\347\256\227\346\263\225/index.html" index 2d7f94bd..1a3c5985 100644 --- "a/tags/\347\256\227\346\263\225/index.html" +++ "b/tags/\347\256\227\346\263\225/index.html" @@ -1 +1 @@ -Tag: 算法 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 算法 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git "a/tags/\347\263\273\347\273\237/index.html" "b/tags/\347\263\273\347\273\237/index.html" index f7566d9b..902c7499 100644 --- "a/tags/\347\263\273\347\273\237/index.html" +++ "b/tags/\347\263\273\347\273\237/index.html" @@ -1 +1 @@ -Tag: 系统 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 系统 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git "a/tags/\347\276\244\350\256\272/index.html" "b/tags/\347\276\244\350\256\272/index.html" index 2158e63b..c0ddc7f4 100644 --- "a/tags/\347\276\244\350\256\272/index.html" +++ "b/tags/\347\276\244\350\256\272/index.html" @@ -1 +1 @@ -Tag: 群论 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 群论 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git "a/tags/\351\200\273\350\276\221/index.html" "b/tags/\351\200\273\350\276\221/index.html" index 695b5a38..0b79e9d3 100644 --- "a/tags/\351\200\273\350\276\221/index.html" +++ "b/tags/\351\200\273\350\276\221/index.html" @@ -1 +1 @@ -Tag: 逻辑 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 逻辑 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file diff --git "a/tags/\351\233\206\345\220\210\350\256\272/index.html" "b/tags/\351\233\206\345\220\210\350\256\272/index.html" index 0ebd067f..799557dc 100644 --- "a/tags/\351\233\206\345\220\210\350\256\272/index.html" +++ "b/tags/\351\233\206\345\220\210\350\256\272/index.html" @@ -1 +1 @@ -Tag: 集合论 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file +Tag: 集合论 | Balloon Party = 風船のパーティー = fuusen no party
        \ No newline at end of file

        *+T$VM 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 SrCJ>l{K!fvNtTG~dH)|Q@_lds diff --git a/assets/yatcpu/lab1/SingleCycle.html b/assets/yatcpu/lab1/SingleCycle.html index 1c8936c6..4ca23f27 100644 --- a/assets/yatcpu/lab1/SingleCycle.html +++ b/assets/yatcpu/lab1/SingleCycle.html @@ -1 +1 @@ -| Balloon Party = 風船のパーティー = fuusen no party