我在练习使用指针时偶然发现了一些我不理解的东西。该程序执行以下操作:
当我交换这两个时出现问题
//Get info about new movie from user
string movieTitle;
int movieYear; //Not used at the moment
cout << "Enter title for movie " << (i+1) << endl;
getline(cin,movieTitle);
和
//Create new movie
movies_t amovie;
movies_t * pmovie;
pmovie = &amovie;
当我将“用户输入部分”放在“创建新电影”部分上方时,就像现在一样,我得到了:
但是当我交换它们时:
我不明白为什么,因为它们根本不会相互影响。
数据显示如下:
for(i=0;i<movieVector.size();i++)
{
cout << movieVector[i].title << endl;
}
这些是相关函数(main 和 newThreads)
void newThreads(vector<movies_t> *movieVectorPointer)
{
boost::thread_group group; //Start thread group
int i;
for(i=0; i<2; i++) //Make X amount of threads (2 in this case)
{
//Get info about new movie from user
string movieTitle;
int movieYear; //Not used at the moment
cout << "Enter title for movie " << (i+1) << endl;
getline(cin,movieTitle);
//Create new movie
movies_t amovie;
movies_t * pmovie;
pmovie = &amovie;
//Let user know we are now starting on this thread
cout << "Doing thread " << i << endl;
//Start new thread
newThreadStruct startNewThread(movieTitle,movieYear,pmovie,movieVectorPointer);
group.create_thread(startNewThread);
}
group.join_all(); //Wait for all threads in group to finish
}
int main()
{
cout << "Hello world!" << endl; //I am born.
vector<movies_t> movieVector; //Create vector to store movies in
newThreads(&movieVector); //Start making new threads. Pass movieVector's address so it can be used within threads.
/* The below code will not be executed until all threads are done */
cout << "Amount of movies " << movieVector.size() << endl; //Let user know how many movies we made
//Show all movies we made
int i;
for(i=0;i<movieVector.size();i++)
{
cout << movieVector[i].title << endl;
}
cout << "Bye world!" << endl; //This life has passed.
return 0;
}
这里是完整的代码,以防万一:
#include <iostream>
#include <boost/thread.hpp>
using namespace std;
//A movie will hold a title and a year. Only title is used in this code.
struct movies_t {
string title;
int year;
};
//This function is where the data is added to our new movie,a fter which our finished movie will be added to the vector. It is called from within the "newThreadStruct" operator.
int doMovieWork(string movieTitle,int movieYear,movies_t *moviePointer,vector<movies_t> *movieVector)
{
moviePointer->title = movieTitle; //Add title to new movie
movieVector->push_back(*moviePointer); //Add movie to vector
return 0;
};
//This struct will be used to create new Boost threads. It accepts various arguments.
struct newThreadStruct
{
newThreadStruct(string movieTitle,int movieYear,movies_t *moviePointer,vector<movies_t> *movieVector) : movieTitle(movieTitle),movieYear(movieYear),moviePointer(moviePointer),movieVector(movieVector) { }
void operator()()
{
doMovieWork(movieTitle,movieYear,moviePointer,movieVector);
}
string movieTitle;
int movieYear;
movies_t *moviePointer;
vector<movies_t> *movieVector;
};
void newThreads(vector<movies_t> *movieVectorPointer)
{
boost::thread_group group; //Start thread group
int i;
for(i=0; i<2; i++) //Make X amount of threads (2 in this case)
{
//Get info about new movie from user
string movieTitle;
int movieYear; //Not used at the moment
cout << "Enter title for movie " << (i+1) << endl;
getline(cin,movieTitle);
//Create new movie
movies_t amovie;
movies_t * pmovie;
pmovie = &amovie;
//Let user know we are now starting on this thread
cout << "Doing thread " << i << endl;
//Start new thread
newThreadStruct startNewThread(movieTitle,movieYear,pmovie,movieVectorPointer);
group.create_thread(startNewThread);
}
group.join_all(); //Wait for all threads in group to finish
}
int main()
{
cout << "Hello world!" << endl; //I am born.
vector<movies_t> movieVector; //Create vector to store movies in
newThreads(&movieVector); //Start making new threads. Pass movieVector's address so it can be used within threads.
/* The below code will not be executed until all threads are done */
cout << "Amount of movies " << movieVector.size() << endl; //Let user know how many movies we made
//Show all movies we made
int i;
for(i=0;i<movieVector.size();i++)
{
cout << movieVector[i].title << endl;
}
cout << "Bye world!" << endl; //This life has passed.
return 0;
}
最佳答案
获取用户输入和初始化 movies_t 对象的顺序是一个转移注意力的顺序。实际问题在这里:
//Create new movie
movies_t amovie;
movies_t * pmovie;
pmovie = &amovie;
/* ... */
//Start new thread
newThreadStruct startNewThread(movieTitle,movieYear,pmovie,movieVectorPointer);
group.create_thread(startNewThread);
您正在将局部变量 (amovie) 的地址传递给线程。您无法直接控制此线程何时启动、何时尝试访问您传递给它的指针或何时退出。但是您不会在主线程循环中等待工作线程使用本地。一旦循环循环,您传递的变量就会超出范围。当工作线程尝试使用它时,您会调用未定义的行为。这非常糟糕。
可能解决此问题的最简单(也是最幼稚)的方法是动态创建 amovie 对象:
//Create new movie
movies_t * pmovie = new movies_t;
...然后当您使用完它时,在某处删除它。从我对您的代码的初步检查来看,在哪里 delete 不是很明显 - 但它可能在 main 的末尾。
这种幼稚的方法在指针所有权、内存管理以及潜在的死锁和竞争条件方面打开了一大堆蠕虫。您现在已经进入了多线程编程领域——C++ 编程中最难正确完成的事情之一。上面的幼稚方法将“起作用”(就像防止代码崩溃一样),尽管并非没有缺陷——但是如果您正走在多线程编程的道路上,那么是时候开始研究如何正确地完成它了。这超出了我在这里的小帖子的范围。
关于c++ - 在用户插入数据之前创建结构对象时,Vector 返回乱码,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/8360506/
总的来说,我对ruby还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用
出于纯粹的兴趣,我很好奇如何按顺序创建PI,而不是在过程结果之后生成数字,而是让数字在过程本身生成时显示。如果是这种情况,那么数字可以自行产生,我可以对以前看到的数字实现垃圾收集,从而创建一个无限系列。结果只是在Pi系列之后每秒生成一个数字。这是我通过互联网筛选的结果:这是流行的计算机友好算法,类机器算法:defarccot(x,unity)xpow=unity/xn=1sign=1sum=0loopdoterm=xpow/nbreakifterm==0sum+=sign*(xpow/n)xpow/=x*xn+=2sign=-signendsumenddefcalc_pi(digits
关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。
在控制台中反复尝试之后,我想到了这种方法,可以按发生日期对类似activerecord的(Mongoid)对象进行分组。我不确定这是完成此任务的最佳方法,但它确实有效。有没有人有更好的建议,或者这是一个很好的方法?#eventsisanarrayofactiverecord-likeobjectsthatincludeatimeattributeevents.map{|event|#converteventsarrayintoanarrayofhasheswiththedayofthemonthandtheevent{:number=>event.time.day,:event=>ev
我想将html转换为纯文本。不过,我不想只删除标签,我想智能地保留尽可能多的格式。为插入换行符标签,检测段落并格式化它们等。输入非常简单,通常是格式良好的html(不是整个文档,只是一堆内容,通常没有anchor或图像)。我可以将几个正则表达式放在一起,让我达到80%,但我认为可能有一些现有的解决方案更智能。 最佳答案 首先,不要尝试为此使用正则表达式。很有可能你会想出一个脆弱/脆弱的解决方案,它会随着HTML的变化而崩溃,或者很难管理和维护。您可以使用Nokogiri快速解析HTML并提取文本:require'nokogiri'h
我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i
使用带有Rails插件的vim,您可以创建一个迁移文件,然后一次性打开该文件吗?textmate也可以这样吗? 最佳答案 你可以使用rails.vim然后做类似的事情::Rgeneratemigratonadd_foo_to_bar插件将打开迁移生成的文件,这正是您想要的。我不能代表textmate。 关于ruby-使用VimRails,您可以创建一个新的迁移文件并一次性打开它吗?,我们在StackOverflow上找到一个类似的问题: https://sta
我对最新版本的Rails有疑问。我创建了一个新应用程序(railsnewMyProject),但我没有脚本/生成,只有脚本/rails,当我输入ruby./script/railsgeneratepluginmy_plugin"Couldnotfindgeneratorplugin.".你知道如何生成插件模板吗?没有这个命令可以创建插件吗?PS:我正在使用Rails3.2.1和ruby1.8.7[universal-darwin11.0] 最佳答案 随着Rails3.2.0的发布,插件生成器已经被移除。查看变更日志here.现在
我的瘦服务器配置了nginx,我的ROR应用程序正在它们上运行。在我发布代码更新时运行thinrestart会给我的应用程序带来一些停机时间。我试图弄清楚如何优雅地重启正在运行的Thin实例,但找不到好的解决方案。有没有人能做到这一点? 最佳答案 #Restartjustthethinserverdescribedbythatconfigsudothin-C/etc/thin/mysite.ymlrestartNginx将继续运行并代理请求。如果您将Nginx设置为使用多个上游服务器,例如server{listen80;server
我有一个表单,其中有很多字段取自数组(而不是模型或对象)。我如何验证这些字段的存在?solve_problem_pathdo|f|%>... 最佳答案 创建一个简单的类来包装请求参数并使用ActiveModel::Validations。#definedsomewhere,atthesimplest:require'ostruct'classSolvetrue#youcouldevencheckthesolutionwithavalidatorvalidatedoerrors.add(:base,"WRONG!!!")unlesss