Harvard CS50 学习笔记(四)
目录
摘要
Harvard CS50 学习笔记(四)。
4 Memory
4.1 Lecture
4.1.1 Storing finite information
4.1.2 Pixel art
4.1.3 RGB
4.1.4 Hexadecimal
16
进制一般前面有0x
,用于与10
进制区分。0x11=17
4.1.5 Memory addresses
4.1.6 address.c
4.1.7 Pointers
- 指针,是用于记录值的内存地址变量。
4.1.8 Declaring pointers
-
单个
&
,是address operator(地址运算符)。 -
&i
表示取变量i
的地址。 -
*
表示要记录指针(声明指针变量)。1 2 3 4
int n = 50; // 定义一个指向int型的指针变量p,并用n的地址来初始化p int *p = &n; printf("%p\n", p);
参考
4.1.9 “Address of”
4.1.10 Dereference operator
- deference operator(引用解析运算符)——
*
。 - reference——(变量的引用)。
- dereference——其实就是de(反)- reference(引用),即**(指针的)变量**。
- 一个
*
,表示我们要获取reference(引用),即地址值。 - 再加一个
*
,表示我们要反引用,也译为解引用,就是获取那个引用(地址)所储存的值(获取到之后其实可以改变地址所存储的值)。
4.1.11 Visualizing pointers
- 现在指针一般都是8字节(64比特)。
4.1.12 Assignment operators
C
中,不会自动做什么事情。- 如果有一个变量
n =50
,其指针为p =0x123
。 - 此时创建一个新变量
c = n
,p
不会有任何变化。
4.1.13 Strings in memory
data:image/s3,"s3://crabby-images/437e7/437e7eb82156231e5c96a7633f0e63ec11d1f73b" alt="/images/Harvard_CS50/Lecture_4/string.png"
- 字符串变量,实质为指向第一个字符地址的指针。
4.1.14 char *
string s = "HI!";
等同于char *s = "HI!";
4.1.15 Different Variables with Different Pointers
data:image/s3,"s3://crabby-images/7f745/7f74509c84d4ed14628ccafe52b3008957f70fc0" alt="/images/Harvard_CS50/Lecture_4/address_1.png"
data:image/s3,"s3://crabby-images/d07b0/d07b033f8a993d340ea4612473b7aab78dcb39e5" alt="/images/Harvard_CS50/Lecture_4/address_2.png"
- 为什么打印的结果不同?
4.1.16 address.c
- 结果不同的原因解析:
- 右边比较好理解,字符串变量的地址,与字符串第一个字符的地址相同。
- 左边的关键在于,第
7
行,把字符串的第一个字符H
,赋值给了一个新的变量c
,此操作在内存里进行了复制,也就是内存中多了一个新的变量c
,在不同的位置,储存着与字符串第一个字符相同的字符H
。 - 再打印变量
c
的地址,自然与字符串变量s
的地址不同。
4.1.17 Pointers to array values
data:image/s3,"s3://crabby-images/c69c6/c69c6185f721bac238fc268e520485392d3e2aac" alt="/images/Harvard_CS50/Lecture_4/array_pointer_1.png"
data:image/s3,"s3://crabby-images/c0397/c0397bb2f3c3f8ed74a32c98abfc4e3d293a7e7e" alt="/images/Harvard_CS50/Lecture_4/array_pointer_2.png"
- 左图比较好理解。
- 为什么右图的第
22
行打印出了不一样的地址?因为那是指针的地址。 - 基本类型如果要获取地址值,需要在变量名前加
&
。 - 数组与字符串(底层就是数组)不需要。
4.1.18 Defining strings
<cs50.h>
中的string
类型实质为typedef char *string;
4.1.19 Pointer arithmetic
data:image/s3,"s3://crabby-images/56dad/56dad3484ee3b35fc9849a4db3659b0b400643c1" alt="/images/Harvard_CS50/Lecture_4/array_pointer_3.png"
- 注意,这里的
+1
操作,加了4
个字节,因为一个int
占4
字节,而不是加1
个字节或1
个比特,编译器帮我们处理了。
4.1.20 Arrays as pointers
- 还是上图,此处并没有用
&
符号来表示数组的地址,但打印的时候,却使用了*
符号来解析地址。 - 因为可以把数组的变量名,直接作为指针使用。
- 即可以把数组,当作指向其第一个元素的指针。
4.1.21 Comparing integers
4.1.22 Comparing strings
4.1.23 Copying values
data:image/s3,"s3://crabby-images/6a174/6a1742d9abfc70fbd5debc78ba58e404d891dadb" alt="/images/Harvard_CS50/Lecture_4/copying_values_1.png"
4.1.24 malloc and free
malloc()
申请内存。free()
释放内存。- 需要
#include <stdlib.h>
- 实际上,这个
malloc()
就相当于Java
里面的new
一样,只是Java
把各种细节给你封装抽象成一个new
,GC
会自动给你回收,不需要自己去free()
。
4.1.25 Copying with malloc and free
4.1.26 Valgrind
- 检查与内存相关的潜在BUG。
4.1.27 Garbage values
- 不初始化,就会有无法预测的垃圾值。
4.1.28 Binky Pointer Fun
- 试图deference(引用解析)垃圾值,会造成错误。
4.1.29 Swap
4.1.30 swap.c
data:image/s3,"s3://crabby-images/7a0e6/7a0e6041af07a69bfa909c652a2714d88aad9643" alt="/images/Harvard_CS50/Lecture_4/swap_1.png"
- 没有传递引用,复制了值,原本的值不变。
4.1.31 Stack and heap
4.1.32 Visualizing swap.c
4.1.33 Pointers in swap.c
data:image/s3,"s3://crabby-images/31014/310140a643b80269524ea9e803c33d2876d08c56" alt="/images/Harvard_CS50/Lecture_4/swap_2.png"
4.1.34 scanf
4.1.35 Segmentation fault
4.1.36 Header bytes
4.1.37 Filter
4.2 Shorts
4.2.1 Hexadecimal
data:image/s3,"s3://crabby-images/e3bcf/e3bcfea0349acdc31c9070f2e4a0388c225d09d7" alt="/images/Harvard_CS50/Lecture_4/hexadecimal/hexadecimal_1.png"
data:image/s3,"s3://crabby-images/9d492/9d49206b2faa1549fb92bf9e3586fc5f6a3d9b32" alt="/images/Harvard_CS50/Lecture_4/hexadecimal/hexadecimal_2.png"
data:image/s3,"s3://crabby-images/d87e4/d87e4a05a6b2bcc7f9651463349d34770ad40ab0" alt="/images/Harvard_CS50/Lecture_4/hexadecimal/hexadecimal_3.png"
data:image/s3,"s3://crabby-images/ef5fe/ef5fe9b8e7dc0ca66e472fabd638ed9bed3d0dfa" alt="/images/Harvard_CS50/Lecture_4/hexadecimal/hexadecimal_4.png"
data:image/s3,"s3://crabby-images/e0f1f/e0f1fe70da85005e82f626a2d09d3a826eb68c32" alt="/images/Harvard_CS50/Lecture_4/hexadecimal/hexadecimal_5.png"
data:image/s3,"s3://crabby-images/3fcf2/3fcf2072cb73a662ddb9e9af3b4247a7ccd6c164" alt="/images/Harvard_CS50/Lecture_4/hexadecimal/hexadecimal_6.png"
data:image/s3,"s3://crabby-images/8e20e/8e20ea5a983ecfe0f5b3088cc443f99e889bfc67" alt="/images/Harvard_CS50/Lecture_4/hexadecimal/hexadecimal_7.png"
4.2.2 Pointers
data:image/s3,"s3://crabby-images/31307/31307765ff1bd52256da90cc7a5fddf03554ad77" alt="/images/Harvard_CS50/Lecture_4/pointers/pointers_1.png"
data:image/s3,"s3://crabby-images/de49c/de49c9452e5eec26693d68531e176c6cf651745f" alt="/images/Harvard_CS50/Lecture_4/pointers/pointers_2.png"
data:image/s3,"s3://crabby-images/114e7/114e71de35a0778a6e49ceb85009849806f43f31" alt="/images/Harvard_CS50/Lecture_4/pointers/pointers_3.png"
data:image/s3,"s3://crabby-images/ae53a/ae53a5ae809e89f54a8bfc43ef22e2e03c935e66" alt="/images/Harvard_CS50/Lecture_4/pointers/pointers_4.png"
data:image/s3,"s3://crabby-images/10c0b/10c0b24b6b032143dc749d4757532ebfc6f2fd44" alt="/images/Harvard_CS50/Lecture_4/pointers/pointers_5.png"
data:image/s3,"s3://crabby-images/678a3/678a372bb47c2fc1ce0f2657ec612132a21f839c" alt="/images/Harvard_CS50/Lecture_4/pointers/pointers_6.png"
- 指针就是地址值。
data:image/s3,"s3://crabby-images/90847/90847e22f37c1aed98339f4002a4f702f63b0a4c" alt="/images/Harvard_CS50/Lecture_4/pointers/pointers_7.png"
data:image/s3,"s3://crabby-images/33b45/33b45739e0e57808114ca547bedb681b9c892bb1" alt="/images/Harvard_CS50/Lecture_4/pointers/pointers_8.png"
data:image/s3,"s3://crabby-images/e98ae/e98ae5635add53082169a7ac986fa122f211c056" alt="/images/Harvard_CS50/Lecture_4/pointers/pointers_9.png"
- 指针的类型就是其指向地址值数据的类型。
data:image/s3,"s3://crabby-images/8cddb/8cddb1cd2d8f4ff8646b719bae0c798e5d883615" alt="/images/Harvard_CS50/Lecture_4/pointers/pointers_10.png"
- 指针必须初始化,如果初始化的事情没有明确的值,就指向空(NULL)。
data:image/s3,"s3://crabby-images/f7813/f7813a435229fc6e81b8d0b015e8dcad89978ed4" alt="/images/Harvard_CS50/Lecture_4/pointers/pointers_11.png"
data:image/s3,"s3://crabby-images/cc3f3/cc3f3e6f28b2324fd1f81fd9b0c2a0c6eeaebedd" alt="/images/Harvard_CS50/Lecture_4/pointers/pointers_12.png"
data:image/s3,"s3://crabby-images/0590c/0590c299869d5887ffa0e534668592de109b6484" alt="/images/Harvard_CS50/Lecture_4/pointers/pointers_13.png"
data:image/s3,"s3://crabby-images/0004e/0004e141e38f2ae78bfca80fb59bee2b4f560ad2" alt="/images/Harvard_CS50/Lecture_4/pointers/pointers_14.png"
data:image/s3,"s3://crabby-images/bf38f/bf38f7dd3f9fdd2155f207a486e7dfa3e4d3fbd4" alt="/images/Harvard_CS50/Lecture_4/pointers/pointers_15.png"
data:image/s3,"s3://crabby-images/280cd/280cdea96e0ef04fb5d7a34acec0985e5e23a29d" alt="/images/Harvard_CS50/Lecture_4/pointers/pointers_16.png"
data:image/s3,"s3://crabby-images/d507f/d507f1da9730ba5b80e47677e5f9c38375fe15de" alt="/images/Harvard_CS50/Lecture_4/pointers/pointers_17.png"
data:image/s3,"s3://crabby-images/ee0e9/ee0e9ac7a0d874a02cde3d19295792c69894457b" alt="/images/Harvard_CS50/Lecture_4/pointers/pointers_18.png"
data:image/s3,"s3://crabby-images/b2418/b2418ced9a05e561f854c463f9199c3003c16205" alt="/images/Harvard_CS50/Lecture_4/pointers/pointers_19.png"
4.2.3 Defining Custom Types
data:image/s3,"s3://crabby-images/c9348/c9348806d881c6087fb9ac7a6f4858c8c75cfdb1" alt="/images/Harvard_CS50/Lecture_4/defining_custom_types/defining_custom_types_1.png"
data:image/s3,"s3://crabby-images/3bb9b/3bb9b72407596d8a6359f01f1824f8f6b89c1287" alt="/images/Harvard_CS50/Lecture_4/defining_custom_types/defining_custom_types_2.png"
data:image/s3,"s3://crabby-images/65d03/65d03525fefede941d271ff67af3011c5ba4791a" alt="/images/Harvard_CS50/Lecture_4/defining_custom_types/defining_custom_types_3.png"
data:image/s3,"s3://crabby-images/0c694/0c694ec6521261139d8e846a1e038a95d5c69f22" alt="/images/Harvard_CS50/Lecture_4/defining_custom_types/defining_custom_types_4.png"
data:image/s3,"s3://crabby-images/00fab/00fab784d5f07444856a64ba78a15837876e9ddb" alt="/images/Harvard_CS50/Lecture_4/defining_custom_types/defining_custom_types_5.png"
data:image/s3,"s3://crabby-images/fae4a/fae4a9790e35d3daf13ec51ae89227aabfb6adc0" alt="/images/Harvard_CS50/Lecture_4/defining_custom_types/defining_custom_types_6.png"
4.2.4 Dynamic Memory Allocation
data:image/s3,"s3://crabby-images/5af26/5af26341447fe9d981db4c72bd6ff202d6955e1b" alt="/images/Harvard_CS50/Lecture_4/dynamic_memory_allocation/dynamic_memory_allocation_1.png"
- 一般有名字的变量在栈,没名字的在堆。
data:image/s3,"s3://crabby-images/e129f/e129f2de92833c5cb7d49e94195fadd85755a899" alt="/images/Harvard_CS50/Lecture_4/dynamic_memory_allocation/dynamic_memory_allocation_2.png"
- 堆与栈实质为一块内存空间的不同部分。
data:image/s3,"s3://crabby-images/a19bb/a19bb48ce1707af1d987f20501f39cda1c4168bf" alt="/images/Harvard_CS50/Lecture_4/dynamic_memory_allocation/dynamic_memory_allocation_3.png"
data:image/s3,"s3://crabby-images/7103e/7103e054700bde6f406db52ed02eb65d78989ccf" alt="/images/Harvard_CS50/Lecture_4/dynamic_memory_allocation/dynamic_memory_allocation_4.png"
data:image/s3,"s3://crabby-images/ffcae/ffcae4230468c68dd56af3787eacfddd4d502443" alt="/images/Harvard_CS50/Lecture_4/dynamic_memory_allocation/dynamic_memory_allocation_5.png"
data:image/s3,"s3://crabby-images/80e23/80e23e72667eb0801baa34984dbbfaeccfa6c529" alt="/images/Harvard_CS50/Lecture_4/dynamic_memory_allocation/dynamic_memory_allocation_6.png"
data:image/s3,"s3://crabby-images/f42d0/f42d067d04fa3a11a8a4b242466cd5cf9b7ffc5d" alt="/images/Harvard_CS50/Lecture_4/dynamic_memory_allocation/dynamic_memory_allocation_7.png"
data:image/s3,"s3://crabby-images/d31b7/d31b71ee5aac08a66bd36d86181d2aeba1661e53" alt="/images/Harvard_CS50/Lecture_4/dynamic_memory_allocation/dynamic_memory_allocation_8.png"
- 使用
malloc()
必须释放内存。 - 只有
malloc()
使用的内存需要被释放(静态声明在堆的变量不需要释放)。 - 不要重复释放。
data:image/s3,"s3://crabby-images/1a47e/1a47e2d3f8f456a4914a470454cac896d1bb81ae" alt="/images/Harvard_CS50/Lecture_4/dynamic_memory_allocation/dynamic_memory_allocation_9.png"
data:image/s3,"s3://crabby-images/d524e/d524e9e39f1160b3f54230a2d5af145b35d319c6" alt="/images/Harvard_CS50/Lecture_4/dynamic_memory_allocation/dynamic_memory_allocation_10.png"
data:image/s3,"s3://crabby-images/0b699/0b699c54ae8bd78af0c85d98716a725d8962cd7a" alt="/images/Harvard_CS50/Lecture_4/dynamic_memory_allocation/dynamic_memory_allocation_11.png"
4.2.5 Call Stacks
data:image/s3,"s3://crabby-images/b7387/b73878e5eceaba1734469aaa2047fad3e1bdccda" alt="/images/Harvard_CS50/Lecture_4/call_tacks/call_tacks_1.png"
data:image/s3,"s3://crabby-images/28b6e/28b6eee004c9c327c00bc8115ba2e58729141459" alt="/images/Harvard_CS50/Lecture_4/call_tacks/call_tacks_2.png"
4.2.6 File Pointers
data:image/s3,"s3://crabby-images/15c01/15c01180667a6d5fa5047a1482f5572ec74ca96e" alt="/images/Harvard_CS50/Lecture_4/file_pointers/file_pointers_1.png"
C
操作文件有抽象的数据类型,叫FILE
。- 一般情况下,只要操作文件,都用文件的指针,即
FILE*
。
data:image/s3,"s3://crabby-images/f47bb/f47bbb0d82496a3572e145022db2a7089d759159" alt="/images/Harvard_CS50/Lecture_4/file_pointers/file_pointers_2.png"
- 文件相关的函数,都在
stdio.h
里。
data:image/s3,"s3://crabby-images/56b3e/56b3e131674371146adce9839cfc1d2378720815" alt="/images/Harvard_CS50/Lecture_4/file_pointers/file_pointers_3.png"
- 打开一个文件,即获取这个文件的指针。
- 确保
NULL
的检查。 - 操作可以为
r-read
,w-write
,a-append
。
data:image/s3,"s3://crabby-images/dda63/dda6390ae1640d752e605c4e401fa7b091a7a447" alt="/images/Harvard_CS50/Lecture_4/file_pointers/file_pointers_4.png"
data:image/s3,"s3://crabby-images/35189/35189c7db195bf2ab91b87dd818bae26e98c9cd5" alt="/images/Harvard_CS50/Lecture_4/file_pointers/file_pointers_5.png"
data:image/s3,"s3://crabby-images/0d016/0d016876375fafb1d99c0e1e3e27516644a3067e" alt="/images/Harvard_CS50/Lecture_4/file_pointers/file_pointers_6.png"
data:image/s3,"s3://crabby-images/1d2f9/1d2f9ecbedb724d0dce2ccb6905ce33f680ae98b" alt="/images/Harvard_CS50/Lecture_4/file_pointers/file_pointers_7.png"
- 关闭文件指针。
data:image/s3,"s3://crabby-images/17aef/17aef3184878cd5ff0ffa565add3e907e622e8e5" alt="/images/Harvard_CS50/Lecture_4/file_pointers/file_pointers_8.png"
- 读字符函数。
- 获取文件的下一个字符。
- 打开文件的方式必须为
r-read
。
data:image/s3,"s3://crabby-images/f82cd/f82cdc449f2c598fc7be86cea3e54685c14ff0b7" alt="/images/Harvard_CS50/Lecture_4/file_pointers/file_pointers_9.png"
data:image/s3,"s3://crabby-images/80858/808581c4b70d7ea7828338bff5585ac612d35ee9" alt="/images/Harvard_CS50/Lecture_4/file_pointers/file_pointers_10.png"
data:image/s3,"s3://crabby-images/317f7/317f7d26b93408d80665d233bd8d35a9ed8abca0" alt="/images/Harvard_CS50/Lecture_4/file_pointers/file_pointers_11.png"
- 可通过循环读取整个文件。
EOF
表示 End Of File。
data:image/s3,"s3://crabby-images/97278/97278f46d7f23f61be27532b1f80d75d4ced79e6" alt="/images/Harvard_CS50/Lecture_4/file_pointers/file_pointers_12.png"
- 写字符函数。
- 将字符写入文件。
- 打开文件的方式必须为
w-wirte
或a-append
。
data:image/s3,"s3://crabby-images/19b0f/19b0f201ca2ed753ea7844e116ee37b747159c5c" alt="/images/Harvard_CS50/Lecture_4/file_pointers/file_pointers_13.png"
data:image/s3,"s3://crabby-images/cdc46/cdc4661b51d3549b03cf57552e55b1111746acdb" alt="/images/Harvard_CS50/Lecture_4/file_pointers/file_pointers_14.png"
data:image/s3,"s3://crabby-images/4a9fb/4a9fb6b9bf011f702379173c3acb02bf07b87905" alt="/images/Harvard_CS50/Lecture_4/file_pointers/file_pointers_15.png"
data:image/s3,"s3://crabby-images/670e3/670e367478faa66a27e655fd7b3823b868011938" alt="/images/Harvard_CS50/Lecture_4/file_pointers/file_pointers_16.png"
- 可通过循环读与写,实现文件复制。
data:image/s3,"s3://crabby-images/c1cc5/c1cc5f6ecd913ec6ef1c61d8b85d36fb97e5fcb0" alt="/images/Harvard_CS50/Lecture_4/file_pointers/file_pointers_17.png"
- 读文件函数,参数从右往左看。
<file pointer>
为文件指针,必须是读指针。<qty>
读文件的数据块的数量。<size>
读文件的数据块的大小(单位为字节)。<buffer>
内存中的缓存区域,用于存放读到的数据,需要用指针表示(数组名本身为指针)。
data:image/s3,"s3://crabby-images/a3bbd/a3bbd8101c0ac7a731778847baa5cf0a357f539a" alt="/images/Harvard_CS50/Lecture_4/file_pointers/file_pointers_18.png"
data:image/s3,"s3://crabby-images/0d451/0d451d2f694fafa9ff03d76e2e970dc7ffadc60f" alt="/images/Harvard_CS50/Lecture_4/file_pointers/file_pointers_19.png"
data:image/s3,"s3://crabby-images/0f8a0/0f8a0667e828c1479050072a51648737f84c882d" alt="/images/Harvard_CS50/Lecture_4/file_pointers/file_pointers_20.png"
data:image/s3,"s3://crabby-images/95a01/95a011749100b0fcb91dd85689fb504d045812fc" alt="/images/Harvard_CS50/Lecture_4/file_pointers/file_pointers_21.png"
data:image/s3,"s3://crabby-images/bb1af/bb1af308e7a463ed25dc26a6fc64e855e007f3f3" alt="/images/Harvard_CS50/Lecture_4/file_pointers/file_pointers_22.png"
- 写文件函数,参数从右往左看。
<file pointer>
为文件指针,必须是写指针。<qty>
读文件的数据块的数量。<size>
读文件的数据块的大小(单位为字节)。<buffer>
内存中的缓存区域,用于存放要写的数据,需要用指针表示(数组名本身为指针)。
data:image/s3,"s3://crabby-images/ab69e/ab69e25f2ece319b302cdcf3e3ca4fa0f744b64f" alt="/images/Harvard_CS50/Lecture_4/file_pointers/file_pointers_23.png"
data:image/s3,"s3://crabby-images/caf27/caf2730b398e236df13d0d581d65db817fcc980f" alt="/images/Harvard_CS50/Lecture_4/file_pointers/file_pointers_24.png"
data:image/s3,"s3://crabby-images/ffbb5/ffbb50ecedd328fece69fc27208cd21739cc5e2c" alt="/images/Harvard_CS50/Lecture_4/file_pointers/file_pointers_25.png"
data:image/s3,"s3://crabby-images/94a9c/94a9cf861d6fa6e28116bbc5c6b11b04f98df8e4" alt="/images/Harvard_CS50/Lecture_4/file_pointers/file_pointers_26.png"
data:image/s3,"s3://crabby-images/09980/09980248d49a81d69d4904229376b77fc2835105" alt="/images/Harvard_CS50/Lecture_4/file_pointers/file_pointers_27.png"
4.3 Lab 4
|
|
4.4 Problem Set 4
4.4.1 FIlter
|
|
4.4.2 Recover
|
|