WedX - журнал о программировании и компьютерных науках

Как сохранить итератор над стандартным вводом в структуре?

Я создал структуру, в которой должен храниться итератор над файлом или стандартным вводом, но компилятор орет на меня :)

Я решил, что структура Lines должна храниться в мою структуру, чтобы использовать ее позже, и Box позволит хранить переменную с неизвестным размером, поэтому я определяю свою структуру следующим образом:

pub struct A {
    pub input: Box<Lines<BufRead>>,
}

Я хочу сделать что-то вроде этого позже:

let mut a = A {
    input: /* don't know what should be here yet */,
};
if something {
    a.input = Box::new(io::stdin().lock().lines());
} else {
    a.input = Box::new(BufReader::new(file).lines());
}

И наконец

for line in a.input {
    // ...
}

Но я получил ошибку от компилятора

error[E0277]: the size for values of type `(dyn std::io::BufRead + 'static)` cannot be known at compilation time
  --> src/context.rs:11:5
   |
11 |     pub input: Box<Lines<BufRead>>,
   |     ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ doesn't have a size known at compile-time
   |
   = help: the trait `std::marker::Sized` is not implemented for `(dyn std::io::BufRead + 'static)`
   = note: to learn more, visit <https://doc.rust-lang.org/book/second-edition/ch19-04-advanced-types.html#dynamically-sized-types-and-sized>
   = note: required by `std::io::Lines`

Как я могу достичь своей цели?

23.03.2019

Ответы:


1

Самый общий ответ на ваш вопрос: вы не можете/не можете. Блокирующий стандартный ввод возвращает тип, который ссылается на значение Stdin. Вы не можете создать локальное значение (stdin()), взять ссылку на него (.lock()), а затем вернуть эту ссылку.

Если вы просто хотите сделать это внутри функции, не возвращая ее, вы можете создать трейт-объект:

use std::io::{self, prelude::*, BufReader};

fn example(file: Option<std::fs::File>) {
    let stdin;
    let mut stdin_lines;
    let mut file_lines;

    let input: &mut Iterator<Item = _> = match file {
        None => {
            stdin = io::stdin();
            stdin_lines = stdin.lock().lines();
            &mut stdin_lines
        }
        Some(file) => {
            file_lines = BufReader::new(file).lines();
            &mut file_lines
        }
    };

    for line in input {
        // ...
    }
}

Или создайте новую общую функцию, в которую вы можете передать конкретный итератор любого типа:

use std::io::{self, prelude::*, BufReader};

fn example(file: Option<std::fs::File>) {
    match file {
        None => finally(io::stdin().lock().lines()),
        Some(file) => finally(BufReader::new(file).lines()),
    }
}

fn finally(input: impl Iterator<Item = io::Result<String>>) {
    for line in input {
        // ...
    }
}

Вы можете поместить либо трейт-объект, либо универсальный тип в структуру, даже если вы не можете его вернуть:

struct A<'a> {
    input: &mut Iterator<Item = io::Result<String>>,
}
struct A<I> 
where
    I: Iterator<Item = io::Result<String>>,
{
    input: I,
}

Если вы чувствуете себя предприимчивым, вы можете использовать небезопасный код / ​​ящики, обертывающие небезопасный код, для хранения значения Stdin и итератора, ссылающегося на него вместе, что не всегда безопасно.

Смотрите также:

input: Box<Lines<BufRead>>,

Это неверно, потому что Lines не является чертой. Вы хотите либо:

use std::io::{prelude::*, Lines};

pub struct A {
    pub input: Lines<Box<BufRead>>,
}

Or

use std::io;

pub struct A {
    pub input: Box<Iterator<Item = io::Result<String>>>,
}
23.03.2019
Новые материалы

Я хотел выучить язык программирования MVC4, но не мог выучить его раньше, потому что это выглядит сложно…
Просто начните и учитесь самостоятельно Я хотел выучить язык программирования MVC4, но не мог выучить его раньше, потому что он кажется мне сложным, и я бросил его. Это в основном инструмент..

Лицензии с открытым исходным кодом: руководство для разработчиков и создателей
В динамичном мире разработки программного обеспечения открытый исходный код стал мощной парадигмой, способствующей сотрудничеству, инновациям и прогрессу, движимому сообществом. В основе..

Объяснение документов 02: BERT
BERT представил двухступенчатую структуру обучения: предварительное обучение и тонкая настройка. Во время предварительного обучения модель обучается на неразмеченных данных с помощью..

Как проанализировать работу вашего классификатора?
Не всегда просто знать, какие показатели использовать С развитием глубокого обучения все больше и больше людей учатся обучать свой первый классификатор. Но как только вы закончите..

Работа с цепями Маркова, часть 4 (Машинное обучение)
Нелинейные цепи Маркова с агрегатором и их приложения (arXiv) Автор : Бар Лайт Аннотация: Изучаются свойства подкласса случайных процессов, называемых дискретными нелинейными цепями Маркова..

Crazy Laravel Livewire упростил мне создание электронной коммерции (панель администратора и API) [Часть 3]
Как вы сегодня, ребята? В этой части мы создадим CRUD для данных о продукте. Думаю, в этой части я не буду слишком много делиться теорией, но чаще буду делиться своим кодом. Потому что..

Использование машинного обучения и Python для классификации 1000 сезонов новичков MLB Hitter
Чему может научиться машина, глядя на сезоны новичков 1000 игроков MLB? Это то, что исследует это приложение. В этом процессе мы будем использовать неконтролируемое обучение, чтобы..


Для любых предложений по сайту: [email protected]