I'm reading the documentation and trying to write some basic file I/O code as a vehicle to help me learn Rust.
The following doesn't compile:
use std::fs;
use std::io;
use std::path::Path;
pub fn read_filenames_from_dir<P>(path: P) -> Result<Vec<Path>, io::Error>
where
P: AsRef<Path>,
{
let paths = try!(fs::read_dir(path));
Ok(paths.unwrap())
}
With the compilation error:
error[E0277]: the trait bound `[u8]: std::marker::Sized` is not satisfied in `std::path::Path`
--> src/main.rs:5:1
|
5 | / pub fn read_filenames_from_dir<P>(path: P) -> Result<Vec<Path>, io::Error>
6 | | where
7 | | P: AsRef<Path>,
8 | | {
9 | | let paths = try!(fs::read_dir(path));
10 | | Ok(paths.unwrap())
11 | | }
| |_^ `[u8]` does not have a constant size known at compile-time
|
= help: within `std::path::Path`, the trait `std::marker::Sized` is not implemented for `[u8]`
= note: required because it appears within the type `std::path::Path`
= note: required by `std::vec::Vec`
How should I write this function to return the collection of Path
s inside the Path
that's passed in?
You don't. Path
is a type that has no size, and is only usable through indirection (such as &Path
or Box<Path>
). In this sense, it is like the type str
or [u8]
— neither can be directly used, only indirectly.
What you probably want is a PathBuf
, which represents an owned path. It is the equivalent of String
for &str
and Vec<u8>
for &[u8]
.
After changing the return type, you have to properly map the results of the iterator to create your desired type:
use std::{
fs, io,
path::{Path, PathBuf},
};
pub fn read_filenames_from_dir<P>(path: P) -> Result<Vec<PathBuf>, io::Error>
where
P: AsRef<Path>,
{
fs::read_dir(path)?
.into_iter()
.map(|x| x.map(|entry| entry.path()))
.collect()
}
fn main() {
println!("{:?}", read_filenames_from_dir("/etc"));
}
See also:
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With