【含源码】用python做游戏有多简单好玩
有很多同学问我还有其他什么小游戏吗,游戏是怎么做的,难不难。我就用两篇文章来介绍一下,如何使用Python做游戏。
兔子与灌
俄罗斯方块
休闲五子棋
走迷宫
推箱子
消消乐
超多小游戏玩转不停↓
更多小游戏可以评论区讨论哦,喜欢支持的话,点击下方↓ 关注 + 私信 “小游戏” 得源代码哦!!
小游戏源码腾讯文档-在线文档https://docs.qq.com/doc/DU2t4VW1aQU9DcGdZ
用PyGame做游戏非常简单,我们今天第一篇文章,让大家实现一个可以在地图上移动的小猪。
基本框架
首先,无论你是做什么游戏,别管三七二十一,先把下面这段代码复制粘贴到你的编辑器里面。所有游戏都需要这几行代码:
import pygamedef main():pygame.init()pygame.display.set_caption('未闻Code:青南做的游戏') # 游戏标题win = pygame.display.set_mode((800, 600)) # 窗口尺寸,宽800高600running = Truewhile running:for event in pygame.event.get():if event.type == pygame.QUIT: # 点击左上角或者右上角的x关闭窗口时,停止程序running = False
main()
运行效果如下图所示:
加载素材
现在,我们随便找两张图片,一张作为背景,一张作为主角。尺寸不用太在意,差不多就可以了,因为我们可以用代码动态调整。下面两张图是我随便找的素材,大家注意图中红框框住的地方,是这两张图片的尺寸。
我们使用如下代码加载图片:
img_surf = pygame.image.load('图片地址').convert_alpha()
其中的 .convert_alpha() 是保留png图片的透明背景。如果你加载的图片不png图片,可以把 convert_alpha() 改成 convert() 。
如果要修改图片尺寸,使用如下代码:
img_surf = pygame.transform.scale(img_surf, (宽, 高))
要把图片显示在窗口中,使用下面两行代码:
win.blit(素材对象, (素材左上角的横坐标, 素材左上角的纵坐标))
pygame.display.flip()
完整的代码如下:
import pygamedef main():pygame.init()pygame.display.set_caption('未闻Code:青南做的游戏') # 游戏标题win = pygame.display.set_mode((800, 600)) # 窗口尺寸bg_small = pygame.image.load('bg.png').convert_alpha()bg_big = pygame.transform.scale(bg_small, (800, 600))pig = pygame.image.load('pig_in_car.png').convert_alpha()running = Truewhile running:for event in pygame.event.get():if event.type == pygame.QUIT: # 点击左上角或者右上角的x关闭窗口时,停止程序running = Falsewin.blit(bg_big, (0, 0)) # 背景图最先加载,坐标是(left, top)win.blit(pig, (200, 300))pygame.display.flip()main()
运行效果如下图所示:
需要注意的是, win.blit 和 pygame.display.flip() 都要放到while循环里面。其中 win.blit 的第一个参数是我们刚刚加载的素材对象。第二个参数是一个元组,标记这个图片左上角在画布上面的坐标。整个画布左上角对应坐标 (0, 0) 。由于背景图的尺寸也是 (800, 600) ,所以背景图的左上角放到 (0, 0) ,就刚好可以铺满整个画布。
哪里找素材?
我们做的是一个像素风格的游戏,可以到 itch.io 上面找素材:
这个网站提高了大量的游戏素材,并且绝大部分素材,在个人非商业用途的情况下是免费的。你找到自己喜欢的素材以后,就可以直接下载,整个过程你甚至都不需要登录(比国内的垃圾素材网站可良心多了)。
怎么我的素材长这样?
你下载了素材以后,可能会发现一件非常奇怪的事情,怎么素材全部画在一张图上?
实际上,这就是业界惯例,做素材的人会把每一类素材排列到一张图片上,你要用的时候,需要自己去裁剪。例如所有植物放在一张图上,所有雕像放在一张图上,地基贴图也放在一张图上。
上面我们演示用的背景图,初看起来是一张绿色的图,但是它实际上包含了多个地基元素,请注意我用红框框住的部分:
在正式游戏中,我们要把每一个基本元素拆出来,重新组合起来使用。重组的时候,有些元素要复制多份重复使用,有些元素要旋转缩放。最终组合成下面这样看起来 好看 的地图:
一般来说,像素风格的素材,尺寸大多是 16x16 , 32x32 , 64x64 , 128x128 。素材作者正常情况下会提供裁剪说明。如果没有提供的话,你也可以肉眼观察,然后猜一猜。
例如我要从雕像素材里面剪切出红框框住的女神像:
那么,我可以这样写代码:
img_surf = pygame.image.load('雕像素材.png').convert_alpha()
goddess= img_surf.subsurface(( 女神像左上角的横坐标 , 女神像左上角的纵坐标, 女神像的宽, 女神像的高))
运行效果如下图所示:
可能有同学问:为什么女神的坐标是这样的呢?我只能说,这个坐标是我试了很多次,试出来的。
使用小精灵来管理对象
除了背景图,我们添加的每一个元素都是一个对象,例如上面的小猪和女神像。原则上来讲,上面的代码就足够让你把游戏做得漂亮了,想加什么东西,就不停加载图片素材,然后放到合适的位置就可以了。
但我们可以使用面向对象的设计方法,让代码更容易维护,也更简单。PyGame里面,有一个类叫做 Sprite ,我们可以为每一个对象实现一个类,继承 Sprite ,然后把对象的素材设置成 .surf 属性,把对象的位置设置为 .rect 属性。例如上面的代码,我们修改一下:
import pygameclass Bg(pygame.sprite.Sprite):def __init__(self):super(Bg, self).__init__()bg_small = pygame.image.load('bg.png').convert_alpha()grass_land = bg_small.subsurface((0, 0, 128, 128))self.surf = pygame.transform.scale(grass_land, (800, 600))self.rect = self.surf.get_rect(left=0, top=0) # 左上角定位class Pig(pygame.sprite.Sprite):def __init__(self):super(Pig, self).__init__()self.surf = pygame.image.load('pig_in_car.png').convert_alpha()self.rect = self.surf.get_rect(center=(400, 300)) # 中心定位class Goddess(pygame.sprite.Sprite):def __init__(self):super(Goddess, self).__init__()building = pygame.image.load('building.png').convert_alpha()self.surf = building.subsurface(((7 * 64 - 10, 0, 50, 100)))self.rect = self.surf.get_rect(center=(500, 430)) # 女神像的中心放到画布(500, 430)的位置def main():pygame.init()pygame.display.set_caption('未闻Code:青南做的游戏') # 游戏标题win = pygame.display.set_mode((800, 600)) # 窗口尺寸bg = Bg()goddess = Goddess()pig = Pig()all_sprites = [bg, goddess, pig] # 注意添加顺序,后添加的对象图层在先添加的对象的图层上面running = Truewhile running:for event in pygame.event.get():if event.type == pygame.QUIT: # 点击左上角或者右上角的x关闭窗口时,停止程序running = Falsefor sprite in all_sprites:win.blit(sprite.surf, sprite.rect)pygame.display.flip()if __name__ == '__main__':main()
运行效果如下图所示:
注意代码中的 all_sprites = [bg, goddess, pig] ,这里我使用的是列表。后面会有更高级的数据结构 SpriteGroup 来储存他们。今天使用列表就足够了。
素材对象.get_rect() 会返回一个坐标定位对象,这个对象有多个属性,例如 .left , .top , .center , .width , .height 。在不传参数的情况下,默认 .left=0 , .top=0 ,PyGame会自动根据这个对象的尺寸计算 .width , .height 和 .center 。我们可以通过传入参数的形式主动设定。当你设定左上角的时候,它自动就能算出中心点的坐标;当你传入中心坐标的时候,它自动就能算出左上角的坐标。
理论上来讲,在每个类里面,素材对象可以用任何名字,不一定要用 .surf 。坐标定位对象也不一定要用 .rect ,只要你在 win.blit 的时候对应起来就可以了。但是如果你统一使用 .surf 和 .rect 会给你带来很多好处。这一点我们到物体碰撞那个地方再讲。因此我建议你就使用这两个名字。
让小猪动起来
既然是游戏,那肯定要按键盘让主角动起来。否则跟一幅画有什么区别呢?大家注意 main() 函数里面的 while running 这个循环,如果你在循环里面加上一行代码: print(111) ,你会发现当你运行这个游戏的时候, 111 会一直不停的打印出来。
PyGame本质上,就是通过 win.blit 不停地画图,由于这个 while 循环每秒要运行很多次,如果每次运行的时候,我们让 win.blit 的第二个参数,也就是素材对象的坐标有细微的差异,那么在人眼看起来,这个素材对象就在运动了。
我们的目标是按住键盘的上下左右方向键,小猪向4个不同的方向移动。在PyGame里面,获得键盘按住不放的键,使用如下代码实现:
keys = pygame.key.get_pressed()
它返回的是一个长得像列表的对象(但不是列表),当我们要判断某个键是否被按下的时候,只需要判断 if keys[想要判断的键] ,如果返回 True ,说明被按住了。基于这个原理,我们来写两段代码。首先修改 Pig 类,新增一个 .update 方法:
class Pig(pygame.sprite.Sprite):def __init__(self):super(Pig, self).__init__()self.surf = pygame.image.load('pig_in_car.png').convert_alpha()self.rect = self.surf.get_rect(center=(400, 300)) # 中心定位def update(self, keys):if keys[pygame.K_LEFT]:self.rect.move_ip((-5, 0)) # 横坐标向左elif keys[pygame.K_RIGHT]:self.rect.move_ip((5, 0)) # 横坐标向右elif keys[pygame.K_UP]:self.rect.move_ip((0, -5)) #纵坐标向上elif keys[pygame.K_DOWN]:self.rect.move_ip((0, 5)) # 纵坐标向下# 防止小猪跑到屏幕外面if self.rect.left < 0:self.rect.left = 0if self.rect.right > 800:self.rect.right = 800if self.rect.top < 0:self.rect.top = 0if self.rect.bottom > 600:self.rect.bottom = 600
.update 方法接收一个参数 keys ,就是我们按键返回的长得像列表的对象。然后判断是哪个方向键被按下了。根据被按下的键, .rect 坐标定位对象修改相应方向的值。 rect.move_ip 这里的 ip 是 inplace 的简写,也就是修改 .rect 这个属性自身。它的参数是一个元组,对应横坐标和纵坐标。横纵坐标小于0表示向左或者向上,大于0表示向右或者向下。
原来的 main() 函数只需要在 win.blit 之前增加两行代码:
keys = pygame.key.get_pressed()
pig.update(keys)
完整代码如下:
import pygameclass Bg(pygame.sprite.Sprite):def __init__(self):super(Bg, self).__init__()bg_small = pygame.image.load('bg.png').convert_alpha()grass_land = bg_small.subsurface((0, 0, 128, 128))self.surf = pygame.transform.scale(grass_land, (800, 600))self.rect = self.surf.get_rect(left=0, top=0) # 左上角定位class Pig(pygame.sprite.Sprite):def __init__(self):super(Pig, self).__init__()self.surf = pygame.image.load('pig_in_car.png').convert_alpha()self.rect = self.surf.get_rect(center=(400, 300)) # 中心定位def update(self, keys):if keys[pygame.K_LEFT]:self.rect.move_ip((-5, 0))elif keys[pygame.K_RIGHT]:self.rect.move_ip((5, 0))elif keys[pygame.K_UP]:self.rect.move_ip((0, -5))elif keys[pygame.K_DOWN]:self.rect.move_ip((0, 5))# 防止小猪跑到屏幕外面if self.rect.left < 0:self.rect.left = 0if self.rect.right > 800:self.rect.right = 800if self.rect.top < 0:self.rect.top = 0if self.rect.bottom > 600:self.rect.bottom = 600class Goddess(pygame.sprite.Sprite):def __init__(self):super(Goddess, self).__init__()building = pygame.image.load('building.png').convert_alpha()self.surf = building.subsurface(((7 * 64 - 10, 0, 50, 100)))self.rect = self.surf.get_rect(center=(500, 430)) # 女神像的中心放到画布(500, 430)的位置def main():pygame.init()pygame.display.set_caption('未闻Code:青南做的游戏') # 游戏标题win = pygame.display.set_mode((800, 600)) # 窗口尺寸bg = Bg()goddess = Goddess()pig = Pig()all_sprites = [bg, goddess, pig] # 注意添加顺序,后添加的对象图层在先添加的对象的图层上面running = Truewhile running:for event in pygame.event.get():if event.type == pygame.QUIT: # 点击左上角或者右上角的x关闭窗口时,停止程序running = Falsekeys = pygame.key.get_pressed()pig.update(keys)for sprite in all_sprites:win.blit(sprite.surf, sprite.rect)pygame.display.flip()if __name__ == '__main__':main()
最后的运行效果如下面这个视频所示:
PyGame做游戏真的非常简单,只要会加载素材,就能做出一个还能看得过去的游戏。今天我们学会了怎么添加素材,怎么捕获键盘事件。
PyGame可以读取Gif图片,但是你会发现加载进来以后,Gif不会动。下一篇文章,我们来讲讲如何让你控制的角色动起来,例如控制一个小娃娃,移动的时候,它的脚也跟着动。以及对象的碰撞检测。
相关文章:
【含源码】用python做游戏有多简单好玩
有很多同学问我还有其他什么小游戏吗,游戏是怎么做的,难不难。我就用两篇文章来介绍一下,如何使用Python做游戏。 兔子与灌 俄罗斯方块 休闲五子棋 走迷宫 推箱子 消消乐 超多小游戏玩转不停↓ 更多小游戏可以评论区讨论哦,喜欢…...
C++常用函数
std::sort std::sort 函数用于对数组或容器进行排序,可以按照默认的升序排序或指定比较函数进行排序。 语法如下: template <class RandomAccessIterator> void sort(RandomAccessIterator first, RandomAccessIterator last);template <clas…...
Android Framework基础到深入篇
Android Framework基础到深入篇 KernelSU Android上基于内核的Root方案 Android系统源码下载/编译篇...
【Go进阶训练营】聊一下go的gc原理
背景 正好周末时间,就打算梳理以下自己对go gc的理解。跳出语言层面来说,gc分为两种,一种是手动创建,手动销毁。另一种就是由自动分配自动销毁,前者就是c,c的代表,后者就是java,go。 而整个流程…...
英飞凌Tricore原理及应用介绍05_中断处理之中断路由(IR)模块详解
目录 1.概述1.1相关缩写2 TC3xx中IR特性介绍3.SRN(中断服务请求优先级)3.1 寄存器中的各Bit位讲解3.2 如何改变SRN配置4. 实际应用介绍4.1 如何利用SRC寄存器检查OS中断配置是否正确?1.概述 在Tricore架构中允许有多个中断源包括片上外设及外部中断世间产生的中断请求,以打…...
微搭问答002-移动端上传的文件如何在PC端下载
遇到一个问题,就是上传的图片,在手机上可以下载了,但在电脑上怎么下载到电脑 里,包括上传的文件 点击查看页面就可以吧,在企业工作台里 我做了查看页面,小程序可以,但H5和电脑页面不行 你创建一…...
初识JVM
目录 引言 JVM是什么? JVM和java有什么联系? JDK、JRE、JVM有什么区别 为什么学习JVM? JVM——从内存管理开始 运行时数据区域 分区讲解 堆 方法区 程序计数器 本地技术栈 虚拟机栈 对象的创建 指针碰撞: 空闲列表…...
实践分享:Vue 项目如何迁移小程序
最近我们小组刚经历了将成熟的 HTML5 项目转换成小程序,并在app中运行的操作!记录下来分享给各位。 项目:将已有的 Vue 项目转为小程序, 在集成了FinClip SDK 的 App 中运行。 技术:uni-app、FinClip 两个注意事项&…...
JavaScript学习笔记(6.0)
JavaScript类 使用关键字class创建类。 始终添加constructor()方法 class ClassName{constructor(){...} } calss Car{constructor(name,year){this.namename;this.yearyear; } } 创建了一个名为Car的类,并且拥有两个初始属性name和year。 JavaScript类不是对…...
某小公司面试记录
记录一次面试过程,还有一些笔试题,挺简单的,排序,去重,this指向,深浅拷贝,微任务的执行顺序,变量提升等。 ES6数组新增的方法 Array.from: 将两类对象转为真正的数组&am…...
SPI读写SD卡速度有多快?
SD卡是一个嵌入式中非常常用的外设,可以用于存储一些大容量的数据。但用单片机读写SD卡速度一般都有限(对于高速SD卡,主要是受限于单片机本身的接口速度),在高速、实时数据存储时可能会有影响。但具体速度可以达到多少…...
MySQL:索引与事物
目录 简单了解索引的底层数据结构 索引的概念: 索引存在的意义: 索引的使用: 索引实现的数据结构 B树 B 树 B 树的特点 B 树的优势 事物 事物的概念 事物的使用 事物的四大特性 并发可能引起的问题 脏读问题 不可重复读 幻读…...
mybatis实战
目录配置自动下划线驼峰MyBatis解析的SQL和实际传参不符的问题传参是整型,结果是false日期比较入参是字符串入参是Date父子递归查询上下级查询方法一方法二传参数组inmapper中接口注解映射配置 自动下划线驼峰 使用mybatis的自动下划线驼峰转换 mybatis有一个选项…...
【UEFI实战】BIOS与IPMI
KCS KCS全称是Keyboard Controller Style,关于这个名称不用过多的追究,只需要知道它是系统(BIOS和OS)和BMC通信的一种基本方式即可。本文将介绍BIOS下的KCS接口,包括接口使用方式和数据。内容参考自《ipmi-second-gen…...
90%的人都不算会网络安全,这才是真正的白帽子技术【红队】
我敢说,现在网上90%的文章都没有把网络安全该学的东西讲清楚。 为什么?因为全网更多的都是在讲如何去渗透和公鸡,却没有把网安最注重的防御讲明白。 老话说得好:“攻击,是为了更好的防御。”如果连初衷都忘了&#x…...
关于vuex的使用
1.首先安装vuex npm install vuex --save 这时如果直接安装vuex,不指定版本的话,就会直接安装最新的vuex的版本。所以会出现报错。 报错就安装这个 npm install --save vuex3 2.创建文件夹, 有的时候安装好会自动创建vuex的文件夹 …...
第53篇-某商城sign参数分析-webpack【2023-03-07】
声明:该专栏涉及的所有案例均为学习使用,严禁用于商业用途和非法用途,否则由此产生的一切后果均与作者无关!如有侵权,请私信联系本人删帖! 文章目录 一、前言二、网站分析三、完整代码一、前言 今天再来试一个webpack的例子吧,网址: aHR0cHM6Ly9tLnlxYi5jb20vYmFuay9…...
探秘MySQL——排查与调优
文章目录一、问题排查一:SQL执行出错二、问题排查二:慢查询0.几个重要参数1.配置慢查询日志命令行配置(重启失效)修改配置文件(永久生效)2.查看慢查询日志3.问题排查1:Look_time耗时4.问题排查2…...
【9.数据页结构】
概述 InnoDB 的数据是按「数据页」为单位来读写的,也就是说,当需要读一条记录的时候,并不是将这个记录本身从磁盘读出来,而是以页为单位,将其整体读入内存。数据库的 I/O 操作的最小单位是页,InnoDB 数据页…...
演唱会总是抢不到票?教你用Python制作一个自动抢票脚本
人生苦短 我用python 这个大家应该都知道吧? 是中国综合类现场娱乐票务营销平台, 业务覆盖演唱会、 话剧、音乐剧、体育赛事等领域。 如何快速抢票? 那么, 今天带大家用Python来制作一个自动抢票的脚本小程序 本文源码python安…...
【系统开发】WebSocket + SpringBoot + Vue 搭建简易网页聊天室
文章目录一、数据库搭建二、后端搭建2.1 引入关键依赖2.2 WebSocket配置类2.3 配置跨域2.4 发送消息的控制类三、前端搭建3.1 自定义文件websocket.js3.2 main.js中全局引入websocket3.3 App.vue中声明websocket对象3.4 聊天室界面.vue3.5 最终效果一、数据库搭建 很简单的一个…...
Learning C++ No.14【STL No.4】
引言: 北京时间:2023/3/9/12:58,下午两点有课,现在先把引言给搞定,这样就能激励我更早的把这篇博客给写完了,万事开头难这句话还是很有道理的,刚好利用现在昏昏欲睡的时候,把这个没…...
高速PCB设计指南(八)
七、产品内部的电磁兼容性设计 1 印刷电路板设计中的电磁兼容性 1.1 印刷线路板中的公共阻抗耦合问题 数字地与模拟地分开,地线加宽。 1.2 印刷线路板的布局 ※对高速、中速和低速混用时,注意不同的布局区域。 ※对低模拟电路和数字逻辑要分离。…...
什么是腾讯云关系型数据库(MySQL/SQL Server/MariaDB/PostgreSQL详解)
什么是腾讯云关系型数据库?腾讯云关系型数据库提供 MySQL、SQL Server、MariaDB、PostgreSQL详细介绍。腾讯云关系型数据库让您在云中轻松部署、管理和扩展的关系型数据库,提供安全可靠、伸缩灵活的按需云数据库服务。腾讯云关系型数据库提供 MySQL、SQL…...
进程通信相关概念
一、概念 1.1 通信方式有哪些 管道:水管,男纸条放入水管,女看了拿走不回复 消息队列:大盒子,男放入纸条,女看了不拿走,男女都可放 共享内存:直接桌子,男放桌上&#…...
05.Java的运算符
1.运算符计算机的最基本的用途之一就是执行数学运算,比如:int a 10;int b 20;a b;a < b;上述 和 < 等就是运算符,即:对操作数进行操作时的符号,不同运算符操作的含义不同。作为一门计算机语言,Ja…...
轮转数组(力扣189)
轮转数组 题目描述: 给定一个整数数组 nums,将数组中的元素向右轮转 k 个位置,其中 k 是非负数。 示例1: 输入: nums [1,2,3,4,5,6,7], k 3 输出: [5,6,7,1,2,3,4] 解释: 向右轮转 1 步: [7,1,2,3,4,5,6] 向右轮转 2 步: [6,7…...
主流的“对象转换工具”使用示例大全以及性能的对比
目录 前言 源码地址 代码示例 引入依赖 先定两个实体用于转换 定义一个接口让所有转换器都集成 Apache BeanUtils BeanCopier bean-mapping bean-mapping-asm Dozer 自己写get/set JMapper json2json MapStruct(推荐) ModelMapper OriK…...
分享10个不错的C语言开源项目
今天跟大家分享10个重量级的C语言开源项目,C语言确实经得住考验: Redis:Redis是一个开源的高性能的键值对数据库。它以C语言编写,具有极高的性能和可靠性。 Nginx:Nginx是一个高性能的HTTP和反向代理服务器࿰…...
【阅读笔记】JavaScript设计模式与开发实践2--闭包与单例、策略模式
目录闭包与高阶函数Function 扩展函数柯里化函数单例模式透明的单例模式惰性单例策略模式策略模式发展策略模式实现闭包与高阶函数 Array.prototype.sort 接受一个函数当作参数,用户可以自行在该函数内指定排序方式 // 由小到大排序 let res [1, 4, 2].sort((a, …...
淘宝客网站制作/百度一下 官方网
NumPy算术运算 NumPy 数组的“加减乘除”算术运算,分别对应 add()、subtract()、multiple() 以及 divide() 函数。 注意:做算术运算时,输入数组必须具有相同的形状,或者符合数组的广播规则,才可以执行运算。 下面看一…...
外贸没有公司 如何做企业网站/德州seo整站优化
大家是不是经常看到如下代码List list Arrays.asList("a","b","c");list.stream().forEach(System.out::println);这个“::”到底是什么语法呢?在JAVA 8 中,可以通过 “::” 关键字来访问类的构造方法,对象方…...
什么是静态网站和动态网站/新手做外贸怎么入门
oracle 的命名规则:1、要以字母开头2、包含字母和数字,以及# $3、不能超过30个字符 oracle基本数据类型数据类型参数描述char(n)n1 to 2000字节定长字符串,n字节长,如果不指定长度,缺省为1个字节长(一个汉字…...
iapp做网站/百度搜索引擎seo
源:将USBASP改造成STK502编程器 现在淘宝上还有不少USBasp或者USBisp的51/AVR编程器,它们使用了开源的USBasp硬件,以及PROGISP软件或者其它一些下载软件。其实我们可以非常方便的将它们改为STK502编程器,不但稳定性更好࿰…...
手机网站开发 教程/seo就是搜索引擎广告
温习一下这些天学习的CLR编程,存储过程,函数。 编写CLR的存储过程,运行起来的效率,果然比普通的SQL语句,存储过程或是函数均高。以后专案需求,或是执行效率较高的SQL,得写成CLR程序,…...
企业产品推广网站/网络营销手段
一个人成为穷人或者富人,到底是天注定还是靠打拼? 天赋与才能,对赚钱有多大作用呢? 什么事情靠勤奋努力有用? 什么事情靠勤奋努力就是瞎忙? 为什么投资更靠运气? 为什么选择比努力更重要&#x…...