zip是壓縮后的,你直接讀取就是亂碼,沒有能直接讀取的,想讀取真實內(nèi)容,必須先按解壓算法解壓出來。
專注于為中小企業(yè)提供成都網(wǎng)站設(shè)計、網(wǎng)站建設(shè)、外貿(mào)網(wǎng)站建設(shè)服務,電腦端+手機端+微信端的三站合一,更高效的管理,為中小企業(yè)惠水免費做網(wǎng)站提供優(yōu)質(zhì)的服務。我們立足成都,凝聚了一批互聯(lián)網(wǎng)行業(yè)人才,有力地推動了近千家企業(yè)的穩(wěn)健成長,幫助中小企業(yè)通過網(wǎng)站建設(shè)實現(xiàn)規(guī)模擴充和轉(zhuǎn)變。
簡單一點的 直接調(diào)用系統(tǒng)命令
比如system("unzip ___filename___ -d target_path");
這樣做 實際就相當于在命令行下敲了這樣的一個命令
不過 這個要求系統(tǒng)內(nèi) 必須安裝了unzip這個軟件 即支持這個命令
復雜一些的 你可以再網(wǎng)上找一個開源的gzip 很多很好找的 然后看懂代碼 調(diào)用對應的接口函數(shù)
這樣做的好處不需要依賴系統(tǒng)環(huán)境
利用哈夫曼編碼制作壓縮軟件,內(nèi)容如下:
#include stdio.h
#include string.h
#include stdlib.h
#include conio.h
struct head
{
unsigned char b; //記錄字符在數(shù)組中的位置
long count; //字符出現(xiàn)頻率(權(quán)值)
long parent,lch,rch; //定義哈夫曼樹指針變量
char bits[256]; //定義存儲哈夫曼編碼的數(shù)組
}
header[512],tmp;
/*壓縮*/
void compress()
{
char filename[255],outputfile[255],buf[512];
unsigned char c;
long i,j,m,n,f;
long min1,pt1,flength,length1,length2;
double div;
FILE *ifp,*ofp;
printf("\t請您輸入需要壓縮的文件:");
gets(filename);
ifp=fopen(filename,"rb");
if(ifp==NULL)
{
printf("\n\t文件打開失敗!\n\n");
return;
}
printf("\t請您輸入壓縮后的文件名:");
gets(outputfile);
ofp=fopen(strcat(outputfile,".hub"),"wb");
if(ofp==NULL)
{
printf("\n\t壓縮文件失敗!\n\n");
return;
}
flength=0;
while(!feof(ifp))
{
fread(c,1,1,ifp);
header[c].count++; //字符重復出現(xiàn)頻率+1
flength++; //字符出現(xiàn)原文件長度+1
}
flength--;
length1=flength; //原文件長度用作求壓縮率的分母
header[c].count--;
for(i=0;i512;i++)
{
if(header[i].count!=0) header[i].b=(unsigned char)i;
/*將每個哈夫曼碼值及其對應的ASCII碼存放在一維數(shù)組header[i]中,
且編碼表中的下標和ASCII碼滿足順序存放關(guān)系*/
else header[i].b=0;
header[i].parent=-1;header[i].lch=header[i].rch=-1; //對結(jié)點進行初始化
}
for(i=0;i256;i++) //根據(jù)頻率(權(quán)值)大小,對結(jié)點進行排序,選擇較小的結(jié)點進樹
{
for(j=i+1;j256;j++)
{
if(header[i].countheader[j].count)
{
tmp=header[i];
header[i]=header[j];
header[j]=tmp;
}
}
}
for(i=0;i256;i++) if(header[i].count==0) break;
n=i; //外部葉子結(jié)點數(shù)為n個時,內(nèi)部結(jié)點數(shù)為n-1,整個哈夫曼樹的需要的結(jié)點數(shù)為2*n-1.
m=2*n-1;
for(i=n;im;i++) //構(gòu)建哈夫曼樹
{
min1=999999999; //預設(shè)的最大權(quán)值,即結(jié)點出現(xiàn)的最大次數(shù)
for(j=0;ji;j++)
{
if(header[j].parent!=-1) continue;
//parent!=-1說明該結(jié)點已存在哈夫曼樹中,跳出循環(huán)重新選擇新結(jié)點*/
if(min1header[j].count)
{
pt1=j;
min1=header[j].count;
continue;
}
}
header[i].count=header[pt1].count;
header[pt1].parent=i; //依據(jù)parent域值(結(jié)點層數(shù))確定樹中結(jié)點之間的關(guān)系
header[i].lch=pt1; //計算左分支權(quán)值大小
min1=999999999;
for(j=0;ji;j++)
{
if(header[j].parent!=-1) continue;
if(min1header[j].count)
{
pt1=j;
min1=header[j].count;
continue;
}
}
header[i].count+=header[pt1].count;
header[i].rch=pt1; //計算右分支權(quán)值大小
header[pt1].parent=i;
}
for(i=0;in;i++) //哈夫曼無重復前綴編碼
{
f=i;
header[i].bits[0]=0; //根結(jié)點編碼0
while(header[f].parent!=-1)
{
j=f;
f=header[f].parent;
if(header[f].lch==j) //置左分支編碼0
{
j=strlen(header[i].bits);
memmove(header[i].bits+1,header[i].bits,j+1);
//依次存儲連接“0”“1”編碼
header[i].bits[0]='0';
}
else //置右分支編碼1
{
j=strlen(header[i].bits);
memmove(header[i].bits+1,header[i].bits,j+1);
header[i].bits[0]='1';
}
}
}
fseek(ifp,0,SEEK_SET); //從文件開始位置向前移動0字節(jié),即定位到文件開始位置
fwrite(flength,sizeof(int),1,ofp);
/*用來將數(shù)據(jù)寫入文件流中,參數(shù)flength指向欲寫入的數(shù)據(jù)地址,
總共寫入的字符數(shù)以參數(shù)size*int來決定,返回實際寫入的int數(shù)目1*/
fseek(ofp,8,SEEK_SET);
buf[0]=0; //定義緩沖區(qū),它的二進制表示00000000
f=0;
pt1=8;
/*假設(shè)原文件第一個字符是"A",8位2進制為01000001,編碼后為0110識別編碼第一個'0',
那么我們就可以將其左移一位,看起來沒什么變化。下一個是'1',應該|1,結(jié)果00000001
同理4位都做完,應該是00000110,由于字節(jié)中的8位并沒有全部用完,我們應該繼續(xù)讀下一個字符,
根據(jù)編碼表繼續(xù)拼完剩下的4位,如果字符的編碼不足4位,還要繼續(xù)讀一個字符,
如果字符編碼超過4位,那么我們將把剩下的位信息拼接到一個新的字節(jié)里*/
while(!feof(ifp))
{
c=fgetc(ifp);
f++;
for(i=0;in;i++)
{
if(c==header[i].b) break;
}
strcat(buf,header[i].bits);
j=strlen(buf);
c=0;
while(j=8) //對哈夫曼編碼位操作進行壓縮存儲
{
for(i=0;i8;i++)
{
if(buf[i]=='1') c=(c1)|1;
else c=c1;
}
fwrite(c,1,1,ofp);
pt1++; //統(tǒng)計壓縮后文件的長度
strcpy(buf,buf+8); //一個字節(jié)一個字節(jié)拼接
j=strlen(buf);
}
if(f==flength) break;
}
if(j0) //對哈夫曼編碼位操作進行壓縮存儲
{
strcat(buf,"00000000");
for(i=0;i8;i++)
{
if(buf[i]=='1') c=(c1)|1;
else c=c1;
}
fwrite(c,1,1,ofp);
pt1++;
}
fseek(ofp,4,SEEK_SET);
fwrite(pt1,sizeof(long),1,ofp);
fseek(ofp,pt1,SEEK_SET);
fwrite(n,sizeof(long),1,ofp);
for(i=0;in;i++)
{
fwrite((header[i].b),1,1,ofp);
c=strlen(header[i].bits);
fwrite(c,1,1,ofp);
j=strlen(header[i].bits);
if(j%8!=0) //若存儲的位數(shù)不是8的倍數(shù),則補0
{
for(f=j%8;f8;f++)
strcat(header[i].bits,"0");
}
while(header[i].bits[0]!=0)
{
c=0;
for(j=0;j8;j++) //字符的有效存儲不超過8位,則對有效位數(shù)左移實現(xiàn)兩字符編碼的連接
{
if(header[i].bits[j]=='1') c=(c1)|1; //|1不改變原位置上的“0”“1”值
else c=c1;
}
strcpy(header[i].bits,header[i].bits+8); //把字符的編碼按原先存儲順序連接
fwrite(c,1,1,ofp);
}
}
length2=pt1--;
div=((double)length1-(double)length2)/(double)length1; //計算文件的壓縮率
fclose(ifp);
fclose(ofp);
printf("\n\t壓縮文件成功!\n");
printf("\t壓縮率為 %f%%\n\n",div*100);
return;
}
/*解壓縮*/
void uncompress()
{
char filename[255],outputfile[255],buf[255],bx[255];
unsigned char c;
long i,j,m,n,f,p,l;
long flength;
FILE *ifp,*ofp;
printf("\t請您輸入需要解壓縮的文件:");
gets(filename);
ifp=fopen(strcat(filename,".hub"),"rb");
if(ifp==NULL)
{
printf("\n\t文件打開失敗!\n");
return;
}
printf("\t請您輸入解壓縮后的文件名:");
gets(outputfile);
ofp=fopen(outputfile,"wb");
if(ofp==NULL)
{
printf("\n\t解壓縮文件失敗!\n");
return;
}
fread(flength,sizeof(long),1,ifp); //讀取原文件長度,對文件進行定位
fread(f,sizeof(long),1,ifp);
fseek(ifp,f,SEEK_SET);
fread(n,sizeof(long),1,ifp);
for(i=0;in;i++)
{
fread(header[i].b,1,1,ifp);
fread(c,1,1,ifp);
p=(long)c; //讀取原文件字符的權(quán)值
header[i].count=p;
header[i].bits[0]=0;
if(p%80) m=p/8+1;
else m=p/8;
for(j=0;jm;j++)
{
fread(c,1,1,ifp);
f=c;
itoa(f,buf,2); //將f轉(zhuǎn)換為二進制表示的字符串
f=strlen(buf);
for(l=8;lf;l--)
{
strcat(header[i].bits,"0");
}
strcat(header[i].bits,buf);
}
header[i].bits[p]=0;
}
for(i=0;in;i++) //根據(jù)哈夫曼編碼的長短,對結(jié)點進行排序
{
for(j=i+1;jn;j++)
{
if(strlen(header[i].bits)strlen(header[j].bits))
{
tmp=header[i];
header[i]=header[j];
header[j]=tmp;
}
}
}
p=strlen(header[n-1].bits);
fseek(ifp,8,SEEK_SET);
m=0;
bx[0]=0;
while(1) //通過哈夫曼編碼的長短,依次解碼,從原來的位存儲還原到字節(jié)存儲
{
while(strlen(bx)(unsigned int)p)
{
fread(c,1,1,ifp);
f=c;
itoa(f,buf,2);
f=strlen(buf);
for(l=8;lf;l--) //在單字節(jié)內(nèi)對相應位置補0
{
strcat(bx,"0");
}
strcat(bx,buf);
}
for(i=0;in;i++)
{
if(memcmp(header[i].bits,bx,header[i].count)==0) break;
}
strcpy(bx,bx+header[i].count); /*從壓縮文件中的按位存儲還原到按字節(jié)存儲字符,
字符位置不改變*/
c=header[i].b;
fwrite(c,1,1,ofp);
m++; //統(tǒng)計解壓縮后文件的長度
if(m==flength) break; //flength是原文件長度
}
fclose(ifp);
fclose(ofp);
printf("\n\t解壓縮文件成功!\n");
if(m==flength) //對解壓縮后文件和原文件相同性比較進行判斷(根據(jù)文件大?。?/p>
printf("\t解壓縮文件與原文件相同!\n\n");
else printf("\t解壓縮文件與原文件不同!\n\n");
return;
}
/*主函數(shù)*/
int main()
{
int c;
while(1) //菜單工具欄
{
printf("\t _______________________________________________\n");
printf("\n");
printf("\t * 壓縮、解壓縮 小工具 * \n");
printf("\t _______________________________________________\n");
printf("\t _______________________________________________\n");
printf("\t| |\n");
printf("\t| 1.壓縮 |\n");
printf("\t| 2.解壓縮 |\n");
printf("\t| 0.退出 |\n");
printf("\t|_______________________________________________|\n");
printf("\n");
printf("\t 說明:(1)采用哈夫曼編碼\n");
printf("\t (2)適用于文本文件\n");
printf("\n");
do //對用戶輸入進行容錯處理
{
printf("\n\t*請選擇相應功能(0-2):");
c=getch();
printf("%c\n",c);
if(c!='0' c!='1' c!='2')
{
printf("\t@_@請檢查您的輸入在0~2之間!\n");
printf("\t請再輸入一遍!\n");
}
}while(c!='0' c!='1' c!='2');
if(c=='1') compress(); //調(diào)用壓縮子函數(shù)
else if(c=='2') uncompress(); //調(diào)用解壓縮子函數(shù)
else
{
printf("\t歡迎您再次使用該工具^_^\n");
exit(0); //退出該工具
}
system("pause"); //任意鍵繼續(xù)
system("cls"); //清屏
}
return 0;
}
定義:zip([iterable, ...])
zip()是Python的一個內(nèi)建函數(shù),它接受一系列可迭代的對象作為參數(shù),將對象中對應的元素打包成一個個tuple(元組),然后返回由這些 tuples組成的list(列表)。若傳入?yún)?shù)的長度不等,則返回list的長度和參數(shù)中長度最短的對象相同。利用*號操作符,可以將list unzip(解壓),看下面的例子就明白了:
1 2 3 4 5 6 7 8 9
a = [1,2,3] b = [4,5,6] c = [4,5,6,7,8] zipped = zip(a,b) [(1, 4), (2, 5), (3, 6)] zip(a,c) [(1, 4), (2, 5), (3, 6)] zip(*zipped) [(1, 2, 3), (4, 5, 6)]
對于這個并不是很常用函數(shù),下面舉幾個例子說明它的用法:
* 二維矩陣變換(矩陣的行列互換)
比如我們有一個由列表描述的二維矩陣
a = [[1, 2, 3], [4, 5, 6], [7, 8, 9]]
通過python列表推導的方法,我們也能輕易完成這個任務
1 2
print [ [row[col] for row in a] for col in range(len(a[0]))] [[1, 4, 7], [2, 5, 8], [3, 6, 9]]
另外一種讓人困惑的方法就是利用zip函數(shù):
1 2 3 4 5
a = [[1, 2, 3], [4, 5, 6], [7, 8, 9]] zip(*a) [(1, 4, 7), (2, 5, 8), (3, 6, 9)] map(list,zip(*a)) [[1, 4, 7], [2, 5, 8], [3, 6, 9]]
這種方法速度更快但也更難以理解,將list看成tuple解壓,恰好得到我們“行列互換”的效果,再通過對每個元素應用list()函數(shù),將tuple轉(zhuǎn)換為list
* 以指定概率獲取元素
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27
import random def random_pick(seq,probabilities): x = random.uniform(0, 1) cumulative_probability = 0.0 for item, item_probability in zip(seq, probabilities): cumulative_probability += item_probability if x cumulative_probability: break return item for i in range(15): random_pick("abc",[0.1,0.3,0.6]) 'c' 'b' 'c' 'c' 'a' 'b' 'c' 'c' 'c' 'a' 'b' 'b' 'c' 'a' 'c'
這個函數(shù)有個限制,指定概率的列表必須和元素一一對應,而且和為1,否則這個函數(shù)可能不能像預想的那樣工作。
稍微解釋下,先利用random.uniform()函數(shù)生成一個0-1之間的隨機數(shù)并復制給x,利用zip()函數(shù)將元素和他對應的概率打包成tuple,然后將每個元素的概率進行疊加,直到和大于x終止循環(huán)
這樣,”a”被選中的概率就是x取值位于0-0.1的概率,同理”b”為0.1-0.4,”c”為0.4-1.0,假設(shè)x是在0-1之間平均取值的,顯然我們的目的已經(jīng)達到
我的建議是去看MSDN,里面有詳細的庫函數(shù)的說明和使用方法。C的庫函數(shù)有很多,一條條說不現(xiàn)實,就是一個專業(yè)的程序員也不可能將庫函數(shù)都記下來。我從網(wǎng)上搜了點庫函數(shù)的用法,你看看吧。還是那句話,想了解庫函數(shù),還是看MSDN。
下面也是我從網(wǎng)上找到的。網(wǎng)上這種東西很多。但我認為要是真正精通,還是MSDN
文件的打開(fopen函數(shù))
fopen函數(shù)用來打開一個文件,其調(diào)用的一般形式為:
文件指針名=fopen(文件名,使用文件方式);
其中,
“文件指針名”必須是被說明為FILE 類型的指針變量;
“文件名”是被打開文件的文件名;
“使用文件方式”是指文件的類型和操作要求。
“文件名”是字符串常量或字符串數(shù)組。
例如:
FILE *fp;
fp=("file a","r");
其意義是在當前目錄下打開文件file a,只允許進行“讀”操作,并使fp指向該文件。
又如:
FILE *fphzk
fphzk=("c:\\hzk16","rb")
其意義是打開C驅(qū)動器磁盤的根目錄下的文件hzk16,這是一個二進制文件,只允許按二進制方式進行讀操作。兩個反斜線“\\ ”中的第一個表示轉(zhuǎn)義字符,第二個表示根目錄。
使用文件的方式共有12種,下面給出了它們的符號和意義。
文件使用方式
意義
“rt”
只讀打開一個文本文件,只允許讀數(shù)據(jù)
“wt”
只寫打開或建立一個文本文件,只允許寫數(shù)據(jù)
“at”
追加打開一個文本文件,并在文件末尾寫數(shù)據(jù)
“rb”
只讀打開一個二進制文件,只允許讀數(shù)據(jù)
“wb”
只寫打開或建立一個二進制文件,只允許寫數(shù)據(jù)
“ab”
追加打開一個二進制文件,并在文件末尾寫數(shù)據(jù)
“rt+”
讀寫打開一個文本文件,允許讀和寫
“wt+”
讀寫打開或建立一個文本文件,允許讀寫
“at+”
讀寫打開一個文本文件,允許讀,或在文件末追加數(shù)據(jù)
“rb+”
讀寫打開一個二進制文件,允許讀和寫
“wb+”
讀寫打開或建立一個二進制文件,允許讀和寫
“ab+”
讀寫打開一個二進制文件,允許讀,或在文件末追加數(shù)據(jù)
對于文件使用方式有以下幾點說明:
1) 文件使用方式由r,w,a,t,b,+六個字符拼成,各字符的含義是:
r(read): 讀
w(write): 寫
a(append): 追加
t(text): 文本文件,可省略不寫
b(banary): 二進制文件
+: 讀和寫
2) 凡用“r”打開一個文件時,該文件必須已經(jīng)存在,且只能從該文件讀出。
3) 用“w”打開的文件只能向該文件寫入。若打開的文件不存在,則以指定的文件名建立該文件,若打開的文件已經(jīng)存在,則將該文件刪去,重建一個新文件。
4) 若要向一個已存在的文件追加新的信息,只能用“a”方式打開文件。但此時該文件必須是存在的,否則將會出錯。
5) 在打開一個文件時,如果出錯,fopen將返回一個空指針值NULL。在程序中可以用這一信息來判別是否完成打開文件的工作,并作相應的處理。因此常用以下程序段打開文件:
6) if((fp=fopen("c:\\hzk16","rb")==NULL)
{
printf("\nerror on open c:\\hzk16 file!");
getch();
exit(1);
}
這段程序的意義是,如果返回的指針為空,表示不能打開C盤根目錄下的hzk16文件,則給出提示信息“error on open c:\ hzk16 file!”,下一行g(shù)etch()的功能是從鍵盤輸入一個字符,但不在屏幕上顯示。在這里,該行的作用是等待,只有當用戶從鍵盤敲任一鍵時,程序才繼續(xù)執(zhí)行,因此用戶可利用這個等待時間閱讀出錯提示。敲鍵后執(zhí)行exit(1)退出程序。
7) 把一個文本文件讀入內(nèi)存時,要將ASCII碼轉(zhuǎn)換成二進制碼,而把文件以文本方式寫入磁盤時,也要把二進制碼轉(zhuǎn)換成ASCII碼,因此文本文件的讀寫要花費較多的轉(zhuǎn)換時間。對二進制文件的讀寫不存在這種轉(zhuǎn)換。
8) 標準輸入文件(鍵盤),標準輸出文件(顯示器),標準出錯輸出(出錯信息)是由系統(tǒng)打開的,可直接使用。
13.3.2 文件關(guān)閉函數(shù)(fclose函數(shù))
文件一旦使用完畢,應用關(guān)閉文件函數(shù)把文件關(guān)閉,以避免文件的數(shù)據(jù)丟失等錯誤。
fclose函數(shù)調(diào)用的一般形式是:
fclose(文件指針);
例如:
fclose(fp);
正常完成關(guān)閉文件操作時,fclose函數(shù)返回值為0。如返回非零值則表示有錯誤發(fā)生。
13.4 文件的讀寫
對文件的讀和寫是最常用的文件操作。在C語言中提供了多種文件讀寫的函數(shù):
·字符讀寫函數(shù) :fgetc和fputc
·字符串讀寫函數(shù):fgets和fputs
·數(shù)據(jù)塊讀寫函數(shù):freed和fwrite
·格式化讀寫函數(shù):fscanf和fprinf
下面分別予以介紹。使用以上函數(shù)都要求包含頭文件stdio.h。
13.4.1 字符讀寫函數(shù)fgetc和fputc
字符讀寫函數(shù)是以字符(字節(jié))為單位的讀寫函數(shù)。每次可從文件讀出或向文件寫入一個字符。
1. 讀字符函數(shù)fgetc
fgetc函數(shù)的功能是從指定的文件中讀一個字符,函數(shù)調(diào)用的形式為:
字符變量=fgetc(文件指針);
例如:
ch=fgetc(fp);
其意義是從打開的文件fp中讀取一個字符并送入ch中。
對于fgetc函數(shù)的使用有以下幾點說明:
1) 在fgetc函數(shù)調(diào)用中,讀取的文件必須是以讀或讀寫方式打開的。
2) 讀取字符的結(jié)果也可以不向字符變量賦值,
例如:
fgetc(fp);
但是讀出的字符不能保存。
3) 在文件內(nèi)部有一個位置指針。用來指向文件的當前讀寫字節(jié)。在文件打開時,該指針總是指向文件的第一個字節(jié)。使用fgetc 函數(shù)后,該位置指針將向后移動一個字節(jié)。 因此可連續(xù)多次使用fgetc函數(shù),讀取多個字符。應注意文件指針和文件內(nèi)部的位置指針不是一回事。文件指針是指向整個文件的,須在程序中定義說明,只要不重新賦值,文件指針的值是不變的。文件內(nèi)部的位置指針用以指示文件內(nèi)部的當前讀寫位置,每讀寫一次,該指針均向后移動,它不需在程序中定義說明,而是由系統(tǒng)自動設(shè)置的。
【例13.1】讀入文件c1.doc,在屏幕上輸出。
#includestdio.h
main()
{
FILE *fp;
char ch;
if((fp=fopen("d:\\jrzh\\example\\c1.txt","rt"))==NULL)
{
printf("\nCannot open file strike any key exit!");
getch();
exit(1);
}
ch=fgetc(fp);
while(ch!=EOF)
{
putchar(ch);
ch=fgetc(fp);
}
fclose(fp);
}
本例程序的功能是從文件中逐個讀取字符,在屏幕上顯示。程序定義了文件指針fp,以讀文本文件方式打開文件“d:\\jrzh\\example\\ex1_1.c”,并使fp指向該文件。如打開文件出錯,給出提示并退出程序。程序第12行先讀出一個字符,然后進入循環(huán),只要讀出的字符不是文件結(jié)束標志(每個文件末有一結(jié)束標志EOF)就把該字符顯示在屏幕上,再讀入下一字符。每讀一次,文件內(nèi)部的位置指針向后移動一個字符,文件結(jié)束時,該指針指向EOF。執(zhí)行本程序?qū)@示整個文件。
2. 寫字符函數(shù)fputc
fputc函數(shù)的功能是把一個字符寫入指定的文件中,函數(shù)調(diào)用的形式為:
fputc(字符量,文件指針);
其中,待寫入的字符量可以是字符常量或變量,例如:
fputc('a',fp);
其意義是把字符a寫入fp所指向的文件中。
對于fputc函數(shù)的使用也要說明幾點:
1) 被寫入的文件可以用寫、讀寫、追加方式打開,用寫或讀寫方式打開一個已存在的文件時將清除原有的文件內(nèi)容,寫入字符從文件首開始。如需保留原有文件內(nèi)容,希望寫入的字符以文件末開始存放,必須以追加方式打開文件。被寫入的文件若不存在,則創(chuàng)建該文件。
2) 每寫入一個字符,文件內(nèi)部位置指針向后移動一個字節(jié)。
3) fputc函數(shù)有一個返回值,如寫入成功則返回寫入的字符,否則返回一個EOF??捎么藖砼袛鄬懭胧欠癯晒Α?/p>
【例13.2】從鍵盤輸入一行字符,寫入一個文件,再把該文件內(nèi)容讀出顯示在屏幕上。
#includestdio.h
main()
{
FILE *fp;
char ch;
if((fp=fopen("d:\\jrzh\\example\\string","wt+"))==NULL)
{
printf("Cannot open file strike any key exit!");
getch();
exit(1);
}
printf("input a string:\n");
ch=getchar();
while (ch!='\n')
{
fputc(ch,fp);
ch=getchar();
}
rewind(fp);
ch=fgetc(fp);
while(ch!=EOF)
{
putchar(ch);
ch=fgetc(fp);
}
printf("\n");
fclose(fp);
}
程序中第6行以讀寫文本文件方式打開文件string。程序第13行從鍵盤讀入一個字符后進入循環(huán),當讀入字符不為回車符時,則把該字符寫入文件之中,然后繼續(xù)從鍵盤讀入下一字符。每輸入一個字符,文件內(nèi)部位置指針向后移動一個字節(jié)。寫入完畢,該指針已指向文件末。如要把文件從頭讀出,須把指針移向文件頭,程序第19行rewind函數(shù)用于把fp所指文件的內(nèi)部位置指針移到文件頭。第20至25行用于讀出文件中的一行內(nèi)容。
【例13.3】把命令行參數(shù)中的前一個文件名標識的文件,復制到后一個文件名標識的文件中,如命令行中只有一個文件名則把該文件寫到標準輸出文件(顯示器)中。
#includestdio.h
main(int argc,char *argv[])
{
FILE *fp1,*fp2;
char ch;
if(argc==1)
{
printf("have not enter file name strike any key exit");
getch();
exit(0);
}
if((fp1=fopen(argv[1],"rt"))==NULL)
{
printf("Cannot open %s\n",argv[1]);
getch();
exit(1);
}
if(argc==2) fp2=stdout;
else if((fp2=fopen(argv[2],"wt+"))==NULL)
{
printf("Cannot open %s\n",argv[1]);
getch();
exit(1);
}
while((ch=fgetc(fp1))!=EOF)
fputc(ch,fp2);
fclose(fp1);
fclose(fp2);
}
本程序為帶參的main函數(shù)。程序中定義了兩個文件指針fp1和fp2,分別指向命令行參數(shù)中給出的文件。如命令行參數(shù)中沒有給出文件名,則給出提示信息。程序第18行表示如果只給出一個文件名,則使fp2指向標準輸出文件(即顯示器)。程序第25行至28行用循環(huán)語句逐個讀出文件1中的字符再送到文件2中。再次運行時,給出了一個文件名,故輸出給標準輸出文件stdout,即在顯示器上顯示文件內(nèi)容。第三次運行,給出了二個文件名,因此把string中的內(nèi)容讀出,寫入到OK之中。可用DOS命令type顯示OK的內(nèi)容。
13.4.2 字符串讀寫函數(shù)fgets和fputs
1. 讀字符串函數(shù)fgets
函數(shù)的功能是從指定的文件中讀一個字符串到字符數(shù)組中,函數(shù)調(diào)用的形式為:
fgets(字符數(shù)組名,n,文件指針);
其中的n是一個正整數(shù)。表示從文件中讀出的字符串不超過 n-1個字符。在讀入的最后一個字符后加上串結(jié)束標志'\0'。
例如:
fgets(str,n,fp);
的意義是從fp所指的文件中讀出n-1個字符送入字符數(shù)組str中。
【例13.4】從string文件中讀入一個含10個字符的字符串。
#includestdio.h
main()
{
FILE *fp;
char str[11];
if((fp=fopen("d:\\jrzh\\example\\string","rt"))==NULL)
{
printf("\nCannot open file strike any key exit!");
getch();
exit(1);
}
fgets(str,11,fp);
printf("\n%s\n",str);
fclose(fp);
}
本例定義了一個字符數(shù)組str共11個字節(jié),在以讀文本文件方式打開文件string后,從中讀出10個字符送入str數(shù)組,在數(shù)組最后一個單元內(nèi)將加上'\0',然后在屏幕上顯示輸出str數(shù)組。輸出的十個字符正是例13.1程序的前十個字符。
對fgets函數(shù)有兩點說明:
1) 在讀出n-1個字符之前,如遇到了換行符或EOF,則讀出結(jié)束。
2) fgets函數(shù)也有返回值,其返回值是字符數(shù)組的首地址。
2. 寫字符串函數(shù)fputs
fputs函數(shù)的功能是向指定的文件寫入一個字符串,其調(diào)用形式為:
fputs(字符串,文件指針);
其中字符串可以是字符串常量,也可以是字符數(shù)組名,或指針變量,例如:
fputs(“abcd“,fp);
其意義是把字符串“abcd”寫入fp所指的文件之中。
【例13.5】在例13.2中建立的文件string中追加一個字符串。
#includestdio.h
main()
{
FILE *fp;
char ch,st[20];
if((fp=fopen("string","at+"))==NULL)
{
printf("Cannot open file strike any key exit!");
getch();
exit(1);
}
printf("input a string:\n");
scanf("%s",st);
fputs(st,fp);
rewind(fp);
ch=fgetc(fp);
while(ch!=EOF)
{
putchar(ch);
ch=fgetc(fp);
}
printf("\n");
fclose(fp);
}
本例要求在string文件末加寫字符串,因此,在程序第6行以追加讀寫文本文件的方式打開文件string。然后輸入字符串,并用fputs函數(shù)把該串寫入文件string。在程序15行用rewind函數(shù)把文件內(nèi)部位置指針移到文件首。再進入循環(huán)逐個顯示當前文件中的全部內(nèi)容。
13.4.3 數(shù)據(jù)塊讀寫函數(shù)fread和fwtrite
C語言還提供了用于整塊數(shù)據(jù)的讀寫函數(shù)??捎脕碜x寫一組數(shù)據(jù),如一個數(shù)組元素,一個結(jié)構(gòu)變量的值等。
讀數(shù)據(jù)塊函數(shù)調(diào)用的一般形式為:
fread(buffer,size,count,fp);
寫數(shù)據(jù)塊函數(shù)調(diào)用的一般形式為:
fwrite(buffer,size,count,fp);
其中:
buffer 是一個指針,在fread函數(shù)中,它表示存放輸入數(shù)據(jù)的首地址。在fwrite函數(shù)中,它表示存放輸出數(shù)據(jù)的首地址。
size 表示數(shù)據(jù)塊的字節(jié)數(shù)。
count 表示要讀寫的數(shù)據(jù)塊塊數(shù)。
fp 表示文件指針。
例如:
fread(fa,4,5,fp);
其意義是從fp所指的文件中,每次讀4個字節(jié)(一個實數(shù))送入實數(shù)組fa中,連續(xù)讀5次,即讀5個實數(shù)到fa中。
【例13.6】從鍵盤輸入兩個學生數(shù)據(jù),寫入一個文件中,再讀出這兩個學生的數(shù)據(jù)顯示在屏幕上。
#includestdio.h
struct stu
{
char name[10];
int num;
int age;
char addr[15];
}boya[2],boyb[2],*pp,*qq;
main()
{
FILE *fp;
char ch;
int i;
pp=boya;
qq=boyb;
if((fp=fopen("d:\\jrzh\\example\\stu_list","wb+"))==NULL)
{
printf("Cannot open file strike any key exit!");
getch();
exit(1);
}
printf("\ninput data\n");
for(i=0;i2;i++,pp++)
scanf("%s%d%d%s",pp-name,pp-num,pp-age,pp-addr);
pp=boya;
fwrite(pp,sizeof(struct stu),2,fp);
rewind(fp);
fread(qq,sizeof(struct stu),2,fp);
printf("\n\nname\tnumber age addr\n");
for(i=0;i2;i++,qq++)
printf("%s\t%5d%7d %s\n",qq-name,qq-num,qq-age,qq-addr);
fclose(fp);
}
本例程序定義了一個結(jié)構(gòu)stu,說明了兩個結(jié)構(gòu)數(shù)組boya和boyb以及兩個結(jié)構(gòu)指針變量pp和qq。pp指向boya,qq指向boyb。程序第16行以讀寫方式打開二進制文件“stu_list”,輸入二個學生數(shù)據(jù)之后,寫入該文件中,然后把文件內(nèi)部位置指針移到文件首,讀出兩塊學生數(shù)據(jù)后,在屏幕上顯示。
------
回復此文章 |
回復主題:Re:c語言文件操作 | 作者: haohao | 軍銜:六級軍士 | 發(fā)表時間:2004-07-28 11:08:47
下午應該把這個搞定了.
------
回復此文章 |
回復主題:Re:Re:c語言文件操作 | 作者: haohao | 軍銜:六級軍士 | 發(fā)表時間:2004-07-28 11:10:23
13.7 C庫文件
C系統(tǒng)提供了豐富的系統(tǒng)文件,稱為庫文件,C的庫文件分為兩類,一類是擴展名為".h"的文件,稱為頭文件,在前面的包含命令中我們已多次使用過。在".h"文件中包含了常量定義、類型定義、宏定義、函數(shù)原型以及各種編譯選擇設(shè)置等信息。另一類是函數(shù)庫,包括了各種函數(shù)的目標代碼,供用戶在程序中調(diào)用。 通常在程序中調(diào)用一個庫函數(shù)時,要在調(diào)用之前包含該函數(shù)原型所在的".h" 文件。
下面給出Turbo C的全部".h"文件。
Turbo C頭文件
n ALLOC.H 說明內(nèi)存管理函數(shù)(分配、釋放等)。
n ASSERT.H 定義 assert調(diào)試宏。
n BIOS.H 說明調(diào)用IBM—PC ROM BIOS子程序的各個函數(shù)。
n CONIO.H 說明調(diào)用DOS控制臺I/O子程序的各個函數(shù)。
n CTYPE.H 包含有關(guān)字符分類及轉(zhuǎn)換的名類信息(如 isalpha和toascii等)。
n DIR.H 包含有關(guān)目錄和路徑的結(jié)構(gòu)、宏定義和函數(shù)。
n DOS.H 定義和說明MSDOS和8086調(diào)用的一些常量和函數(shù)。
n ERRON.H 定義錯誤代碼的助記符。
n FCNTL.H 定義在與open庫子程序連接時的符號常量。
n FLOAT.H 包含有關(guān)浮點運算的一些參數(shù)和函數(shù)。
n GRAPHICS.H 說明有關(guān)圖形功能的各個函數(shù),圖形錯誤代碼的常量定義,正對不同驅(qū)動程序的各種顏色值,及函數(shù)用到的一些特殊結(jié)構(gòu)。
n IO.H 包含低級I/O子程序的結(jié)構(gòu)和說明。
n LIMIT.H 包含各環(huán)境參數(shù)、編譯時間限制、數(shù)的范圍等信息。
n MATH.H 說明數(shù)學運算函數(shù),還定了 HUGE VAL 宏, 說明了matherr和matherr子程序用到的特殊結(jié)構(gòu)。
n MEM.H 說明一些內(nèi)存操作函數(shù)(其中大多數(shù)也在STRING.H中說明)。
n PROCESS.H 說明進程管理的各個函數(shù),spawn…和EXEC …函數(shù)的結(jié)構(gòu)說明。
n SETJMP.H 定義longjmp和setjmp函數(shù)用到的jmp buf類型,說明這兩個函數(shù)。
n SHARE.H 定義文件共享函數(shù)的參數(shù)。
n SIGNAL.H 定義SIG[ZZ(Z] [ZZ)]IGN和SIG[ZZ(Z] [ZZ)]DFL常量,說明rajse和signal兩個函數(shù)。
n STDARG.H 定義讀函數(shù)參數(shù)表的宏。(如vprintf,vscarf函數(shù))。
n STDDEF.H 定義一些公共數(shù)據(jù)類型和宏。
n STDIO.H 定義Kernighan和Ritchie在Unix System V 中定義的標準和擴展的類型和宏。還定義標準I/O 預定義流:stdin,stdout和stderr,說明 I/O流子程序。
n STDLIB.H 說明一些常用的子程序:轉(zhuǎn)換子程序、搜索/ 排序子程序等。
n STRING.H 說明一些串操作和內(nèi)存操作函數(shù)。
n SYS\STAT.H 定義在打開和創(chuàng)建文件時用到的一些符號常量。
n SYS\TYPES.H 說明ftime函數(shù)和timeb結(jié)構(gòu)。
n SYS\TIME.H 定義時間的類型time[ZZ(Z] [ZZ)]t。
n TIME.H 定義時間轉(zhuǎn)換子程序asctime、localtime和gmtime的結(jié)構(gòu),ctime、 difftime、 gmtime、 localtime和stime用到的類型,并提供這些函數(shù)的原型。
n VALUE.H 定義一些重要常量,包括依賴于機器硬件的和為與Unix System V相兼容而說明的一些常量,包括浮點和雙精度值的范圍。
13.8 本章小結(jié)
1. C系統(tǒng)把文件當作一個“流”,按字節(jié)進行處理。
2. C文件按編碼方式分為二進制文件和ASCII文件。
3. C語言中,用文件指針標識文件,當一個文件被打開時,可取得該文件指針。
4. 文件在讀寫之前必須打開,讀寫結(jié)束必須關(guān)閉。
5. 文件可按只讀、只寫、讀寫、追加四種操作方式打開,同時還必須指定文件的類型是二進制文件還是文本文件。
6. 文件可按字節(jié),字符串,數(shù)據(jù)塊為單位讀寫,文件也可按指定的格式進行讀寫。
7. 文件內(nèi)部的位置指針可指示當前的讀寫位置,移動該指針可以對文件實現(xiàn)隨機讀寫。