- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
嗨,我正在为我的应用程序使用 wcf SOAP 服务,我发送如下请求。
postStr = [NSString stringWithFormat:@"<?xml version=\"1.0\"?>\n"
"<s:Envelope xmlns:s=\"http://schemas.xmlsoap.org/soap/envelope/\">\n"
"<s:Body>\n"
"<InsUpdDelActivityInfo xmlns=\"http://tempuri.org/\">\n"
"<objEventsContent xmlns:d4p1=\"http://schemas.datacontract.org/2004/07/iCampuslite.Model.ActivityStream\" xmlns:i=\"http://www.w3.org/2001/XMLSchema-instance\">\n"
"<d4p1:ActCommentId i:nil=\"true\" />\n"
"<d4p1:ActSubTypeCd i:nil=\"true\" />\n"
"<d4p1:ActType>Status</d4p1:ActType>\n"
"<d4p1:ActTypeCd>1</d4p1:ActTypeCd>\n"
"<d4p1:ActivityAnswersList />\n"
"<d4p1:ActivityComments />\n"
"<d4p1:ActivityId i:nil=\"true\" />\n"
"<d4p1:ActivityLike />\n"
"<d4p1:ActivityName>%@</d4p1:ActivityName>\n"
"<d4p1:ActivityStreamImagesBytes>%@</d4p1:ActivityStreamImagesBytes>\n"
"<d4p1:AnswerDesc i:nil=\"true\" />\n"
"<d4p1:AnswerId>0</d4p1:AnswerId>\n"
"<d4p1:CommentDesc i:nil=\"true\" />\n"
"<d4p1:CommentId i:nil=\"true\" />\n"
"<d4p1:CreatedUserId>%@</d4p1:CreatedUserId>\n"
"<d4p1:CreatedUserName i:nil=\"true\" />\n"
"<d4p1:FileOrLinkName i:nil=\"true\" />\n"
"<d4p1:IsLiked>0</d4p1:IsLiked>\n"
"<d4p1:IsTotalSchool i:nil=\"true\" />\n"
"<d4p1:LikeCount>0</d4p1:LikeCount>\n"
"<d4p1:LinkImage i:nil=\"true\" />\n"
"<d4p1:ObjTypeCdId i:nil=\"true\" />\n"
"<d4p1:ObjTypeId i:nil=\"true\" />\n"
"<d4p1:OperationMode>I</d4p1:OperationMode>\n"
"<d4p1:OperationType i:nil=\"true\" />\n"
"<d4p1:OperationTypeId i:nil=\"true\" />\n"
"<d4p1:OrganizationId>%@</d4p1:OrganizationId>\n"
"<d4p1:OtherActivityId i:nil=\"true\" />\n"
"%@\n"
"</objEventsContent>\n"
"<ismobile>true</ismobile>\n"
"</InsUpdDelActivityInfo>\n"
"</s:Body>\n"
"</s:Envelope>", statusText, [appDelegateObj.loginUserInfoDict valueForKey:@"a:UserId"], [appDelegateObj.loginUserInfoDict valueForKey:@"a:OrgId"], workspaceStr];
NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:[NSURL URLWithString:[NSString stringWithFormat:@"%@iCampusliteMobileService/ActivityStreamSl.svc", appDelegateObj.baseURL]]];
NSString *messageLength = [NSString stringWithFormat:@"%lu", (unsigned long)[postStr length]];
[request addValue:@"text/xml; charset=utf-8" forHTTPHeaderField:@"Content-Type"];
[request addValue:@"http://tempuri.org/IActivityStreamSl/InsUpdDelActivityInfo" forHTTPHeaderField:@"SOAPAction"];
[request addValue:messageLength forHTTPHeaderField:@"Content-Length"];
[request setHTTPMethod:@"POST"];
[request setHTTPBody:[postStr dataUsingEncoding:NSUTF8StringEncoding]];
NSError *error = nil;
NSData *returnData = [NSURLConnection sendSynchronousRequest:request returningResponse:nil error:&error];
在上面的请求中有一个元素ActivityStreamImagesBytes
,它是我必须传递图像的base64binary
参数。
我尝试过使用多种不同的格式。
这里是 wcf 服务截图
这是服务端代码
public string byteArrayToImage(byte[] byteArrayIn,string fileName)
{
if (byteArrayIn != null)
{
ActivityStreamSl.LogMsg("Byte Array Count : "+byteArrayIn.Length.ToString(), "D:\\log.txt");
var serverfile = "D:somepath\somepath\somefolder";
var getfile = HelperClass.Filesavehelper(Constants.UploadPaths.ActivityStream, "testfilename.png", serverfile);
FileStream file = new FileStream(getfile, FileMode.Create);
file.Write(byteArrayIn, 0, byteArrayIn.Length);
file.Close();
file.Dispose();
return getfile;
}
else
{
ActivityStreamSl.LogMsg("Byte array is null","D:\\log.txt");
}
return "";
}
服务器需要一个字节数组,但我不知道如何发送它?而且我不知道 base64binary 数据类型是什么?我是否必须发送 base64 编码的字符串或字节数组或只是来自 NSData *data = UIImageJPEGRepresentation([UIImage imageNamed:@"popular.png"], 0.7);
任何帮助都是可观的
最佳答案
WCF 服务明确要求 base64binary 因此您必须仅以 base64binary 发送它。
base64binary 只不过是 base64编码的二进制数据。
1) 使用以下代码片段将图像转换为二进制
NSData *binaryImageData = UIImageJPEGRepresentation([UIImage imageNamed:@"Photo.png"], 0.0);
2) 将二进制编码成base64格式
这是我用来处理 base64 二进制和 UIImage 的类
@interface base64BinaryandImagehandler:NSObject
+(NSString *) base64BinaryStringFromBinaryData: (NSData *)data length: (int)length;
+(UIImage*)Base64BinaryToImage:(NSString *)Base64;
@end
@implementation base64BinaryandImagehandler:NSObject
static char base64EncodingTable[64] = {
'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P',
'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z', 'a', 'b', 'c', 'd', 'e', 'f',
'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v',
'w', 'x', 'y', 'z', '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '+', '/'
};
+(NSString *) base64BinaryStringFromBinaryData: (NSData *)data length: (int)length{
unsigned long ixtext, lentext;
long ctremaining;
unsigned char input[3], output[4];
short i, charsonline = 0, ctcopy;
const unsigned char *raw;
NSMutableString *result;
lentext = [data length];
if (lentext < 1)
return @"";
result = [NSMutableString stringWithCapacity: lentext];
raw = [data bytes];
ixtext = 0;
while (true) {
ctremaining = lentext - ixtext;
if (ctremaining <= 0)
break;
for (i = 0; i < 3; i++) {
unsigned long ix = ixtext + i;
if (ix < lentext)
input[i] = raw[ix];
else
input[i] = 0;
}
output[0] = (input[0] & 0xFC) >> 2;
output[1] = ((input[0] & 0x03) << 4) | ((input[1] & 0xF0) >> 4);
output[2] = ((input[1] & 0x0F) << 2) | ((input[2] & 0xC0) >> 6);
output[3] = input[2] & 0x3F;
ctcopy = 4;
switch (ctremaining) {
case 1:
ctcopy = 2;
break;
case 2:
ctcopy = 3;
break;
}
for (i = 0; i < ctcopy; i++)
[result appendString: [NSString stringWithFormat: @"%c", base64EncodingTable[output[i]]]];
for (i = ctcopy; i < 4; i++)
[result appendString: @"="];
ixtext += 3;
charsonline += 4;
if ((length > 0) && (charsonline >= length))
charsonline = 0;
}
return result;
}
+(UIImage*)Base64BinaryToImage:(NSString *)Base64{
NSString* base64String=[[NSString alloc]initWithFormat:@"data:image/png;base64,%@",Base64];
NSURL *base64url = [NSURL URLWithString:base64String];
NSData *imageData = [NSData dataWithContentsOfURL:base64url];
UIImage *img=[UIImage imageWithData:imageData];
return img;
}
@end
这里是二进制转base64二进制的方法
NSString *Base64Binary=[base64BinaryandImagehandler base64BinaryStringFromBinaryData:binaryImageData length:binaryImageData.length];
您可以找到更多关于 here 的信息还有
这里是C#中的base64编解码部分
public static string Base64Encode(string plainText) {
var plainTextBytes = System.Text.Encoding.UTF8.GetBytes(plainText);
return System.Convert.ToBase64String(plainTextBytes);
}
public static string Base64Decode(string base64EncodedData) {
var base64EncodedBytes = System.Convert.FromBase64String(base64EncodedData);
return System.Text.Encoding.UTF8.GetString(base64EncodedBytes);
}
无论如何你可以忽略 C# 部分,因为你只做 iPhone 部分。
您可以使用下面的链接来测试您编码的 base 64 是否正确。
更新 1这是将确认接收到的字符串是否为 base64 格式的代码
private bool IsBase64String(string str){
try{
// If not exception is caught, then it is a base64 string
MemoryStream stream = new MemoryStream(Convert.FromBase64String(str));
return true;
}
catch{
// If exception is caught, then I assumed it is a normal string
return false;
}
}
例如,下面的代码将在 Images 文件夹下为给定的 base64 数据 和您选择的文件名 创建一个图像 duraiamuthan.jpg 并返回图像的路径,以便您可以在数据库中更新它(它将检查到达的数据是 base64 还是二进制[以防万一 WCF 框架在内部将 base64 转换为二进制])
public string Getpath_CreateImageFromEncodedData(string Base64EncodedData, string fileName)
{
string fileHandle = "";
try
{
if(IsBase64String(Base64EncodedData))
byte[] imageBytes = Convert.FromBase64String(Base64EncodedData);
else
byte[] imageBytes= Base64EncodedData;
MemoryStream ms = new MemoryStream(imageBytes, 0, imageBytes.Length);
ms.Write(imageBytes, 0, imageBytes.Length);
fileHandle = System.Web.HttpContext.Current.Server.MapPath("Images/" + fileName);
FileStream fsObj = new FileStream(fileHandle, FileMode.Create, FileAccess.Write);
ms.WriteTo(fsObj);
fsObj.Close();
ms.Close();
return fileHandle;
}
catch (Exception ex)
{
return "";
}
}
更新 2(针对超时问题)
NSURLRequest 和 NSMutableURLRequest 默认超时时间为 60秒取决于 Internet 连接、Internet 流量网络服务器拥塞,ImageCreating 时间和所有它可能不是enough.so 你可以设置比使用以下值更高的值语法 [urlReqObj setTimeoutInterval:180];
超时间隔是只需几秒钟。
增加 IIS 中的 ConnectionTimeout 默认值是 120 秒,如果请求中有太多请求排队,有时会失败。增加 ConnectionTimeOut 会有所帮助。(此超时也适用于连接的空闲时间)。设置它在 IIS 高级设置中在 IIS 管理器中单击您的 服务 -> 高级设置 -> 连接限制
,您可以在那里设置超时,或者您可以简单地覆盖 web.config
中的连接超时。
打开在 IIS 中保持事件状态
,以便重复使用相同的连接,从而提高 IIS 的效率。在 IIS 管理器中单击您的服务 -> 高级设置 -> 连接限制 -> 启用 HTTP Keep-Alives
如果您打算将图像的二进制文件存储在数据库中。您可以在 connectionstring
中增加数据库 connectiontimeout
或者增加 sqlcommandtimeout
希望这对您有所帮助。有任何疑问欢迎提问
关于ios - 使用 SOAP 使用 WCF 服务将图像从 iPhone 上传到服务器,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29077775/
WCF 服务、WCF RIA 服务和 WCF 数据服务之间有什么区别? 最佳答案 WCF 是一般服务的通信基础设施。 WCF RIA 服务自动生成客户端和服务器代理对象以方便应用程序开发,并依赖 WC
我想在我的 WPF 项目中使用 WCF 服务 (.svc)。, 我正在尝试创建一个服务。但在 Visual Studio 中,我们有“WCF 服务库”和“WCF 服务应用程序”。我两个都试了。 当我们
我正在开发 WCF Web 服务,并使用 WCF 服务应用程序模板来执行此操作。 创建“WCF 服务应用程序”是否满足此要求?与 WCF 服务应用程序相比,创建 WCF 服务库有哪些优势? 最佳答案
我是 WCF 的新手,对 Web 服务进行编码的经验有限。 在工作中,所有面向网络服务的事物都被要求使用 WCF。我需要做的工作涉及查询一个非 WCF Web 服务,该服务显然是用 Java 构建的,
我有一个数据契约(Contract)说用户。它是可序列化的并且可以通过网络传输。我想要一个操作契约(Contract) SaveUser()。我可以将 SaveUser(User user) 作为运营
我一直在开发一个使用 WCF 访问服务器端逻辑和数据库的 WPF 应用程序。 我从一个 WCF 客户端代理对象开始,我反复使用它来调用服务器上的方法。使用代理一段时间后,服务器最终会抛出异常: Sys
不要添加关于不同 WCF 堆栈的另一篇 SO 帖子,但我想在浪费更多开发时间之前确保我朝着正确的方向前进...... 我的场景 - 我们公司有许多 Web 应用程序,它们都访问同一系列的数据库。所有应
我是WCF技术的新手,我想知道RESTful WCF服务和普通WCF服务有什么区别。 RESTful 服务相对于普通 WCF 服务有哪些优势? 谢谢。 最佳答案 REST服务基于HTTP协议(prot
我正在构建的应用程序公开了多个 WCF 服务(A、B)。在内部,它消耗了在我们的内部网络(X、Y)上运行的其他几个 WCF 服务。 使用 WCF 消息日志记录,我希望仅记录我们的服务 A、B 与调用它
我们需要从另一个 WCF 服务调用 WCF 服务。为了测试这一点,我构建了一个示例控制台应用程序来显示一个简单的字符串。设置是: 控制台应用程序 -> WCF 服务 1 -> WCF 服务 2 Con
假设永远不会直接查询数据的情况。 AKA,总会有一些必须发生的过滤逻辑和/或业务逻辑。 什么时候是在 ajax/js 之外使用数据服务的好理由? 请不要访问此页面 http://msdn.micros
我在尝试将所有常规 WCF 调用转换为异步 WCF 调用时遇到问题。我发现我重构了很多代码,但不确定具体该怎么做。我使用了我找到的方法 here但遇到了我需要事情按顺序发生的问题。 private v
我在 IIS 上有一个 WCF 服务,一些 .net Web 应用程序正在使用它。我的任务是编写一个新的 WCF 服务,要求现有的 Web 应用程序可以使用新服务而无需更改它们的 web.config
我正在尝试用外部提供 WSDL 的 WCF 等效服务替换 WSE 服务。 首先,我使用 svcutil 和 wsdl 生成所有服务和客户端类(ATP,我只关心服务实现。)我生成了一个空的 WCF 服务
场景是这样的:有2个WCF Web Services,一个是客户端(WCFClient),一个是服务端(WCFServer),部署在不同的机器上。我需要他们两个之间的证书通信。 在服务器 WCF 上,
我在 Visual Studio 2013 中创建一个 WCF 服务并将其发布到 IIS。我可以在另一个项目中添加服务引用并使用该服务的方法。当我转到 IIS 服务器管理器时,我看到 WCF 激活及其
我是 .net 的新手,对 WCF 知之甚少,如果有任何愚蠢的问题,请耐心等待。我想知道如果我的代码没有显式生成任何线程,WCF 如何处理 SELF-HOST 场景中的同时调用。因此,在阅读了很多关于
我正在为应用程序开发一个面向服务的体系结构,我希望这些服务既可以通过 WCF 公开,也可以通过一个简单的库使用。理想情况下,我想减少重复代码。 从概念上讲,这映射到: Client => WCF Se
我有一个小型测试网络服务来模拟我在现实世界应用程序中注意到的一些奇怪的东西。由于演示显示与应用程序相同的行为,为了简洁起见,我将使用演示。 简而言之,我的服务接口(interface)文件如下所示(您
我首先为我的 WCF 服务启动了我的订阅者,然后继续发布我的发布者的帖子。我的订阅者能够收到帖子。 其次,我关闭了我的第一个订阅者并再次打开它以订阅相同的服务,即所谓的已订阅该服务的第二个订阅者。再一
我是一名优秀的程序员,十分优秀!