Prova:
char *remove(char* mystr) {
char *retstr;
char *lastdot;
if (mystr == NULL)
return NULL;
if ((retstr = malloc (strlen (mystr) + 1)) == NULL)
return NULL;
strcpy (retstr, mystr);
lastdot = strrchr (retstr, '.');
if (lastdot != NULL)
*lastdot = '\0';
return retstr;
}
Dovrete liberare la stringa restituita da soli. Trova semplicemente l'ultimo .
nella stringa e lo sostituisce con un carattere di terminazione null. Gestirà gli errori (passando NULL o esaurendo la memoria) restituendo NULL.
Non funzionerà con le cose come /this.path/is_bad
dal momento che troverà la .
nella parte non-file, ma si poteva gestire questo anche facendo un strrchr
di /
, o qualunque sia il vostro separatore di percorso è, e garantire la sua posizione è NULL o prima della posizione .
.
Una soluzione scopo più generale a questo problema potrebbe essere:
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
// remove_ext: removes the "extension" from a file spec.
// mystr is the string to process.
// dot is the extension separator.
// sep is the path separator (0 means to ignore).
// Returns an allocated string identical to the original but
// with the extension removed. It must be freed when you're
// finished with it.
// If you pass in NULL or the new string can't be allocated,
// it returns NULL.
char *remove_ext (char* mystr, char dot, char sep) {
char *retstr, *lastdot, *lastsep;
// Error checks and allocate string.
if (mystr == NULL)
return NULL;
if ((retstr = malloc (strlen (mystr) + 1)) == NULL)
return NULL;
// Make a copy and find the relevant characters.
strcpy (retstr, mystr);
lastdot = strrchr (retstr, dot);
lastsep = (sep == 0) ? NULL : strrchr (retstr, sep);
// If it has an extension separator.
if (lastdot != NULL) {
// and it's before the extenstion separator.
if (lastsep != NULL) {
if (lastsep < lastdot) {
// then remove it.
*lastdot = '\0';
}
} else {
// Has extension separator with no path separator.
*lastdot = '\0';
}
}
// Return the modified string.
return retstr;
}
int main (int c, char *v[]) {
char *s;
printf ("[%s]\n", (s = remove_ext ("hello", '.', '/'))); free (s);
printf ("[%s]\n", (s = remove_ext ("hello.", '.', '/'))); free (s);
printf ("[%s]\n", (s = remove_ext ("hello.txt", '.', '/'))); free (s);
printf ("[%s]\n", (s = remove_ext ("hello.txt.txt", '.', '/'))); free (s);
printf ("[%s]\n", (s = remove_ext ("/no.dot/in_path", '.', '/'))); free (s);
printf ("[%s]\n", (s = remove_ext ("/has.dot/in.path", '.', '/'))); free (s);
printf ("[%s]\n", (s = remove_ext ("/no.dot/in_path", '.', 0))); free (s);
return 0;
}
e questo produce:
[hello]
[hello]
[hello]
[hello.txt]
[/no.dot/in_path]
[/has.dot/in]
[/no]
C++ è ok? –
Sai che "estensione file" e "ultimi tre caratteri di un nome file" sono ** non ** la stessa cosa? Alcuni file hanno più o meno 3 caratteri nella loro estensione: 'Foo.java',' foo.c' sono buoni esempi. Alcuni hanno anche più punti nelle estensioni: 'foo.tar.gz'. E ancora altri avranno un punto all'esterno dell'estensione: 'foo.bar.txt'. In breve: è un compito non banale. –
Quale piattaforma? Non è banale farlo correttamente in C, e non c'è nulla nelle librerie standard (perché i nomi dei file sono specifici della piattaforma). Ad esempio, una barra rovesciata è un separatore di percorso in Windows ma non in * nix, quindi per 'some \ path.to \ file' il risultato di" rimozione dell'estensione "in Windows è' some \ path.to \ file' (perché non c'è estensione), ma su * nix è 'some \ path' (perché l'estensione è' a \ file'). –