当前位置:首页 » 《随便一记》 » 正文

如何写出一段好的代码(以实现strlen为例)_Allen9012的博客

27 人参与  2022年05月10日 11:09  分类 : 《随便一记》  评论

点击全文阅读


如何写出一段好的代码

    • 优秀的代码要具备以下条件
    • 常见的coding技巧:
      • 5分答案
      • 6分答案
      • 6.5分答案
      • 7分答案
      • 7.5分答案
      • 8分答案
      • 9分答案
      • 10分答案
    • 拓展练习

今天简单来谈谈如何写出一段好的代码

一般来说,我们总要试着写出优秀的代码,有些人调侃为了保证自己的“核心竞争力”要不写注释,多写嵌套,完了之后还bug一堆,写出那些个及其难以调试和让一般人读不懂的代码,这自然是不合理的,调侃毕竟是调侃,一直这样只会受到同事鄙夷的目光~~

那么我们要如何写出优秀的代码呢?

首先推荐阅读《高质量c/c++程序设计指南》

优秀的代码要具备以下条件

  1. 代码运行正常
  2. bug很少
  3. 效率高
  4. 可读性高
  5. 可维护性高
  6. 注释清晰
  7. 文档齐全

常见的coding技巧:

  1. 使用assert
  2. 尽量使用const
  3. 养成良好的编码风格
  4. 添加必要的注释
  5. 避免编码的陷阱

比如说接下来我们来实现一个库函数strcpy

首先来看一下MSDN中库函数strcpy的参数

image-20211121202352525

所以我们需要两个参数,两个指针一个指向复制目标的地址,还有一个指向复制的源

比如说我给出这样的主函数:

int main()
{
	//strcpy
	char arr1[] = "别复制到我这里";
	char arr2[] = "别把我复制过去";
	my_strcpy(arr1,arr2);//把arr2 拷贝到 arr1 
	printf("%s\n",arr1);
	return 0;
}

比如说我这样来实现:

5分答案

void my_strcpy(char*dest, char*src)
{
	while (*src!='\0')
	{
	*dest = *src;
	src++;
	dest++;
	}
}

看上去好像没什么问题实际上10分里面只能给5分,不及格,因为这样的代码存在的问题就是当我走到\0的时候并没有把\0给复制过去,所以相当于没有实现我要求的功能,自然是不及格的,所以代码仍然需要改进

6分答案

void my_strcpy(char*dest, char*src)
{
	while (*src!='\0')
	{
	*dest = *src;
	src++;
	dest++;
	}
	*dest = *src;//还得把斜杠0搞过去
}

最后把斜杠0搞过去了,以为这样就好了,没有现在只不过刚刚及格,还存在可以优化的地方,可以缩短行数,如下

6.5分答案

void my_strcpy(char*dest, char*src)
{
	while (*src != '\0')  
	{
		*dest++ = *src++;
	}
	*dest = *src;//还得把斜杠0搞过去
}

没错,这个答案将循环体里面的3行代码优化成了一行代码,然而这样的代码还是可以优化

7分答案

void my_strcpy(char*dest, char*src)
{
//	while (*src != '\0')  
//	{
//		*dest++ = *src++;
//	
//	}
	while (*dest++ = *src++)//判断条件里面先赋值后++
	{
		;//空语句返回继续循环
	}
	//斜杠0  字符ASCII是0 停下来了
}

这里一下子把\0和指针后移的操作和道理一起,看上去很妙,最后,遇到斜杠0 字符ASCII是0,while遇到0停下来了但是

有没有想过万一传进来的是空指针怎么办?程序什么也不会反应而且也没有return 0

image-20211121210051718

7.5分答案

void my_strcpy(char*dest, char*src)
{
	if (dest != NULL&&src != NULL)
	{
       while (*dest++ = *src++)
		{
			;
		}
	}
}

这时候我们做一个判断防止传入的是NULL,如果传入了NULL就不进入循环,这样还是arr1原样至少不会啥也没有

image-20211121210325450

8分答案

void my_strcpy(char*dest, char*src)
{
	assert(dest!=NULL);//若为假则会报错
	assert(src!= NULL);
		while (*dest++ = *src++)
		{
			;
		}
}

assert防止传进来的是空指针,这个就比较好,因为NULL直接报错,这样直接就知道哪一行哪里出错了,其它人一看就知道是老司机,不是小白菜

image-20211121214154420

image-20211121213910953

但是还没有完,因为库里给到的函数是返回char*的而我们只是void*

9分答案

char* my_strcpy(char*dest,char*src)
{
	assert(dest&&src);
    char*ret=dest;
		while (*dest++ = *src++{
			;
		}
    return ret;
}
int main()
{
	//strcpy
	char arr1[] = "别复制到我这里";
	char arr2[] = "别把我复制过去";
	printf("%s\n",my_strcpy(arr1,arr2));
	return 0;
}

现在离10分答案只差一步了

我们先了解一下const

const可以修饰变量和指针

image-20211121220154632

修饰变量的时候变量变成了常变量,不能让你改变值,但是可以通过指针改变

	const int n = 10;
	int* p = &n;

那么怎么不让被指针修改呢?

const 放在*的左边
const修饰的指针指向的内容,表示指针指向的内容不能通过指针来改变
但是指针变量本身是可以改变的

	const int*  p = &n;
	*p = 20;
//err

const 放在*的右边
const修饰的是指针变量本身,指针变量的内容不能被修改
但是指针指向的内容是可以通过指针来改变的

	int* const p = &n;
	p = &m;
//err

10分答案

char* my_strcpy(char*dest,const char*src)
{
	assert(dest&&src);
    char*ret=dest;
		while (*dest++ = *src++)
		{
			;
		}
    return ret;
}
int main()
{
	//strcpy
	char arr1[] = "别复制到我这里";
	char arr2[] = "别把我复制过去";
	printf("%s\n",my_strcpy(arr1,arr2));
	return 0;
}

const使得源头的数据不能被改变,这样的话源头更加安全,更加准确

拓展练习

实现strlen

#include <stdio.h>
int my_strlen(const char* str)
{
	int count = 0;
	assert(str != NULL);
	while (*str)
	{
		count++;
		str++;
	}
	return count;
}
int main()
{
	const char* p = "abcdef";
	int len = my_strlen(p);
	printf("len = %d\n", len);
	return 0;
}

小结:

简单的就strcpy函数的实现来展开关于写优秀代码的问题,当然要写出优秀的代码肯定不止这些,strcpy还是很简单的,未来会遇到更加复杂的代码,应该积累经验,多多练习

如果老铁们有收获的话,希望给个一键三连哦,谢谢


点击全文阅读


本文链接:http://m.zhangshiyu.com/post/39831.html

指针  答案  代码  
<< 上一篇 下一篇 >>

  • 评论(0)
  • 赞助本站

◎欢迎参与讨论,请在这里发表您的看法、交流您的观点。

关于我们 | 我要投稿 | 免责申明

Copyright © 2020-2022 ZhangShiYu.com Rights Reserved.豫ICP备2022013469号-1