- 浏览: 111602 次
- 性别:
- 来自: 昆明
-
文章分类
- 全部博客 (151)
- 120D02 (5)
- 直升机 (1)
- 我的技术资料收集 (82)
- 的技术资料收集 (4)
- .NET Solution (2)
- ASP.NET (1)
- Linq to sql (1)
- 数据库技术(MS SQL) (2)
- 架构/设计 (1)
- 敏捷/持续集成 (1)
- C#.NET开发 (1)
- Matlab开发 (1)
- WinForm开发 (1)
- 开源技术 (1)
- jQuery (1)
- 我的博文 (4)
- js (2)
- android (2)
- 9. 读书笔记 (1)
- CSS3 (1)
- HTML5 (1)
- JavaScript (5)
- 移动开发 (2)
- 编程心得 (1)
- Linux操作系统 (1)
- (BI)商业智能 (1)
- IOS (1)
- Windows Phone (2)
- C# API (1)
- JQuery系列 (1)
- TFS (1)
- C# (2)
- ExtJs (1)
- .NET (1)
- Nginx (1)
- WCF学习笔记 (1)
- Computer Graphic (1)
- IT产品 (1)
- 工具分享 (1)
- MySelf (1)
- C#专栏 (1)
- 管理 (1)
- 基于Oracle Logminer数据同步 (1)
- 日常 (1)
- 实用工具 (1)
- 网页设计 (1)
- avalon (1)
- flash (1)
- DDD (1)
- 01 技术Android (1)
- WCF (1)
- selenium (1)
最新评论
-
464410531:
三国杀。。。。。。。。。。。。。。。。。。。。。。。。。。。。 ...
实用的职场宝典:不提拔你,就因为你只想把工作做好
之前有发表博文,简单的讲解一下观察者模式的大概内容(http://www.cnblogs.com/wenjiang/archive/2013/05/07/3065040.html),主要是利用java对观察者模式的内置支持来实现观察者模式,现在想要换个思路,自定义观察者模式。
这次使用Eclipse的单元测试框架,前面那个例子就不适合了,所以特意挑一个有关时钟报时的例子,方便测试。
敏捷开发的原则就是测试先于代码,这里就采用这个原则,先从测试代码开始:
private TimeScreen screen;
private TimeSource source;
public ClockTest(String name) {
super(name);
}
public void testTimeChange() {
TimeSource source = new TimeSource();
TimeScreen screen = new TimeScreen();
Clock clock = new Clock(source, screen);
source.setTime(3, 4, 5);
assertEquals(3, screen.getHours());
assertEquals(4, screen,getMinutes());
assertEquals(5,screen.getSeconds());
}
}
该测试主要测试:当时钟时间改变时,屏幕能否跟着改变。
因为时钟可能是电子时钟或者其他时钟,所以,我们定义一个时间来源的抽象:Source:
public void setSource(Clock clock);
}
同样屏幕也要有一个抽象:
public void setTime(itn hours, int minutes, int seconds);
}
接着是Clock的代码:
priate Screen screen;
public Clock(Source source, Screen screen){
source.setClock(this);
this.screen = screen;
}
public void update(int hours, int minutes, int seconds){
screen.setTime(hours, minutes, seconds);
}
}
Clock通知屏幕更新时间,所以它必须拥有Screen的运用,又因为它是从Source获取时间,所以它必须将自己传给Source,也就是说,它是Screen和Source之间的邮差。
我们来实现具体的Screen和Source:
private Clock clock;
public void setTime(int hours, int minutes, int seconds){
clock.update(hours, minutes, seconds);
}
public void setClock(Clock clock){
this.clock = clock;
}
}
private int hours;
private int minutes;
private int seconds;
public int getHours(){
return this.hours;
}
public int getMinutes(){
return this.minutes;
}
public int getSeconds(){
return this.seconds;
}
public void setTime(int hours, int minutes, int seconds){
this.hours = hours;
this.minutes = minutes;
this.seconds = seconds;
}
}
UML图如:
上面的代码能够通过测试,但并不是一个好方案,最主要的问题就是TimeSource持有Clock的引用。Clock确实是Source和Screen的邮差,但我们并不依赖于具体的邮差帮我们传送数据,邮差本身也可以是一个抽象:
public void update(int hours, int minutes, int seconds);
}
为了贴近今天的主题,特意将这个抽象命名为TimeObserver,因为它就是一个观察者,观察数据什么时候改变,然后通知相应的屏幕。
然后就是实现这个抽象:
private Screen screen;
public Clock(Source source, Screen screen){
source.setObserver(this);
this.screen = screen;
}
public void update(int hours, int minutes, int seconds){
screen.setTime(hours, minutes, seconds);
}
}
接着就是将原本引用Clock的地方都改为TimeObserver就行,像是下面这样:
public void setObserver(TimeObserver observer);
}
加入这样的抽象的好处非常明显,就是消除我们之前的依赖,也就是通过提供间接层的方式消除依赖的做法,这也是接口的作用。
通过查看代码,我们发现TimeObserver的update()其实就是调用Screen的setTime(),这是因为它必须通知Screen修改显示的时间,那么我们是否可以直接将Screen传递给Source的方法,而不是像之前那样需要TimeObserver?显然我们的测试代码需要进一步修改,修改的地方只有一处:
然后是我们的Source:
private int hours;
private int minutes;
private int seconds;
public int getHours(){
return this.hours;
}
public int getMinutes(){
return this.minutes;
}
public int getSeconds(){
return this.seconds;
}
public void update(int hours, int minutes, int seconds){
this.hours = hours;
this.minutes = minutes;
this.seconds = seconds;
}
}
为什么我们一开始会有一个Clock呢?因为我们需要一个邮差,但为什么不让我们的Source直接通知Screen呢?我们经常会犯这样的错误,尤其是在使用接口的时候,认为凡事有个间接层都是好的,都是动态的,其实不然,接口的确是个好东西,但是,该让谁实现这个接口就是一个问题。我们很容易像是上面一样,引入了一个具体类型Clock,而且代码的运行也没有错,它依然能够工作得很好,我们还可以和别人炫耀:看看我的邮差工作得多努力!
如何设计好接口,是面向对象编程中一个很重要的努力方向。
我的个人经验,当然,这经验是微不足道的(面向对象编程经验只有一年OTZ),如果两个类型之间需要进行通信,应该是让它们的抽象之间进行通信,也就是它们各自实现的接口,这样就能消除具体类型的耦合,而且这种通信的方式是以传参的方式进行。这样,我们的对象层次上既保证一定的解耦,又能保证逻辑上的耦合关系的完整。
之前的测试实在是太简单了,只是单独测试一个Screen,现实生活中的情况是非常复杂的,我们很可能需要多个屏幕,而且它们是不同材质,不同地方,这需要增加一个测试:
TimeSource source = new TimeSource();
TimeScreen screen = new TimeScreen();
source.registerObserver(screen);
TimeScreen screen2 = new TimeScreen();
source.registerObserver(screen2);
source.setTime(3, 4, 5);
assertScreenEquals(screen, 3, 4, 5);
assertScreenEquals(screen2, 3, 4, 5);
}
private void assertScreenEquals(TimeSource source, int hours, int minutes, int seconds){
assertEquals(hours, screen.getHours());
assertEquals(minutes, screen.getMinutes());
assertEquals(seconds, screen.getSeconds());
}
我们在Source里增加了一个方法:registerObserver(),正如其名,就是将相关的Screen注册进Source需要通知的名单中,新的Source如:
public void registerObserver(TimeObserver observer);
}
接着我们的TimeSource如:
private List<TimeObserver> observers = new ArrayList<TimeObserver>();
public void registerObserver(TimeObserver observer){
list.add(observer);
}
public void setTime(int hours, int minutes, int seconds){
for(TimeObserver observer : observers){
observer.update(hours, minutes, seconds);
}
}
}
我们用ArrayList来作为存储需要通知的Screen的名单,然后在时间更新的时候,逐个通知它们更新自己的时间。
但问题也来了,任何一个Source的实现类都必须实现注册和更新的代码,哪怕它们都是一样的。这样代码的重复性太高了,我们得想办法解决这个问题。
将Source从接口变为类型就可以解决了:
private List<TimeObserver> observers = new ArrayList<TimeObserver>();
protected void notify(int hours, int minutes, int seconds){
for(TimeObserver observer : observers){
observer.update(hours, minutes, seconds);
}
}
public void registerObserver(TimeObserver observer){
observers.add(observer);
}
}
然后我们的TimeSource只需要这样:
public void setTime(int hours, int minutes, int seconds){
notify(hours, minutes, seconds);
}
}
我们的派生类型的确是不需要重新写注册和更新的代码,只要调用基类的相关方法就行。
从上面我们可以知道,接口可以为我们提供间接层,减少具体类型的依赖,使得我们的代码更具动态,但是,它会使我们面临代码重复性较高的危险,更可怕的是,它会让我们陷入这样的怪论:"只要能呱呱叫,就是鸭子"。这是面向对象编程的一个经典现象,因为所有实现类都要实现接口规定的方法,而且我们不能阻止非目标类型对该接口的实现。
使用继承可以解决上面的怪论:"只有鸭子才能呱呱叫"。这是继承的本质,它规定的是一种类型,而不是一组行为协定。当然,接口也有自己的对策:将行为协议划分得更细,最好就是一组相关的行为放到一个接口里。前面之所以会出现这样的怪论,是因为程序员可能会这样设计接口:
public void fly();
public void shout();
}
这样的接口就会让人产生误解,正确的接口应该是这样的:
public void fly();
}
public interface ShoutAble{
public void shout();
}
接口的命名应该是动词,而不是名词,因为它规定的是一组行为协议。
但继承也存在自己的问题:"不是所有的鸭子都会呱呱叫",有些鸭子可能不会叫,但是它们是有方法可以呱呱叫的,这就会出现错误。
所以,使用继承解决问题的时候,我们必须明确一点:派生类能从基类中继承的职责到底是什么?
在这里,很明确的就是,我们的Source根本就没有必要理会注册和更新的行为,它本来应该只知道时间而已。于是,我们需要将这部分的职责从Source中移除。
使用委托是一个不错的选择:
private List<TimeObserver> observers = new ArrayList<TimeObserver>();
public void registerObserver(TimeObserver observer){
list.add(observer);
}
public void setTime(int hours, int minutes, int seconds){
for(TimeObserver observer : observers){
observer.update(hours, minutes, seconds);
}
}
}
public class TimeSource implements Source{
private TimeNotify notify = new TimeNotify();
public void registerObserver(TimeObserver observer){
notify.registerObserver(observer);
}
public void setTime(int hours, int minutes, int seconds){
notify.notify(hours, minutes, seconds);
}
}
private List<TimeObserver> observers = new ArrayList<TimeObserver>();
public void registerObserver(TimeObserver observer){
list.add(observer);
}
public void setTime(int hours, int minutes, int seconds){
for(TimeObserver observer : observers){
observer.update(hours, minutes, seconds);
}
}
}
public class TimeSource implements Source{
private TimeNotify notify = new TimeNotify();
public void registerObserver(TimeObserver observer){
notify.registerObserver(observer);
}
public void setTime(int hours, int minutes, int seconds){
notify.notify(hours, minutes, seconds);
}
}
private List<TimeObserver> observers = new ArrayList<TimeObserver>();
public void registerObserver(TimeObserver observer){
list.add(observer);
}
public void setTime(int hours, int minutes, int seconds){
for(TimeObserver observer : observers){
observer.update(hours, minutes, seconds);
}
}
}
public class TimeSource implements Source{
private TimeNotify notify = new TimeNotify();
public void registerObserver(TimeObserver observer){
notify.registerObserver(observer);
}
public void setTime(int hours, int minutes, int seconds){
notify.notify(hours, minutes, seconds);
}
}
使用委托是增加了一个间接层,专门用于负责注册和更新的具体实现,而我们的Source只要调用它的相应方法就行。
哦,间接层怎么又来了!明明开头我们消除了一个邮差,现在又来了个新的邮差!!此邮差非彼邮差。之前的邮差是因为我们的Source的具体类型要持有一个邮差的引用才能通知Screen的具体类型,但是事实就是Source的具体类型应该可以直接通知Screen的具体类型,这是职责的分离。但这里我们是职责过分集中在一个类型中,所以需要通过间接层将职责分离出去。
我们知道,这样的解释实在是太模糊了!同样是邮差,为什么一个邮差要被赶走,另一个邮差却要被雇佣,而且评价甚高!!这不公平!!!仔细想想它们的工作就知道了,之前的邮差它负责的工作是更新数据,而且还是命令Screen更新!!这就是冗余,所以它才会被赶走,但是现在这个邮差却负责了新的工作:通知Screen更新数据和注册新的Screen,Source的工作仅仅是命令它做事而已。这样辛苦工作的邮差怎么可能被炒呢!!
现在的我们已经将整个观察者实现出来了,只要将Source改为TimeSubject就行,因为在观察者模式中,被观察的就是Subject,而java中习惯的命名方式是TimeObservable。我们这里采用的是"推模型",也就是通过把数据传给notify和update方法从而把数据从Subject推给观察者Observer,而另一种方式"拉模型"是Observer在收到更新消息后,查询Subject得到。该使用哪种方式,就在于Observer是否知道是哪个Subject发生变化(Subject可以是多个),如果确定的话,可以使用"拉模型",否则使用"推模型"比较方便。
下面就是观察者模式的大概UML图:
观察者模式是一个非常好用的设计模式,它应用的范围非常广泛,解决了很多设计问题,而且存在各种变形,但万变不离其宗,只要我们谨记模式的意图,就能在我们毫无头绪的时候指点迷津,尤其是在一开始设计类的时候,如果画一下UML图,就会发现我们可以用观察者模式来解决这个问题。
本文链接:http://www.cnblogs.com/wenjiang/p/3149990.html,转载请注明。
发表评论
-
Javascript:猜猜弹出的是啥?为啥? - 幸福框架
2013-06-28 13:33 448原帖地址:http://www.cnblogs.com/hap ... -
C#中WindowsForm常见控件的运用 -- - 李晓峰
2013-06-28 13:27 1770原帖地址:http://www.cnblogs.com/liy ... -
海量数据处理利器之Hash——在线邮件地址过滤 - MyDetail
2013-06-27 12:00 673原帖地址:http://www.cnblo ... -
ASP.NET MVC 4 for Visual Studio 2010 下载地址 - 张鸿伟
2013-06-27 11:48 769原帖地址:http://www.cnblogs.com/wei ... -
【ASP.NET Web API教程】6.2 ASP.NET Web API中的JSON和XML序列化 - r01cn
2013-06-26 11:00 928原帖地址:http://www.cnblogs.com/r01 ... -
[珠玑之椟]估算的应用与Little定律 - 五岳
2013-06-26 10:54 657原帖地址:http://www.cnblogs.com/wuy ... -
30行,金额转人民币大写的代码 - 史蒂芬.王
2013-06-26 10:42 1041原帖地址:http://www.cnblogs.com/ste ... -
从银行的钱荒看一个公司的团队建设 产品线过多最终导致最赚钱的项目面临破产 - James Li
2013-06-26 10:36 649原帖地址:http://www.cnblogs.com/Jam ... -
Windows 8 动手实验系列教程 实验6:设置和首选项 - zigzagPath
2013-06-25 13:39 558原帖地址:http://www.cnblogs.com/zig ... -
闲聊可穿戴设备 - shawn.xie
2013-06-25 13:33 630原帖地址:http://www.cnblo ... -
如何使用开源库,吐在VS2013发布之前,顺便介绍下VS2013的新特性"Bootstrap" - 量子计算机
2013-06-25 13:27 884原帖地址:http://www.cnblogs.com/DSh ... -
iOS内存错误EXC_BAD_ACCESS的解决方法(message sent to deallocated instance) - VicStudio
2013-06-23 11:30 562原帖地址:http://www.cnblogs.com/vic ... -
记录asp.net在IE10下事件丢失排错经过 - Adming
2013-06-23 11:24 730原帖地址:http://www.cnblogs.com/wea ... -
记 FineUI 官方论坛所遭受的一次真实网络攻击!做一个像 ice 有道德的黑客! - 三生石上
2013-06-23 11:18 813原帖地址:http://www.cnblogs.com/san ... -
3、使用Oracle Logminer同步Demo
2013-06-19 10:33 584原帖地址:http://www.cnblogs.com/shi ... -
算法实践——数独的基本解法
2013-06-19 10:27 1469原帖地址:http://www.cnblogs.com/gre ... -
简单实现TCP下的大文件高效传输
2013-06-19 10:21 708原帖地址:http://www.cnblogs.com/sma ... -
avalon - 初步接触
2013-06-18 10:06 799原帖地址:http://www.cnblogs.com/aar ... -
Nginx学习笔记(一) Nginx架构
2013-06-18 09:59 544原帖地址:http://www.cnblogs.com/cod ... -
证书打印《二》
2013-06-17 10:47 553原帖地址:http://www.cnblogs.com/bin ...
相关推荐
行为型模式包括责任链模式、命令模式、解释器模式、迭代器模式、中介者模式、备忘录模式、观察者模式、状态模式、策略模式、模板方法模式和访问者模式等。责任链模式通过建立一个对象链来处理请求;命令模式将请求...
内容概要:本文档《数据结构》(02331)第一章主要介绍数据结构的基础概念,涵盖数据与数据元素的定义及其特性,详细阐述了数据结构的三大要素:逻辑结构、存储结构和数据运算。逻辑结构分为线性结构(如线性表、栈、队列)、树形结构(涉及根节点、父节点、子节点等术语)和其他结构。存储结构对比了顺序存储和链式存储的特点,包括访问方式、插入删除操作的时间复杂度以及空间分配方式,并介绍了索引存储和散列存储的概念。最后讲解了抽象数据类型(ADT)的定义及其组成部分,并探讨了算法分析中的时间复杂度计算方法。 适合人群:计算机相关专业学生或初学者,对数据结构有一定兴趣并希望系统学习其基础知识的人群。 使用场景及目标:①理解数据结构的基本概念,掌握逻辑结构和存储结构的区别与联系;②熟悉不同存储方式的特点及应用场景;③学会分析简单算法的时间复杂度,为后续深入学习打下坚实基础。 阅读建议:本章节内容较为理论化,建议结合实际案例进行理解,尤其是对于逻辑结构和存储结构的理解要深入到具体的应用场景中,同时可以尝试编写一些简单的程序来加深对抽象数据类型的认识。
内容概要:本文详细介绍了施耐德M580系列PLC的存储结构、系统硬件架构、上电写入程序及CPU冗余特性。在存储结构方面,涵盖拓扑寻址、Device DDT远程寻址以及寄存器寻址三种方式,详细解释了不同类型的寻址方法及其应用场景。系统硬件架构部分,阐述了最小系统的构建要素,包括CPU、机架和模块的选择与配置,并介绍了常见的系统拓扑结构,如简单的机架间拓扑和远程子站以太网菊花链等。上电写入程序环节,说明了通过USB和以太网两种接口进行程序下载的具体步骤,特别是针对初次下载时IP地址的设置方法。最后,CPU冗余部分重点描述了热备功能的实现机制,包括IP通讯地址配置和热备拓扑结构。 适合人群:从事工业自动化领域工作的技术人员,特别是对PLC编程及系统集成有一定了解的工程师。 使用场景及目标:①帮助工程师理解施耐德M580系列PLC的寻址机制,以便更好地进行模块配置和编程;②指导工程师完成最小系统的搭建,优化系统拓扑结构的设计;③提供详细的上电写入程序指南,确保程序下载顺利进行;④解释CPU冗余的实现方式,提高系统的稳定性和可靠性。 其他说明:文中还涉及一些特殊模块的功能介绍,如定时器事件和Modbus串口通讯模块,这些内容有助于用户深入了解M580系列PLC的高级应用。此外,附录部分提供了远程子站和热备冗余系统的实物图片,便于用户直观理解相关概念。
某型自动垂直提升仓储系统方案论证及关键零部件的设计.zip
2135D3F1EFA99CB590678658F575DB23.pdf#page=1&view=fitH
可以搜索文本内的内容,指定目录,指定文件格式,匹配大小写等
Windows 平台 Android Studio 下载与安装指南.zip
Android Studio Meerkat 2024.3.1 Patch 1(android-studio-2024.3.1.14-windows.zip)适用于Windows系统,文件使用360压缩软件分割成两个压缩包,必须一起下载使用: part1: https://download.csdn.net/download/weixin_43800734/90557033 part2: https://download.csdn.net/download/weixin_43800734/90557035
国网台区终端最新规范
国网台区终端最新规范
1.【锂电池剩余寿命预测】Transformer-GRU锂电池剩余寿命预测(Matlab完整源码和数据) 2.数据集:NASA数据集,已经处理好,B0005电池训练、B0006测试; 3.环境准备:Matlab2023b,可读性强; 4.模型描述:Transformer-GRU在各种各样的问题上表现非常出色,现在被广泛使用。 5.领域描述:近年来,随着锂离子电池的能量密度、功率密度逐渐提升,其安全性能与剩余使用寿命预测变得愈发重要。本代码实现了Transformer-GRU在该领域的应用。 6.作者介绍:机器学习之心,博客专家认证,机器学习领域创作者,2023博客之星TOP50,主做机器学习和深度学习时序、回归、分类、聚类和降维等程序设计和案例分析,文章底部有博主联系方式。从事Matlab、Python算法仿真工作8年,更多仿真源码、数据集定制私信。
Android项目原生java语言课程设计,包含LW+ppt
大学生入门前端-五子棋vue项目
这是一个完整的端到端解决方案,用于分析和预测阿联酋(UAE)地区的二手车价格。数据集包含 10,000 条二手车信息,覆盖了迪拜、阿布扎比和沙迦等城市,并提供了精确的地理位置数据。此外,项目还包括一个基于 Dash 构建的 Web 应用程序代码和一个训练好的 XGBoost 模型,帮助用户探索区域市场趋势、预测车价以及可视化地理空间洞察。 数据集内容 项目文件以压缩 ZIP 归档形式提供,包含以下内容: 数据文件: data/uae_used_cars_10k.csv:包含 10,000 条二手车记录的数据集,涵盖车辆品牌、型号、年份、里程数、发动机缸数、价格、变速箱类型、燃料类型、颜色、描述以及销售地点(如迪拜、阿布扎比、沙迦)。 模型文件: models/stacking_model.pkl:训练好的 XGBoost 模型,用于预测二手车价格。 models/scaler.pkl:用于数据预处理的缩放器。 models.py:模型相关功能的实现。 train_model.py:训练模型的脚本。 Web 应用程序文件: app.py:Dash 应用程序的主文件。 callback
资源内项目源码是来自个人的毕业设计,代码都测试ok,包含源码、数据集、可视化页面和部署说明,可产生核心指标曲线图、混淆矩阵、F1分数曲线、精确率-召回率曲线、验证集预测结果、标签分布图。都是运行成功后才上传资源,毕设答辩评审绝对信服的保底85分以上,放心下载使用,拿来就能用。包含源码、数据集、可视化页面和部署说明一站式服务,拿来就能用的绝对好资源!!! 项目备注 1、该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的,请放心下载使用! 2、本项目适合计算机相关专业(如计科、人工智能、通信工程、自动化、电子信息等)的在校学生、老师或者企业员工下载学习,也适合小白学习进阶,当然也可作为毕设项目、课程设计、大作业、项目初期立项演示等。 3、如果基础还行,也可在此代码基础上进行修改,以实现其他功能,也可用于毕设、课设、作业等。 下载后请首先打开README.txt文件,仅供学习参考, 切勿用于商业用途。
资源内项目源码是来自个人的毕业设计,代码都测试ok,包含源码、数据集、可视化页面和部署说明,可产生核心指标曲线图、混淆矩阵、F1分数曲线、精确率-召回率曲线、验证集预测结果、标签分布图。都是运行成功后才上传资源,毕设答辩评审绝对信服的保底85分以上,放心下载使用,拿来就能用。包含源码、数据集、可视化页面和部署说明一站式服务,拿来就能用的绝对好资源!!! 项目备注 1、该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的,请放心下载使用! 2、本项目适合计算机相关专业(如计科、人工智能、通信工程、自动化、电子信息等)的在校学生、老师或者企业员工下载学习,也适合小白学习进阶,当然也可作为毕设项目、课程设计、大作业、项目初期立项演示等。 3、如果基础还行,也可在此代码基础上进行修改,以实现其他功能,也可用于毕设、课设、作业等。 下载后请首先打开README.txt文件,仅供学习参考, 切勿用于商业用途。
此为代码审查工具 可查 文件数,字节数,总行数,代码行数,注释行数,空白行数,注释率等
内容概要:本文档涵盖了一项关于企业破产概率的详细分析任务,分为书面回答和Python代码实现两大部分。第一部分涉及对业务类型和破产状态的边际分布、条件分布及相对风险的计算,并绘制了相应的二维条形图。第二部分利用Python进行了数据处理和可视化,包括计算比值比、识别抽样技术类型、分析鱼类数据集以及探讨辛普森悖论。此外,还提供了针对鱼类和树木数据的统计分析方法。 适合人群:适用于有一定数学和编程基础的学习者,尤其是对统计学、数据分析感兴趣的大学生或研究人员。 使用场景及目标:①帮助学生掌握统计学概念如边际分布、条件分布、相对风险和比值比的实际应用;②教授如何用Python进行数据清洗、分析和可视化;③提高对不同类型抽样技术和潜在偏见的理解。 其他说明:文档不仅包含了理论知识讲解,还有具体的代码实例供读者参考实践。同时提醒读者在完成作业时需要注意提交格式的要求。
MCP快速入门实战,详细的实战教程
python,playwright基础