Linux读取一行文本的方法

本文介绍了使用C语言进行文件读取的两种方法,包括打印文件内容及获取指定行数的内容。此外,还展示了如何从文本中提取特定信息,并提供了一个具体的例子。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

第一种方法:
1.网上流行的一个方法: 打印出每行的文本信息。

#include <stdio.h>
#include <stdlib.h>

int main(void)
{
     FILE * fp;
     char * line = NULL;
     size_t len = 0;
     ssize_t read;

     fp = fopen("test.txt", "r");
     if (fp == NULL)
         exit(EXIT_FAILURE);

     while ((read = getline(&line, &len, fp)) != -1) {
         printf("Retrieved line of length %zu :\n", read);
         printf("%s", line);
     }
     
     if (line)
         free(line);
     exit(EXIT_SUCCESS);
}
2.稍作修改,取得指定行数的文本信息。
#include <stdio.h>
#include <stdlib.h>

int get_file_line(char *pInputName, char *pOutputBuf, int cnt)
{
	FILE * fp;
	int i=0;
	char * line = NULL;
	size_t len = 0;
	ssize_t read;

	fp = fopen(pInputName, "r");
	if (fp == NULL)
	 return -1;

	if(cnt<=0)
		 return -2;
		 
	while ((read = getline(&line, &len, fp)) != -1) {
		++i;
		if(i>=cnt)
			break;
	}

	if (line)
	{
		memcpy(pOutputBuf,line,strlen(line));
		free(line);
		return 0; 
	}

	return -3;
}

int main(void)
{
	char name[20]="test.txt";
	char buf[256]={0};
	int j;
	for(j=1;j<8;++j)
	{
		memset(buf,0,256);
		get_file_line(name,buf,j);
		printf("[%d] == %s",j,buf);
	}
	printf("\n");
  
}

 在此处做一个小小的拓展,就是在每行的文本信息中提取我们想要的信息:

#define dIndexName "WWW"

#define NAME_LEN_MAX 7

void GetName(void)
{
	int i;
	
	char *p=NULL;
	char line[30]={0};
	char dest_name[20]={0};
	char pFileName[256]="test.txt";
	
	get_file_line(pFileName,line,1);


	printf("line === %s \n",line);
	if((p=strstr(line,dIndexName))&&(p=strchr(line,':')))
	{
		memcpy(dest_name,p+1,NAME_LEN_MAX);
	}


	
	for(i=0;i<NAME_LEN_MAX;++i)
	{
		printf("[%d] === %c \n",i,dest_name[i]);
	}
}

执行结果,如下所示:

第二种方法:
#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>

typedef unsigned char u8;

int get_file_line(char *pInputName, int line)
{
	char pTempString[256];

	u8 *pTempBuffer;
	int tTempFile,tTempFileLen;

	int i,m,n;
	u8 tTempFlag,tTempCmdFlag;
	{
		tTempFile=open(pInputName,O_RDONLY);
		if(tTempFile>=0)
		{
			tTempFileLen=lseek(tTempFile,0,SEEK_END); 
			lseek(tTempFile,0,SEEK_SET); 
			tTempCmdFlag=1;
			if( NULL!=(pTempBuffer=(u8 *)malloc(tTempFileLen)) )
			{
				if( tTempFileLen==read(tTempFile,pTempBuffer,tTempFileLen) )
				{
					m=n=0;
					tTempFlag=0;
					while(n<tTempFileLen)
					{
						if( tTempFlag )
						{
							if( pTempBuffer[n]<0x20)
							{
								if( tTempFlag )
								{
									if( n>m )
									{	
										memcpy(pTempString,pTempBuffer+m,n-m);
										pTempString[n-m]=0;
										printf("String1 %s \n",pTempString);
									}	
									tTempFlag=0;
								}
							}
						}else if( pTempBuffer[n]>=0x20 )
						{
							m=n;						
							tTempFlag=1;
						}
						n++;
					}
					if( (n>m)&&(tTempFlag) )
					{
						memcpy(pTempString,pTempBuffer+m,n-m);
						pTempString[n-m]=0;
						printf("String2 %s \n",pTempString);
					}			
					tTempCmdFlag=0;
				}
				free(pTempBuffer);
			}
			close(tTempFile);			
			return tTempCmdFlag;
		}
	}	
	return 2;
}

void main(void)
{
		char name[20]="tmp.txt";
		get_file_line(name,0);
		
}



转载于:https://my.oschina.net/mavericsoung/blog/138885

<think>好的,用户现在想要在Linux环境下读取文本文件的最后一行,需要示例代码或命令。我需要结合之前提供的引用信息,找出合适的方法。 首先,用户提到的引用[1]里提到了使用sed命令来处理文本,比如删除行或替换内容。其中提到sed的-e选项可以执行多个编辑任务,但用户需要的是读取最后一行,可能需要使用sed的地址定位功能。比如,sed -n '$p'可以打印最后一行,因为$代表最后一行,-n选项抑制默认输出,p命令打印匹配的行。 然后,引用[3]中提到了awk的使用,特别是NR变量表示当前行号。可以用awk 'END{print}'来在处理完所有行后打印最后一行,END模式会在所有输入行处理完后执行。 另外,用户可能需要高效的方法,特别是处理大文件时。引用[4]提到mlocate在更新数据库时的高效性,但这里不直接相关。不过,考虑到大文件,使用tail命令会更高效,因为tail直接跳到文件末尾,不需要读取整个文件。例如,tail -n 1 filename直接输出最后一行,这是最常用的方法。 用户还可能关心其他方法,比如用tac反向输出文件,然后取第一行,即tac file | head -n 1,不过这可能对大文件不太高效,因为需要反向整个文件。 需要确保给出的方法涵盖常见的命令,如tail、sed、awk,并比较它们的优缺点。比如,tail是最简单高效的,sed和awk适用于脚本中的处理,而tac+head可能在某些情况下有用,但效率可能不高。 还需要注意,用户可能需要代码示例,比如在Bash脚本中使用这些命令,或者用C语言实现类似功能。但根据用户的问题,主要关注的是命令行工具,所以可能不需要深入编程部分,但如果有示例代码会更全面。 检查之前的回答是否有相关引用,比如引用[1]中的sed用法,引用[3]中的awk,以及可能提到的性能考虑。需要确保方法正确,并引用相关来源,比如在说明sed方法时引用[1],awk引用[3],而tail可能没有直接引用,但属于常见命令。 最后,整理这些方法,按效率和常见程度排序,给出示例命令,并说明注意事项,比如处理空文件或没有换行符的情况,可能需要添加参数,如tail的-n选项,或者使用awk的END块。</think>### Linux读取文本文件最后一行的5种方法Linux系统中读取文本文件最后一行有多种高效方法,根据文件大小和场景需求选择合适方案: --- #### **方法1:`tail`命令(推荐)** **原理**:直接定位文件末尾,时间复杂度$O(1)$,适用于GB级大文件 **代码**: ```bash tail -n 1 filename.txt ``` **优化参数**: ```bash tail -n 1 -- "-" < filename.txt # 处理以"-"开头的特殊文件名 ``` --- #### **方法2:`sed`流编辑器** **原理**:通过地址定位符`$`选择最后一行,时间复杂度$O(n)$ **代码**(引用[1]): ```bash sed -n '$p' filename.txt ``` **扩展应用**:结合正则表达式过滤内容 ```bash sed -n '/error/ {$p}' logfile.log # 输出含"error"的最后一行 ``` --- #### **方法3:`awk`文本处理器** **原理**:遍历所有行后输出最后一条记录,支持复杂逻辑(引用[3]) **代码**: ```bash awk 'END {print}' filename.txt ``` **增强版**:处理含空行的文件 ```bash awk 'END {if (NR>0) print $0}' filename.txt ``` --- #### **方法4:`tac`反向输出组合** **原理**:反向输出文件后取首行,适合小文件 **代码**: ```bash tac filename.txt | head -n 1 ``` --- #### **方法5:Python单行脚本** **原理**:使用文件对象的`seek`方法直接定位末尾 **代码**: ```python python3 -c "with open('filename.txt') as f: print(f.read().splitlines()[-1]) if f.read(1) else ''" ``` --- ### 性能对比表 | 方法 | 时间复杂度 | 适用场景 | 内存占用 | |---------|------------|-------------------|----------| | `tail` | $O(1)$ | 大文件/实时日志 | 最低 | | `sed` | $O(n)$ | 中等文件/简单过滤 | 中等 | | `awk` | $O(n)$ | 需要条件判断 | 中等 | | `tac` | $O(n)$ | 小文件 | 高 | | Python | $O(n)$ | 复杂预处理 | 高 | --- ### 注意事项 1. **空文件处理**:使用`tail -n 1`时若文件为空会输出空行,建议先检查文件大小 2. **行尾符规范**:Windows换行符(`\r\n`)需用`dos2unix`转换,否则可能截断内容 3. **内存映射优化**:处理超大型文件时,可通过`mmap`实现零拷贝读取(C语言示例): ```c #include <sys/mman.h> // 使用mmap映射文件末尾1KB内容后反向查找'\n' ``` --- ### 扩展阅读 1. 如何实时监控日志文件最后一行变化? ```bash tail -f filename.log | grep --line-buffered "ERROR" ``` 2. 怎样高效删除文件的最后一行? ```bash head -n -1 filename.txt > tmpfile && mv tmpfile filename.txt ``` [^1]: sed流编辑器定位行技术 [^3]: awk的END模式处理机制
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值