Diskuze: Problém s realtime čtením logu
V předchozím kvízu, Test znalostí C# .NET online, jsme si ověřili nabyté zkušenosti z kurzu.
ostrozan:12.7.2016 18:36
Jen tak zběžně na první pohled :
Příkazem "return" ukončíš čtení souboru hned po prvním řádku - to se
pak nediv že tam pak máš jen ten jeden
Máš nějaký důvod číst soubor po řádcích a ne celý (ReadToEnd())?
ostrozan:13.7.2016 5:28
Jak mám rozumět tomu
ale po úpravě souboru už se nezmění v GUI
?
to jako v jiné aplikaci upravuješ soubor a tady v tom sleduješ jak se
mění?
Ano přesně tak. Vysvětlím to lépe. Mám servery, kde při každém zapnutí se vygeneruje nový Log soubor s vlastním ID. Proto mám jako první funkci aby mi to našlo ve složce nejnovější soubor a pak ho předal funkci StreamReader. StreamReader má svůj vlastní loop, který se ale nějak neopakuje. Když se aplikace předělá jenom do Console, tak funguje a čte to ze souboru po úpravě. To že jsem tam mám ReadLine je proto, aby to vracelo jenom ten poslední upravený řádek a soubor se nečetl znovu a znovu. Pokud jsem to dobře pochopil. , Může tam být i ReadToEnd. Po přečtení ze souboru by měl StreamReader vracet hodnotu do GUI "WPF" což se taky stane, ale jenom jednou po startu, pak už to nevrací nic ani po úpravě souboru. Jinak jsem tam přidal Project z VS tak tam to uvidíš co asi myslím.
V skratke nejako takto:
- ReadFile() je napísaná tak, že ti neprečíta celý súbor ako napovedá jej názov, ale vždy len druhý riadok daného súboru (ak existuje). Presvedč sa debugovaním. Prvý riadok vráti Readline() v prvom while a je uložený do premennej line.
- Všetky tie while loopy sú úplne zbytočné a zle navrhnuté, keďže ti tá slučka zbehne iba raz alebo sa to zasekne v nekonečnej slučke.
- Ak máš v logu len jeden riadok, dostaneš sa do nekonečnej slučky, kde budeš uspávať thread na sekundu do nekonečna.
- Z tej WPF aplikácie tú metódu voláš len raz a to z konštruktora WPF okna.
- Načo máš premennú line? Používaš ju len vo while a tam môžeš rovno kontrolovať či sr.ReadLine() != null.
ostrozan:13.7.2016 15:40
Není mi jasné, jak toto může fungovat v konzoli - můžeš sem dát i ten funkční konzolový projekt? To by mně fakt zajímalo - bude to buď geniální řešení, nebo totální prasárna - tipuju žes to někde obšlehnul.
Jinak trochu (no trochu víc) jsem popravil ten tvůj projekt, aby dělal to co chceš - to jsem ovšem ještě nevěděl, že chceš použít jen poslední řádek ze souboru - mimochodem podle toho co píšeš
kde při každém zapnutí se vygeneruje nový Log soubor s vlastním ID.
bych si myslel, že ten řádek tam bude vždy jen jeden / ale i tak jsem tam dal úpravu, že ti to vypreparuje ze souboru poslední řádek .
Nevím jak se ta tvoje konstrukce měla dozvědět , že se ve složce něco
děje - já jsem to vyřešil tak, že se každých 100 ms dotazuju jestli v
posledně přidaném souboru ve složce nedošlo k novému zápisu.
Ten čas si samozřejmě nastav jaký chceš - sekundu - minutu - jak je libo,
ale pod těch 100 ms bych radši nešel
Reader.cs
class Reader
{
private string path,logFile;
public string filePath { get; set; }
public Reader(string path)
{
this.path = path;
var directory = new DirectoryInfo(path);
logFile = directory.GetFiles().OrderByDescending(f => f.LastWriteTime).First().ToString();
filePath = Path.Combine(path, logFile);
}
public string ReadFile()
{
try
{
using (StreamReader sr = new StreamReader(filePath))
{
string data =sr.ReadToEnd();
string[] rows = data.Split('\n');
return rows.Last();
}
}
catch (Exception)
{
string ex = "Could not read the file";
return ex;
}
}
}
MainWindow.xaml.cs
public partial class MainWindow : Window
{
// Your Path to main Log directory
string Path = @"C:\Users\Admin\Desktop\Test";
private DateTime lastLogTime;
public MainWindow()
{
InitializeComponent();
DispatcherTimer timer = new DispatcherTimer();
timer.Interval = TimeSpan.FromMilliseconds(100);
timer.Tick += timer_Tick;
timer.Start();
}
void timer_Tick(object sender, EventArgs e)
{
Reader reader = new Reader(Path);
DateTime logTime = File.GetLastWriteTime(reader.filePath);
if (logTime == lastLogTime) return;
else
{
lastLogTime = logTime;
textBlock.Text += reader.ReadFile()+'\n';
}
}
}
Jak vidíš, poměrně dost jsem ti toho vyházel - budeš li mít nějaké dotazy co,proč a jak - tak se klidně zeptej.
tady máš tvůj upravený projekt
+20 Zkušeností
+2,50 Kč
Toho, že se mi snažíte pomoct si vážím. Jak již jsem psal nahoře, C# jsem se začal učit nedávno ze své zajímavosti a nejsem v tom moc velký expert. Navíc jsem frontend web developer a backend v Laravelu takže C# je pro mě něco jiného. Jinak samozřejmě jsem to nepsal celé sám, ale hledal i na ostatních webech a na stackoverflow. Takže ano je to z toho co jsem se dočetl a pochopil a neni to jediný co jsem zkoušel stejně tak to udělat přes FileWatcher. K těm logům. Ten log má například název "Log_21565.txt" a po restartování serveru se vytvoří nový pod jiným ID např. "Log_85624.txt". Jinak ty logy nemaj jeden řádek ale třeba 20 Mb po delší době. Jinak to konzolové řešení je ZDE
ostrozan:13.7.2016 20:23
Nemáš zač
já jsem zase zkusil tu konzoli a když pominu, že to vůbec není stejný kód
tak mně to fungovalo jen při změně stávajícího souboru - při vytvoření
nového se nestalo nic
A teď dovol pár rad začátečníkovi:
uč se od píky - jedna věc je C# jako jazyk a konzolové aplikace a něco
jiného jsou knihovny frameworku .Net
ještě něco jiného je WPF
bez urážky - a neber si to osobně, ale při tvojí metodě učení moc
nechápeš jak to všechno funguje - zkřížením kódu z konzole, rad ze
StackOverFlow a vlastní invence jsi spáchal strašný bastl - ale ještě
jednou upozorňuju - nechci tě tím nijak urážet - ne všechno je špatně -
zjištění posledně přidaného souboru je super
k chybám které vypsal Andrej Farkaš ještě jednu - i když není
fatální:
pro čtení souboru kombinuješ dva nástroje pro čtení
FileStream a StreamReader - buď jedno, nebo
druhé.
Ale oboje ti bude soubor číst od začátku do konce ať už celé naráz, nebo po řádcích takže i kdyby měl soubor gigabajt , tak bohužel.(Opravte mně někdo, jestli se mýlím)
Zmínkou o FileWatcheru - správně FileSystemWatcher jsi mi
připomněl že ani moje řešení není dokonalé -
takže jsem ho ještě vylepšil.
Reader.cs
class Reader
{
public Reader()
{
}
public string ReadFile(string filePath)
{
try
{
using (StreamReader sr = new StreamReader(filePath))
{
string data =sr.ReadToEnd();
string[] rows = data.Split('\n');
return rows.Last();
}
}
catch (Exception)
{
string ex = "Could not read the file";
return ex;
}
}
}
MainWindow.xaml.cs
public partial class MainWindow : Window
{
// Your Path to main Log directory
string folderPath = @"C:\Users\Admin\Desktop\Test";
private FileSystemWatcher fileSystemWatcher;
public MainWindow()
{
InitializeComponent();
StartFileSystemWatcher();
}
private void StartFileSystemWatcher()
{
fileSystemWatcher = new FileSystemWatcher();
// Set folder path to watch
fileSystemWatcher.Path = folderPath;
// Gets or sets the type of changes to watch for.
// In this case we will watch change of filename, last modified time, size and directory name
fileSystemWatcher.NotifyFilter = NotifyFilters.FileName |NotifyFilters.LastWrite |
NotifyFilters.Size |NotifyFilters.DirectoryName;
// Event handler
fileSystemWatcher.Changed += new FileSystemEventHandler(fileSystemWatcher_Changed);
// START watching
fileSystemWatcher.EnableRaisingEvents = true;
}
private void fileSystemWatcher_Changed(object sender, FileSystemEventArgs e)
{
fileSystemWatcher.EnableRaisingEvents = false;
Reader reader = new Reader();
Dispatcher.Invoke((Action)(() => textBlock.Text += reader.ReadFile(e.FullPath) + '\n'));
fileSystemWatcher.EnableRaisingEvents = true;
}
}
jak jistě poznáš, taky jsem si pomohl na netu, ale když si chceš ulehčit práci a využít cizí kód, musíš vědět co děláš ne jen doufat,že to bude fungovat
A nakonec ještě odkaz na slušný tutor WPF ten zdejší není úplně špatný, ale jaksi zamrzl v začátcích.
Zobrazeno 12 zpráv z 12.