这是我的JSON数据,
{
"total": 60,
"per_page": 10,
"current_page": 1,
"last_page": 6,
"next_page_url": "http://www.ladybirdweb.com/support/api/v1/helpdesk/inbox?page=2",
"prev_page_url": null,
"from": 1,
"to": 10,
"data": [
{
"updated_at": "2017-07-18 07:17:25",
"user_name": "eugene@smartitfirm.com",
"first_name": "Eugene",
"last_name": "Dunayev",
"email": "eugene@smartitfirm.com",
"profile_pic": "https://secure.gravatar.com/avatar/841369e30f04310b23750abbb670c65c?s=80&r=g&d=identicon",
"ticket_number": "AAAA-0000-745",
"id": 4395,
"title": "Demo",
"created_at": "2017-07-12 04:15:15",
"department_name": "Support",
"priotity_name": "Low",
"priority_color": "#00a65a",
"sla_plan_name": "Low",
"help_topic_name": "Support query",
"ticket_status_name": "Open",
"department_id": "3",
"user_dpt": null,
"attachment": "23",
"overdue_date": "2017-07-12 12:15:15"
},
{
"updated_at": "2017-07-18 07:15:40",
"user_name": "rmuller@idagroup.net",
"first_name": "Robin",
"last_name": "W.",
"email": "rmuller@idagroup.net",
"profile_pic": "https://secure.gravatar.com/avatar/90efb0e570dfc699f78c414449cb46d9?s=80&r=g&d=identicon",
"ticket_number": "AAAA-0000-776",
"id": 4426,
"title": "=?UTF-8?Q?Re:_Robin_-_Implementing_Faveo_H?= =?UTF-8?Q?elp_Desk._Let=E2=80=99s_get_you_started.?=",
"created_at": "2017-07-14 16:15:17",
"department_name": "Support",
"priotity_name": "Low",
"priority_color": "#00a65a",
"sla_plan_name": "Low",
"help_topic_name": "Support query",
"ticket_status_name": "Open",
"department_id": "3",
"user_dpt": null,
"attachment": "3",
"overdue_date": "2017-07-17 12:00:00"
}
]
}
在这个特定的词典中,我只是得到了这个。其他显示正确数据的其他词典。 所以,我希望在这里得到回应,
"title: Re: Robin - Implementing Faveo Help Desk. Let’s get you started."
即使它没问题,我也会得到这种格式,然后有任何解决方案可以获得实际数据,因此我将在视图控制器中打印。
代码是,
这里我发送请求,
-(void)reload
{
NSString *url=[NSString stringWithFormat:@"%@helpdesk/inbox?api_key=%@&ip=%@&token=%@",[userDefaults objectForKey:@"companyURL"],API_KEY,IP,[userDefaults objectForKey:@"token"]];
MyWebservices *webservices=[MyWebservices sharedInstance];
[webservices httpResponseGET:url parameter:@"" callbackHandler:^(NSError *error,id json,NSString* msg) {
if (error || [msg containsString:@"Error"]) {
[refresh endRefreshing];
[[AppDelegate sharedAppdelegate] hideProgressView];
if (msg) {
[utils showAlertWithMessage:[NSString stringWithFormat:@"Error-%@",msg] sendViewController:self];
}else if(error) {
[utils showAlertWithMessage:[NSString stringWithFormat:@"Error-%@",error.localizedDescription] sendViewController:self];
NSLog(@"Thread-NO4-getInbox-Refresh-error == %@",error.localizedDescription);
}
return ;
}
if ([msg isEqualToString:@"tokenRefreshed"]) {
[self reload];
NSLog(@"Thread--NO4-call-getInbox");
return;
}
if (json) {
//NSError *error;
NSLog(@"Thread-NO4--getInboxAPI--%@",json);
_mutableArray = [json objectForKey:@"data"];
_nextPageUrl =[json objectForKey:@"next_page_url"];
_currentPage=[[json objectForKey:@"current_page"] integerValue];
_totalTickets=[[json objectForKey:@"total"] integerValue];
_totalPages=[[json objectForKey:@"last_page"] integerValue];
NSLog(@"Thread-NO4.1getInbox-dic--%@", _mutableArray);
dispatch_async(dispatch_get_global_queue( DISPATCH_QUEUE_PRIORITY_LOW, 0), ^{
dispatch_async(dispatch_get_main_queue(), ^{
[[AppDelegate sharedAppdelegate] hideProgressView];
[refresh endRefreshing];
[self.tableView reloadData];
});
});
}
NSLog(@"Thread-NO5-getInbox-closed");
}];
}
}
这里是cellForRowAtIndexPath方法,
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath{
if (indexPath.row == [_mutableArray count]) {
LoadingTableViewCell *cell=[tableView dequeueReusableCellWithIdentifier:@"LoadingCellID"];
if (cell == nil)
{
NSArray *nib = [[NSBundle mainBundle] loadNibNamed:@"LoadingTableViewCell" owner:self options:nil];
cell = [nib objectAtIndex:0];
}
UIActivityIndicatorView *activityIndicator = (UIActivityIndicatorView *)[cell.contentView viewWithTag:1];
[activityIndicator startAnimating];
return cell;
}else{
TicketTableViewCell *cell=[tableView dequeueReusableCellWithIdentifier:@"TableViewCellID"];
if (cell == nil)
{
NSArray *nib = [[NSBundle mainBundle] loadNibNamed:@"TicketTableViewCell" owner:self options:nil];
cell = [nib objectAtIndex:0];
}
NSDictionary *finaldic=[_mutableArray objectAtIndex:indexPath.row];
cell.ticketIdLabel.text=[finaldic objectForKey:@"ticket_number"];
NSString *fname= [finaldic objectForKey:@"first_name"];
NSString *lname= [finaldic objectForKey:@"last_name"];
NSString *userName= [finaldic objectForKey:@"user_name"];
[Utils isEmpty:fname];
[Utils isEmpty:lname];
if (![Utils isEmpty:fname] && ![Utils isEmpty:lname])
{
cell.mailIdLabel.text=[NSString stringWithFormat:@"%@ %@",[finaldic objectForKey:@"first_name"],[finaldic objectForKey:@"last_name"]];
}
else
{ if(![Utils isEmpty:userName])
{
cell.mailIdLabel.text=[finaldic objectForKey:@"user_name"];
}
else
{
cell.mailIdLabel.text=[finaldic objectForKey:@"email"];
}
}
NSString *title1= [finaldic objectForKey:@"title"];
[Utils isEmpty:title1];
if ([Utils isEmpty:title1]){
cell.ticketSubLabel.text=@"No Title";
}
else
{
cell.ticketSubLabel.text=[finaldic objectForKey:@"title"];
}
return cell;
}
}
我的Web服务类方法是,
-(void)httpResponseGET:(NSString *)urlString
parameter:(id)parameter
callbackHandler:(callbackHandler)block{
NSError *error;
NSMutableURLRequest *request = [[NSMutableURLRequest alloc] initWithURL:[NSURL URLWithString:urlString]];
//[request addValue:@"text/html" forHTTPHeaderField:@"Accept"];
[request addValue:@"application/json" forHTTPHeaderField:@"Accept"];
[request addValue:@"application/json" forHTTPHeaderField:@"Content-Type"];
[request setTimeoutInterval:45.0];
NSData *postData = nil;
if ([parameter isKindOfClass:[NSString class]]) {
postData = [((NSString *)parameter) dataUsingEncoding:NSUTF8StringEncoding];
} else {
postData = [NSJSONSerialization dataWithJSONObject:parameter options:0 error:&error];
}
[request setHTTPBody:postData];
[request setHTTPMethod:@"GET"];
NSLog(@"Thread--httpResponseGET--Request : %@", urlString);
NSURLSession *session = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration] ];
[[session dataTaskWithRequest:request completionHandler:^(NSData * data, NSURLResponse * response, NSError * error) {
NSLog(@"Response is required : %@",(NSHTTPURLResponse *) response);
if (error) {
dispatch_async(dispatch_get_main_queue(), ^{
block(error,nil,nil);
});
NSLog(@"Thread--httpResponseGET--dataTaskWithRequest error: %@", [error localizedDescription]);
}else if ([response isKindOfClass:[NSHTTPURLResponse class]]) {
NSInteger statusCode = [(NSHTTPURLResponse *)response statusCode];
if (statusCode != 200) {
NSLog(@"dataTaskWithRequest HTTP status code: %ld", (long)statusCode);
if (statusCode==400) {
if ([[self refreshToken] isEqualToString:@"tokenRefreshed"]) {
dispatch_async(dispatch_get_main_queue(), ^{
block(nil,nil,@"tokenRefreshed");
});
NSLog(@"Thread--httpResponsePOST--tokenRefreshed");
}else {
dispatch_async(dispatch_get_main_queue(), ^{
block(nil,nil,@"tokenNotRefreshed");
});
NSLog(@"Thread--httpResponsePOST--tokenNotRefreshed");
}
}else
dispatch_async(dispatch_get_main_queue(), ^{
block(nil, nil,[NSString stringWithFormat:@"Error-%ld",(long)statusCode]);
});
return ;
}
NSString *replyStr = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
if ([replyStr containsString:@"token_expired"]) {
NSLog(@"Thread--httpResponseGET--token_expired");
if ([[self refreshToken] isEqualToString:@"tokenRefreshed"]) {
dispatch_async(dispatch_get_main_queue(), ^{
block(nil,nil,@"tokenRefreshed");
});
NSLog(@"Thread--httpResponseGET--tokenRefreshed");
}else {
dispatch_async(dispatch_get_main_queue(), ^{
block(nil,nil,@"tokenNotRefreshed");
});
NSLog(@"Thread--httpResponseGET--tokenNotRefreshed");
}
return;
}
NSError *jsonerror = nil;
id responseData = [NSJSONSerialization JSONObjectWithData:data options:NSJSONReadingAllowFragments error:&jsonerror];
dispatch_async(dispatch_get_main_queue(), ^{
block(jsonerror,responseData,nil);
});
}
}] resume];
}
答案 0 :(得分:2)
好的,首先字符串是Q-encode,并且是来自电子邮件的MIME标题。
“Q”编码类似于RFC 1521中定义的“Quoted-Printable”内容传输编码。它旨在允许在ASCII终端上无需解码就可以在ASCII终端上解密包含大多数ASCII字符的文本。
任何8位值都可以用“=”后跟两个十六进制数字表示。例如,如果使用的字符集是ISO-8859-1,那么“=”字符将被编码为“= 3D”,并且空格被编码为“= 20”。 (大写应该用于十六进制数字“A”到“F”。)
8位十六进制值20(例如,ISO-8859-1 SPACE)可能是 表示为“”(下划线,ASCII 95)。 (这个角色可能没有 通过一些互联网邮件网关,但它的使用将大大增加 使用没有的邮件阅读器增强“Q”编码数据的可读性 支持此编码。)请注意,“”始终表示 十六进制20,即使SPACE字符占用不同的代码 在使用的字符集中的位置。
- 醇>
8位值,对应于除以外的可打印ASCII字符 “=”,“?”,“_”(下划线),以及SPACE可以表示为那些 字符。 (但请参阅第5节的限制。)
来源http://www.freesoft.org/CIE/RFC/1522/6.htm
此问题已在其他stackoverflow的帖子中解决:
在上面的帖子中显示了一个代码段代码,用于替换带有base 64支持的Q编码字符串,并将其转换为有效的NSString格式:
@implementation NSString (MimeEncodedWord)
- (BOOL) isMimeEncodedWord
{
return [self hasPrefix:@"=?"] && [self hasSuffix:@"?="];
}
+ (NSString*) stringWithMimeEncodedWord:(NSString*)word
{ // Example: =?iso-8859-1?Q?=A1Hola,_se=F1or!?=
NSArray *components = [word componentsSeparatedByString:@"?"];
if (components.count < 5) return nil;
NSString *charset = [components objectAtIndex:1];
NSStringEncoding encoding = CFStringConvertEncodingToNSStringEncoding(CFStringConvertIANACharSetNameToEncoding((CFStringRef)charset)); // TODO: What happens if the encoding is invalid?
NSString *encodingType = [components objectAtIndex:2];
NSString *encodedText = [components objectAtIndex:3];
if ([encodingType isEqualToString:@"Q"])
{ // quoted-printable
encodedText = [encodedText stringByReplacingOccurrencesOfString:@"_" withString:@" "];
encodedText = [encodedText stringByReplacingOccurrencesOfString:@"=" withString:@"%"];
NSString *decoded = [encodedText stringByReplacingPercentEscapesUsingEncoding:encoding];
return decoded;
} else if ([encodingType isEqualToString:@"B"])
{ // base64
NSData *data = [QSStrings decodeBase64WithString:encodedText];
NSString *decoded = [[NSString alloc] initWithData:data encoding:encoding];
return decoded;
} else {
NSLog(@"%@ is not a valid encoding (must be Q or B)", encodingType);
return nil;
}
}
@end
此代码也可以在Github帖子中找到,作为NSString类类别 在这里:https://github.com/hpique/NSString-MimeEncodedWord
希望有所帮助
答案 1 :(得分:1)
检查以下代码,我希望它能工作。我根据你的代码创建..!此代码取自@Sophy Swicz。
NSString *encodedString =[finaldic objectForKey:@"title"];
[Utils isEmpty:encodedString];
if ([Utils isEmpty:encodedString]){
cell.ticketSubLabel.text=@"No Title";
}
else
{
NSMutableString *decodedString = [[NSMutableString alloc] init];
if ([encodedString hasPrefix:@"=?UTF-8?Q?"] || [encodedString hasSuffix:@"?="])
{
NSScanner *scanner = [NSScanner scannerWithString:encodedString];
NSString *buf = nil;
// NSMutableString *decodedString = [[NSMutableString alloc] init];
while ([scanner scanString:@"=?UTF-8?Q?" intoString:NULL]
|| ([scanner scanUpToString:@"=?UTF-8?Q?" intoString:&buf] && [scanner scanString:@"=?UTF-8?Q?" intoString:NULL])) {
if (buf != nil) {
[decodedString appendString:buf];
}
buf = nil;
NSString *encodedRange;
if (![scanner scanUpToString:@"?=" intoString:&encodedRange]) {
break; // Invalid encoding
}
[scanner scanString:@"?=" intoString:NULL]; // Skip the terminating "?="
// Decode the encoded portion (naively using UTF-8 and assuming it really is Q encoded)
// I'm doing this really naively, but it should work
// Firstly I'm encoding % signs so I can cheat and turn this into a URL-encoded string, which NSString can decode
encodedRange = [encodedRange stringByReplacingOccurrencesOfString:@"%" withString:@"=25"];
// Turn this into a URL-encoded string
encodedRange = [encodedRange stringByReplacingOccurrencesOfString:@"=" withString:@"%"];
// Remove the underscores
encodedRange = [encodedRange stringByReplacingOccurrencesOfString:@"_" withString:@" "];
// [decodedString appendString:[encodedRange stringByReplacingPercentEscapesUsingEncoding:NSUTF8StringEncoding]];
NSString *str1= [encodedRange stringByRemovingPercentEncoding];
[decodedString appendString:str1];
}
NSLog(@"Decoded string = %@", decodedString);
cell.ticketSubLabel.text= decodedString;
}
else{
cell.ticketSubLabel.text= encodedString;
}
}